在 GitHub 上查看

夜间剥头皮策略

该策略在晚上时段利用布林带交易。只有在达到设定的开始时间后,且带宽较窄并且价格突破带外时才开仓。

细节

  • 入场条件
    • 多头:在 Start Hour 之后,收盘价低于下轨且带宽小于 Range Threshold
    • 空头:在 Start Hour 之后,收盘价高于上轨且带宽小于 Range Threshold
  • 多/空:双向。
  • 出场条件
    • 若时间回到次日 Start Hour 之前,平掉现有仓位。
    • 通过 StartProtection 管理止损与止盈。
  • 止损:使用 StartProtection 固定止损和止盈。
  • 默认值
    • BB Period = 40
    • BB Deviation = 1
    • Range Threshold = 450
    • Stop Loss = 370
    • Take Profit = 20
    • Start Hour = 19
    • Candle Type = 1h
  • 过滤条件
    • 类别:均值回归
    • 方向:双向
    • 指标:布林带
    • 止损:是
    • 复杂度:低
    • 时间框架:短期
using System;
using System.Collections.Generic;

using Ecng.Common;

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

namespace StockSharp.Samples.Strategies;

/// <summary>
/// Night scalping strategy using Bollinger Bands.
/// </summary>
public class NightScalperStrategy : Strategy
{
	private const int BufferSize = 128;

	private readonly StrategyParam<int> _bollingerPeriod;
	private readonly StrategyParam<decimal> _bollingerDeviation;
	private readonly StrategyParam<decimal> _rangeThreshold;
	private readonly StrategyParam<DataType> _candleType;

	private readonly decimal[] _closes = new decimal[BufferSize];
	private int _closeIndex;
	private int _closeCount;

	public int BollingerPeriod
	{
		get => _bollingerPeriod.Value;
		set => _bollingerPeriod.Value = value;
	}

	public decimal BollingerDeviation
	{
		get => _bollingerDeviation.Value;
		set => _bollingerDeviation.Value = value;
	}

	public decimal RangeThreshold
	{
		get => _rangeThreshold.Value;
		set => _rangeThreshold.Value = value;
	}

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

	public NightScalperStrategy()
	{
		_bollingerPeriod = Param(nameof(BollingerPeriod), 20)
			.SetDisplay("BB Period", "Bollinger period", "Indicators");

		_bollingerDeviation = Param(nameof(BollingerDeviation), 2.0m)
			.SetDisplay("BB Deviation", "Bollinger deviation", "Indicators");

		_rangeThreshold = Param(nameof(RangeThreshold), 3000m)
			.SetDisplay("Range Threshold", "Maximum band width", "Risk");

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

	/// <inheritdoc />
	public override IEnumerable<(Security sec, DataType dt)> GetWorkingSecurities()
	{
		return [(Security, CandleType)];
	}

	/// <inheritdoc />
	protected override void OnReseted()
	{
		base.OnReseted();

		Array.Clear(_closes);
		_closeIndex = 0;
		_closeCount = 0;
	}

	/// <inheritdoc />
	protected override void OnStarted2(DateTime time)
	{
		base.OnStarted2(time);
		Array.Clear(_closes);
		_closeIndex = 0;
		_closeCount = 0;

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

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

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

		PushClose(candle.ClosePrice);

		if (_closeCount < BollingerPeriod)
			return;

		var mean = GetAverage(BollingerPeriod);
		var deviation = GetStandardDeviation(BollingerPeriod, mean);
		var upper = mean + (deviation * BollingerDeviation);
		var lower = mean - (deviation * BollingerDeviation);
		var width = upper - lower;

		if (!IsFormedAndOnlineAndAllowTrading())
			return;

		if (Position == 0 && width <= RangeThreshold)
		{
			if (candle.LowPrice <= lower)
				BuyMarket();
			else if (candle.HighPrice >= upper)
				SellMarket();
		}
		else if (Position > 0 && candle.ClosePrice >= mean)
		{
			SellMarket();
		}
		else if (Position < 0 && candle.ClosePrice <= mean)
		{
			BuyMarket();
		}
	}

	private void PushClose(decimal close)
	{
		_closes[_closeIndex] = close;
		_closeIndex = (_closeIndex + 1) % BufferSize;

		if (_closeCount < BufferSize)
			_closeCount++;
	}

	private decimal GetAverage(int period)
	{
		var count = Math.Min(period, _closeCount);
		var sum = 0m;

		for (var i = 0; i < count; i++)
		{
			var idx = (_closeIndex - 1 - i + BufferSize) % BufferSize;
			sum += _closes[idx];
		}

		return sum / count;
	}

	private decimal GetStandardDeviation(int period, decimal mean)
	{
		var count = Math.Min(period, _closeCount);
		var sum = 0m;

		for (var i = 0; i < count; i++)
		{
			var idx = (_closeIndex - 1 - i + BufferSize) % BufferSize;
			var diff = _closes[idx] - mean;
			sum += diff * diff;
		}

		return (decimal)Math.Sqrt((double)(sum / count));
	}
}