Открыть на GitHub

Стратегия Brake Parabolic

Стратегия реализует индикатор Brake Parabolic, который строит параболический барьер выше или ниже цены. При пробое барьера тренд разворачивается, и открывается позиция в сторону пробоя. Алгоритм сопровождает экстремумы кривой, задаваемой параметрами A, B и Shift.

Тестирование показало среднегодовую доходность около 48%. Наилучшие результаты достигаются на трендовых рынках старших таймфреймов.

Система ожидает смены стороны барьера. Буллиш‑переворот закрывает шорт и открывает лонг. Медвежий разворот закрывает лонг и открывает шорт. Во время тренда противоположные позиции закрываются, когда индикатор подтверждает направление.

Детали

  • Условия входа:
    • Лонг: барьер переходит сверху цены вниз.
    • Шорт: барьер переходит снизу цены вверх.
  • Направление: обе стороны.
  • Условия выхода: противоположный сигнал или подтверждение обратного тренда.
  • Стопы: отсутствуют, выход по развороту барьера.
  • Значения по умолчанию:
    • A = 1.5
    • B = 1.0
    • BeginShift = 10
    • CandleType = таймфрейм 4 часа
  • Фильтры:
    • Категория: трендовая
    • Направление: обе
    • Индикаторы: пользовательский
    • Стопы: нет
    • Сложность: средняя
    • Таймфрейм: свинг
    • Сезонность: нет
    • Нейросети: нет
    • Дивергенция: нет
    • Уровень риска: средний
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 crossover strategy.
/// </summary>
public class BrakeParabolicStrategy : Strategy
{
	private readonly StrategyParam<decimal> _sarStep;
	private readonly StrategyParam<decimal> _sarMax;
	private readonly StrategyParam<DataType> _candleType;

	private decimal _prevClose;
	private decimal _prevSar;
	private bool _hasPrev;

	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 BrakeParabolicStrategy()
	{
		_sarStep = Param(nameof(SarStep), 0.02m)
			.SetDisplay("SAR Step", "Acceleration step", "Indicators");
		_sarMax = Param(nameof(SarMax), 0.2m)
			.SetDisplay("SAR Max", "Maximum acceleration", "Indicators");
		_candleType = Param(nameof(CandleType), TimeSpan.FromHours(4).TimeFrame())
			.SetDisplay("Candle Type", "Candle type", "General");
	}

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

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

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

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

		SubscribeCandles(CandleType)
			.Bind(sar, ProcessCandle)
			.Start();
	}

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

		var close = candle.ClosePrice;

		if (!_hasPrev)
		{
			_prevClose = close;
			_prevSar = sarVal;
			_hasPrev = true;
			return;
		}

		var crossUp = _prevClose <= _prevSar && close > sarVal;
		var crossDown = _prevClose >= _prevSar && close < sarVal;

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

		_prevClose = close;
		_prevSar = sarVal;
	}
}