在 GitHub 上查看

多时间框架 MACD

多时间框架 MACD 将当前时间框架和更高时间框架的 MACD 信号结合。只有当两个时间框架在均线交叉或零线穿越上达成一致时才入场。

详情

  • 数据:两个时间框架的价格蜡烛。
  • 入场条件
    • 做多:基于 Entry 参数,默认是两个时间框架同时出现看涨交叉。
    • 做空:与做多相反。
  • 出场条件:反向信号或移动止损。
  • 止损:可选的移动止损。
  • 默认值
    • FastLength = 12
    • SlowLength = 26
    • SignalLength = 9
    • CandleType = tf(5)
    • HigherCandleType = tf(1d)
    • ShowCurrentTimeframe = true
    • ShowHigherTimeframe = true
    • Entry = Crossover
    • UseTrailingStop = false
    • TrailingStopPercent = 2
  • 过滤器
    • 类别:趋势
    • 方向:多头和空头
    • 指标:MACD
    • 止损:是
    • 复杂度:中等
    • 时间框架:多时间框架 (5m/1d)
    • 季节性:否
    • 神经网络:否
    • 背离:否
    • 风险等级:中等
using System;
using System.Linq;
using System.Collections.Generic;

using Ecng.Common;
using Ecng.Collections;
using Ecng.Serialization;

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

namespace StockSharp.Samples.Strategies;

/// <summary>
/// MACD crossover strategy with EMA trend filter.
/// </summary>
public class MultiTimeframeMacdStrategy : Strategy
{
	private readonly StrategyParam<int> _fastLength;
	private readonly StrategyParam<int> _slowLength;
	private readonly StrategyParam<int> _trendLength;
	private readonly StrategyParam<DataType> _candleType;

	public int FastLength { get => _fastLength.Value; set => _fastLength.Value = value; }
	public int SlowLength { get => _slowLength.Value; set => _slowLength.Value = value; }
	public int TrendLength { get => _trendLength.Value; set => _trendLength.Value = value; }
	public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }

	public MultiTimeframeMacdStrategy()
	{
		_fastLength = Param(nameof(FastLength), 12)
			.SetGreaterThanZero()
			.SetDisplay("Fast", "Fast EMA", "MACD");
		_slowLength = Param(nameof(SlowLength), 26)
			.SetGreaterThanZero()
			.SetDisplay("Slow", "Slow EMA", "MACD");
		_trendLength = Param(nameof(TrendLength), 50)
			.SetGreaterThanZero()
			.SetDisplay("Trend", "Trend EMA period", "Trend");
		_candleType = Param(nameof(CandleType), TimeSpan.FromHours(1).TimeFrame())
			.SetDisplay("Candle Type", "Type of candles", "General");
	}

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

		var fastEma = new ExponentialMovingAverage { Length = FastLength };
		var slowEma = new ExponentialMovingAverage { Length = SlowLength };
		var trendEma = new ExponentialMovingAverage { Length = TrendLength };

		var prevFast = 0m;
		var prevSlow = 0m;
		var initialized = false;

		var subscription = SubscribeCandles(CandleType);

		subscription
			.Bind(fastEma, slowEma, trendEma, (candle, fastVal, slowVal, trendVal) =>
			{
				if (candle.State != CandleStates.Finished)
					return;

				if (!fastEma.IsFormed || !slowEma.IsFormed || !trendEma.IsFormed)
					return;

				if (!initialized)
				{
					prevFast = fastVal;
					prevSlow = slowVal;
					initialized = true;
					return;
				}

				var macd = fastVal - slowVal;
				var prevMacd = prevFast - prevSlow;

				// MACD crosses zero up + price above trend => buy
				if (prevMacd <= 0 && macd > 0 && candle.ClosePrice > trendVal && Position <= 0)
					BuyMarket();
				// MACD crosses zero down + price below trend => sell
				else if (prevMacd >= 0 && macd < 0 && candle.ClosePrice < trendVal && Position > 0)
					SellMarket();

				prevFast = fastVal;
				prevSlow = slowVal;
			})
			.Start();

		var area = CreateChartArea();
		if (area != null)
		{
			DrawCandles(area, subscription);
			DrawIndicator(area, trendEma);
			DrawOwnTrades(area);
		}
	}
}