在 GitHub 上查看

优化的 Heikin-Ashi 买卖可选策略

Heikin-Ashi 蜡烛能够平滑价格并突出趋势方向。本策略一次只交易一个方向:在绿色蜡烛买入或在红色蜡烛卖出,并可限定交易日期范围。可选的止损和止盈参数用于风险控制。

细节

  • 入场条件:Heikin-Ashi 蜡烛颜色变化。
  • 多空方向:可配置。
  • 离场条件:反向信号或止损止盈。
  • 止损止盈:可选,百分比。
  • 默认值
    • CandleType = 1 天
    • StartDate = 2023-01-01
    • EndDate = 2024-01-01
    • TradeType = BuyOnly
    • UseStopLoss = true
    • StopLossPercent = 2
    • UseTakeProfit = true
    • TakeProfitPercent = 4
  • 筛选条件
    • 类别: 趋势
    • 方向: 可配置
    • 指标: Heikin-Ashi
    • 止损: 可选
    • 复杂度: 基础
    • 时间框架: 日线
    • 季节性: 日期范围
    • 神经网络: 无
    • 背离: 无
    • 风险等级: 中等
using System;

using Ecng.Common;

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

namespace StockSharp.Samples.Strategies;

/// <summary>
/// Heikin-Ashi strategy with EMA trend filter.
/// </summary>
public class OptimizedHeikinAshiBuySellStrategy : Strategy
{
	private readonly StrategyParam<DataType> _candleType;
	private readonly StrategyParam<int> _emaLength;

	private decimal _prevHaOpen;
	private decimal _prevHaClose;
	private bool _haInit;
	private bool _prevBullish;
	private bool _prevBearish;

	public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }
	public int EmaLength { get => _emaLength.Value; set => _emaLength.Value = value; }

	public OptimizedHeikinAshiBuySellStrategy()
	{
		_candleType = Param(nameof(CandleType), TimeSpan.FromMinutes(5).TimeFrame());
		_emaLength = Param(nameof(EmaLength), 50).SetGreaterThanZero();
	}

	/// <inheritdoc />
	protected override void OnReseted()
	{
		base.OnReseted();
		_prevHaOpen = 0;
		_prevHaClose = 0;
		_haInit = false;
		_prevBullish = false;
		_prevBearish = false;
	}

	/// <inheritdoc />
	protected override void OnStarted2(DateTime time)
	{
		base.OnStarted2(time);

		_prevHaOpen = 0;
		_prevHaClose = 0;
		_haInit = false;
		_prevBullish = false;
		_prevBearish = false;

		var ema = new ExponentialMovingAverage { Length = EmaLength };
		var lastSignal = DateTimeOffset.MinValue;
		var cooldown = TimeSpan.FromMinutes(600);

		var subscription = SubscribeCandles(CandleType);
		subscription
			.Bind(ema, (candle, emaVal) =>
			{
				if (candle.State != CandleStates.Finished)
					return;

				if (!ema.IsFormed)
					return;

				decimal haOpen;
				decimal haClose;

				if (!_haInit)
				{
					haOpen = (candle.OpenPrice + candle.ClosePrice) / 2m;
					haClose = (candle.OpenPrice + candle.HighPrice + candle.LowPrice + candle.ClosePrice) / 4m;
					_prevHaOpen = haOpen;
					_prevHaClose = haClose;
					_haInit = true;
					_prevBullish = haClose > haOpen;
					_prevBearish = haClose < haOpen;
					return;
				}

				haOpen = (_prevHaOpen + _prevHaClose) / 2m;
				haClose = (candle.OpenPrice + candle.HighPrice + candle.LowPrice + candle.ClosePrice) / 4m;

				var isBullish = haClose > haOpen;
				var isBearish = haClose < haOpen;

				if (candle.OpenTime - lastSignal >= cooldown)
				{
					// Transition from bearish to bullish + above EMA
					if (!_prevBullish && isBullish && candle.ClosePrice > emaVal && Position <= 0)
					{
						BuyMarket();
						lastSignal = candle.OpenTime;
					}
					// Transition from bullish to bearish + below EMA
					else if (!_prevBearish && isBearish && candle.ClosePrice < emaVal && Position >= 0)
					{
						SellMarket();
						lastSignal = candle.OpenTime;
					}
				}

				_prevHaOpen = haOpen;
				_prevHaClose = haClose;
				_prevBullish = isBullish;
				_prevBearish = isBearish;
			})
			.Start();

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