在 GitHub 上查看

NQ Phantom Scalper Pro 策略

基于 VWAP 带的突破策略,可选成交量和趋势过滤。

详情

  • 入场条件
    • 多头:价格收于上方 VWAP 带之上且成交量放大。
    • 空头:价格收于下方 VWAP 带之下且成交量放大。
  • 多空:双向
  • 出场条件
    • 价格回到 VWAP 或触发 ATR 止损。
  • 止损:基于 ATR
  • 默认值
    • Band #1 Mult = 1.0
    • Band #2 Mult = 2.0
    • ATR Length = 14
    • ATR Stop Mult = 1.0
    • Volume SMA Period = 20
    • Volume Spike Mult = 1.5
    • Trend EMA Length = 50
  • 过滤器
    • 分类:趋势跟随
    • 方向:双向
    • 指标:VWAP、ATR、EMA、SMA
    • 止损:有
    • 复杂度:中等
    • 时间框架:短期
    • 季节性:无
    • 神经网络:无
    • 背离:无
    • 风险等级:中等
using System;

using Ecng.Common;

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

namespace StockSharp.Samples.Strategies;

/// <summary>
/// NQ Phantom Scalper Pro strategy based on VWAP bands with optional volume and trend filters.
/// </summary>
public class NqPhantomScalperProStrategy : Strategy
{
	private readonly StrategyParam<decimal> _band1Mult;
	private readonly StrategyParam<decimal> _atrStopMult;
	private readonly StrategyParam<bool> _useTrendFilter;
	private readonly StrategyParam<int> _trendLength;
	private readonly StrategyParam<DataType> _candleType;

	private decimal _entryPrice;
	private decimal _stopPrice;
	private DateTimeOffset _lastSignal = DateTimeOffset.MinValue;

	public decimal Band1Mult
	{
		get => _band1Mult.Value;
		set => _band1Mult.Value = value;
	}

	public decimal AtrStopMult
	{
		get => _atrStopMult.Value;
		set => _atrStopMult.Value = value;
	}

	public bool UseTrendFilter
	{
		get => _useTrendFilter.Value;
		set => _useTrendFilter.Value = value;
	}

	public int TrendLength
	{
		get => _trendLength.Value;
		set => _trendLength.Value = value;
	}

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

	public NqPhantomScalperProStrategy()
	{
		_band1Mult = Param(nameof(Band1Mult), 1.0m)
			.SetGreaterThanZero();

		_atrStopMult = Param(nameof(AtrStopMult), 1.0m)
			.SetGreaterThanZero();

		_useTrendFilter = Param(nameof(UseTrendFilter), false);

		_trendLength = Param(nameof(TrendLength), 50)
			.SetGreaterThanZero();

		_candleType = Param(nameof(CandleType), TimeSpan.FromMinutes(5).TimeFrame());
	}

	/// <inheritdoc />
	protected override void OnReseted()
	{
		base.OnReseted();
		_entryPrice = 0m;
		_stopPrice = 0m;
		_lastSignal = DateTimeOffset.MinValue;
	}

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

		_entryPrice = 0m;
		_stopPrice = 0m;
		_lastSignal = DateTimeOffset.MinValue;

		var vwap = new VolumeWeightedMovingAverage();
		var atr = new AverageTrueRange { Length = 14 };
		var std = new StandardDeviation { Length = 20 };
		var trendEma = new ExponentialMovingAverage { Length = TrendLength };

		var subscription = SubscribeCandles(CandleType);

		subscription
			.BindEx(vwap, atr, std, trendEma, ProcessCandle)
			.Start();

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

	private void ProcessCandle(ICandleMessage candle, IIndicatorValue vwapValue, IIndicatorValue atrValue, IIndicatorValue stdValue, IIndicatorValue trendValue)
	{
		if (candle.State != CandleStates.Finished)
			return;

		if (!vwapValue.IsFinal || !vwapValue.IsFormed || !atrValue.IsFormed || !stdValue.IsFormed)
			return;

		var vwapVal = vwapValue.ToDecimal();
		var atr = atrValue.ToDecimal();
		var stdVal = stdValue.ToDecimal();
		var trendVal = trendValue.ToDecimal();

		if (atr <= 0 || stdVal <= 0)
			return;

		var upper1 = vwapVal + stdVal * Band1Mult;
		var lower1 = vwapVal - stdVal * Band1Mult;

		var trendOkLong = !UseTrendFilter || candle.ClosePrice > trendVal;
		var trendOkShort = !UseTrendFilter || candle.ClosePrice < trendVal;

		var cooldown = TimeSpan.FromMinutes(360);

		if (candle.OpenTime - _lastSignal < cooldown)
			return;

		// Exit logic - only on stop loss
		if (Position > 0 && _stopPrice > 0 && candle.ClosePrice <= _stopPrice)
		{
			SellMarket();
			_stopPrice = 0m;
			_lastSignal = candle.OpenTime;
			return;
		}

		if (Position < 0 && _stopPrice > 0 && candle.ClosePrice >= _stopPrice)
		{
			BuyMarket();
			_stopPrice = 0m;
			_lastSignal = candle.OpenTime;
			return;
		}

		// Entry logic
		if (Position <= 0 && candle.ClosePrice > upper1 && trendOkLong)
		{
			BuyMarket();
			_entryPrice = candle.ClosePrice;
			_stopPrice = _entryPrice - atr * AtrStopMult * 3;
			_lastSignal = candle.OpenTime;
		}
		else if (Position >= 0 && candle.ClosePrice < lower1 && trendOkShort)
		{
			SellMarket();
			_entryPrice = candle.ClosePrice;
			_stopPrice = _entryPrice + atr * AtrStopMult * 3;
			_lastSignal = candle.OpenTime;
		}
	}
}