在 GitHub 上查看

MACD Sample 策略

该策略复刻了 MetaTrader 的经典 MACD Sample EA。 它使用 MACD 线与信号线的交叉并结合 EMA 趋势过滤,分别设定多头和空头的止盈止损,同时可选用追踪止损。交易仅在设定的时间窗口内进行。

细节

  • 入场条件
    • 做多:MACD 线在零线下方并向上穿越信号线,同时 EMA 上升。
    • 做空:MACD 线在零线上方并向下穿越信号线,同时 EMA 下降。
  • 出场条件
    • 反向的 MACD 交叉。
    • 达到各自的止盈或止损。
    • 触发追踪止损。
  • 多空方向:均可。
  • 默认参数
    • EMA Period = 26
    • MACD Open Level = 3
    • MACD Close Level = 2
    • Take Profit Long = 50
    • Take Profit Short = 75
    • Stop Loss Long = 80
    • Stop Loss Short = 50
    • Trailing Stop = 30
    • 交易时间:UTC 4 点至 19 点
  • 指标:MACD, EMA
  • 时间框架:默认 1 小时 K 线
using System;
using System.Collections.Generic;

using Ecng.Common;

using StockSharp.Algo.Indicators;
using StockSharp.Algo.Strategies;
using StockSharp.BusinessEntities;
using StockSharp.Messages;

namespace StockSharp.Samples.Strategies;

/// <summary>
/// MACD sample strategy with EMA trend filter.
/// Buys on MACD crossover up when above EMA, sells on crossover down when below EMA.
/// </summary>
public class MacdSampleStrategy : Strategy
{
	private readonly StrategyParam<DataType> _candleType;
	private readonly StrategyParam<int> _maTrendPeriod;

	private decimal _prevMacd;
	private decimal _prevSignal;
	private bool _hasPrev;
	private decimal _emaValue;
	private bool _hasEma;

	public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }
	public int MaTrendPeriod { get => _maTrendPeriod.Value; set => _maTrendPeriod.Value = value; }

	public MacdSampleStrategy()
	{
		_candleType = Param(nameof(CandleType), TimeSpan.FromHours(4).TimeFrame())
			.SetDisplay("Candle Type", "Type of candles", "General");

		_maTrendPeriod = Param(nameof(MaTrendPeriod), 50)
			.SetGreaterThanZero()
			.SetDisplay("EMA Period", "EMA trend filter period", "Indicators");
	}

	public override IEnumerable<(Security sec, DataType dt)> GetWorkingSecurities()
		=> [(Security, CandleType)];

	protected override void OnReseted()
	{
		base.OnReseted();
		_prevMacd = 0;
		_prevSignal = 0;
		_hasPrev = false;
		_emaValue = 0;
		_hasEma = false;
	}

	protected override void OnStarted2(DateTime time)
	{
		base.OnStarted2(time);

		var macdSignal = new MovingAverageConvergenceDivergenceSignal();
		var ema = new ExponentialMovingAverage { Length = MaTrendPeriod };

		var subscription = SubscribeCandles(CandleType);
		subscription
			.Bind(ema, ProcessEma);
		subscription
			.BindEx(macdSignal, ProcessMacd)
			.Start();
	}

	private void ProcessEma(ICandleMessage candle, decimal emaVal)
	{
		if (candle.State != CandleStates.Finished)
			return;
		_emaValue = emaVal;
		_hasEma = true;
	}

	private void ProcessMacd(ICandleMessage candle, IIndicatorValue macdValue)
	{
		if (candle.State != CandleStates.Finished)
			return;

		if (!_hasEma)
			return;

		var macdTyped = (MovingAverageConvergenceDivergenceSignalValue)macdValue;
		if (macdTyped.Macd is not decimal macd || macdTyped.Signal is not decimal signal)
			return;

		if (!_hasPrev)
		{
			_prevMacd = macd;
			_prevSignal = signal;
			_hasPrev = true;
			return;
		}

		var close = candle.ClosePrice;

		// Buy: MACD crosses above signal, price above EMA
		if (_prevMacd <= _prevSignal && macd > signal && close > _emaValue)
		{
			if (Position < 0)
				BuyMarket();
			if (Position <= 0)
				BuyMarket();
		}
		// Sell: MACD crosses below signal, price below EMA
		else if (_prevMacd >= _prevSignal && macd < signal && close < _emaValue)
		{
			if (Position > 0)
				SellMarket();
			if (Position >= 0)
				SellMarket();
		}

		_prevMacd = macd;
		_prevSignal = signal;
	}
}