在 GitHub 上查看

Psar Bug 6 Strategy

从 MQL4 脚本 “psar_bug_6” 转换而来。

逻辑

  • 使用可配置加速步长和最大值的抛物线 SAR 指标。
  • 当价格收盘突破 SAR 上方且之前在下方时做多。
  • 当价格收盘跌破 SAR 下方且之前在上方时做空。
  • Reverse 参数可以反转买卖信号。
  • SarClose 选项在 SAR 翻转到另一侧时平掉现有仓位。
  • 固定的止盈和止损距离,以价格单位表示,可选择启用跟踪止损。

参数

  • SarStep – 加速因子步长。
  • SarMax – 最大加速因子。
  • StopLoss – 初始止损距离。
  • TakeProfit – 止盈距离。
  • Trailing – 是否启用跟踪止损。
  • TrailStop – 启用跟踪时的止损距离。
  • SarClose – SAR 反转时是否平仓。
  • Reverse – 是否反转信号。
  • CandleType – 计算所用的蜡烛类型。

说明

策略使用高级 API,通过蜡烛订阅和指标绑定运行,并通过可选的跟踪止损启动保护,出场使用市价单。

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>
/// Parabolic SAR strategy - opens long when price crosses above SAR, short when below.
/// </summary>
public class PsarBug6Strategy : Strategy
{
	private readonly StrategyParam<decimal> _sarStep;
	private readonly StrategyParam<decimal> _sarMax;
	private readonly StrategyParam<DataType> _candleType;

	private decimal _prevSar;
	private decimal _prevClose;
	private bool _initialized;

	public decimal SarStep { get => _sarStep.Value; set => _sarStep.Value = value; }
	public decimal SarMax { get => _sarMax.Value; set => _sarMax.Value = value; }
	public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }

	public PsarBug6Strategy()
	{
		_sarStep = Param(nameof(SarStep), 0.02m)
			.SetDisplay("SAR Step", "Acceleration factor step", "Indicator");

		_sarMax = Param(nameof(SarMax), 0.2m)
			.SetDisplay("SAR Max", "Maximum acceleration factor", "Indicator");

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

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

	protected override void OnReseted()
	{
		base.OnReseted();
		_prevSar = 0;
		_prevClose = 0;
		_initialized = false;
	}

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

		var psar = new ParabolicSar
		{
			AccelerationStep = SarStep,
			AccelerationMax = SarMax
		};

		var subscription = SubscribeCandles(CandleType);
		subscription.Bind(psar, ProcessCandle).Start();
	}

	private void ProcessCandle(ICandleMessage candle, decimal sar)
	{
		if (candle.State != CandleStates.Finished)
			return;

		if (!_initialized)
		{
			_prevSar = sar;
			_prevClose = candle.ClosePrice;
			_initialized = true;
			return;
		}

		var close = candle.ClosePrice;
		var crossUp = close > sar && _prevClose <= _prevSar;
		var crossDown = close < sar && _prevClose >= _prevSar;

		if (crossUp && Position <= 0)
		{
			if (Position < 0) BuyMarket();
			BuyMarket();
		}
		else if (crossDown && Position >= 0)
		{
			if (Position > 0) SellMarket();
			SellMarket();
		}

		_prevSar = sar;
		_prevClose = close;
	}
}