Открыть на GitHub

Стратегия KPrmSt Cross

Обзор

Стратегия KPrmSt Cross является портом эксперта MetaTrader 5 exp_kprmst.mq5. В ней используется осциллятор типа Stochastic (KPrmSt) для поиска разворотов при пересечении основной и сигнальной линий.

Стратегия подписывается на свечи выбранного таймфрейма и рассчитывает индикатор Stochastic (используется как приближённая реализация KPrmSt). Когда линия %K опускается ниже линии %D, открывается длинная позиция, а при обратном пересечении — короткая. При наличии противоположных позиций происходит их реверс.

Параметры

  • Candle Type – таймфрейм свечей.
  • K Period – число баров для основной линии.
  • D Period – период сглаживания сигнальной линии.
  • Slowing – дополнительное сглаживание %K.
  • Stop Loss – защитный стоп в единицах цены. 0 отключает.
  • Take Profit – фиксированный профит в единицах цены. 0 отключает.

Логика торговли

  1. Обрабатываются только завершённые свечи.
  2. Значения осциллятора сохраняются для поиска пересечений.
  3. Если %K переходит ниже %D после нахождения выше неё, открывается покупка или закрывается продажа.
  4. Если %K переходит выше %D после нахождения ниже неё, открывается продажа или закрывается покупка.
  5. Стоп‑лосс и тейк‑профит (при задании) закрывают позицию при достижении уровней.

Примечания

  • Индикатор KPrmSt из оригинала приближается стандартным индикатором Stochastic StockSharp.
  • Опции управления капиталом оригинала не реализованы.
  • Для работы стратегии требуется поток рыночных данных и доступ к торговому шлюзу, поддерживаемому StockSharp.
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>
/// KPrmSt cross strategy based on the stochastic oscillator.
/// Opens long when %K crosses above %D from below.
/// Opens short when %K crosses below %D from above.
/// </summary>
public class KprmStCrossStrategy : Strategy
{
	private readonly StrategyParam<DataType> _candleType;
	private readonly StrategyParam<decimal> _stopLossPct;
	private readonly StrategyParam<decimal> _takeProfitPct;

	private StochasticOscillator _stochastic;
	private decimal? _prevK;
	private decimal? _prevD;

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

	public decimal StopLossPct
	{
		get => _stopLossPct.Value;
		set => _stopLossPct.Value = value;
	}

	public decimal TakeProfitPct
	{
		get => _takeProfitPct.Value;
		set => _takeProfitPct.Value = value;
	}

	public KprmStCrossStrategy()
	{
		_candleType = Param(nameof(CandleType), TimeSpan.FromHours(4).TimeFrame())
			.SetDisplay("Candle Type", "Time frame for indicator calculation", "General");

		_stopLossPct = Param(nameof(StopLossPct), 2m)
			.SetDisplay("Stop Loss %", "Stop loss percentage", "Risk");

		_takeProfitPct = Param(nameof(TakeProfitPct), 3m)
			.SetDisplay("Take Profit %", "Take profit percentage", "Risk");
	}

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

	/// <inheritdoc />
	protected override void OnReseted()
	{
		base.OnReseted();
		_stochastic = default;
		_prevK = null;
		_prevD = null;
	}

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

		_stochastic = new StochasticOscillator();

		Indicators.Add(_stochastic);

		var subscription = SubscribeCandles(CandleType);

		subscription.Bind(ProcessCandle).Start();

		StartProtection(
			takeProfit: new Unit(TakeProfitPct, UnitTypes.Percent),
			stopLoss: new Unit(StopLossPct, UnitTypes.Percent),
			useMarketOrders: true);

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

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

		var stochResult = _stochastic.Process(candle);
		if (!stochResult.IsFormed)
			return;

		if (!IsFormedAndOnlineAndAllowTrading())
			return;

		var stochVal = (StochasticOscillatorValue)stochResult;
		if (stochVal.K is not decimal k || stochVal.D is not decimal d)
			return;

		if (_prevK.HasValue && _prevD.HasValue)
		{
			var wasBelow = _prevK.Value < _prevD.Value;
			var isAbove = k > d;

			// K crosses above D -> buy
			if (wasBelow && isAbove && Position <= 0)
			{
				if (Position < 0) BuyMarket();
				BuyMarket();
			}
			// K crosses below D -> sell
			else if (!wasBelow && !isAbove && _prevK.Value > _prevD.Value && Position >= 0)
			{
				if (Position > 0) SellMarket();
				SellMarket();
			}
		}

		_prevK = k;
		_prevD = d;
	}
}