Открыть на GitHub

Стратегия Simple MACD EA

Общее описание

Simple MACD EA — это перенос классического советника MetaTrader «Simple MACD EA». Стратегия использует две экспоненциальные скользящие средние (EMA), чтобы воссоздать гистограмму MACD и определить доминирующее направление на минутных свечах. Длинная позиция открывается, когда EMA с периодом 100 поднимается выше медленной EMA с настраиваемым периодом. Короткая позиция открывается, когда быстрая EMA опускается ниже медленной. В рынке всегда присутствует только одна позиция.

Логика управления сделками

  • Определение тренда. Разность между EMA(100) и настраиваемой EMA задаёт текущий тренд (+1, 0, -1). Переход значения из отрицательного в положительное приводит к открытию покупки. Обратный переход инициирует продажу.
  • Подтверждение импульса. Стратегия отслеживает разницу между основной EMA и более медленной EMA с периодом MACD level + 1. Если эта разница сокращается против открытой позиции после движения цены не менее чем на пять пунктов в прибыль, позиция закрывается заранее.
  • Временная защита. После того как сделка удерживается заданное число циклов, активируется «мягкий» стоп, снижающий допустимое неблагоприятное движение относительно цены входа.
  • Трейлинг-выход. Когда сделка уходит в прибыль и остаётся открытой достаточное число циклов, включается внутренний трейлинг-стоп. Уровень стопа следует за ценой на заданное количество пунктов и может быть обновлён ограниченное число раз. После исчерпания лимита позиция закрывается.
  • Выход по смене тренда. Если трендовый сигнал разворачивается в противоположную сторону, а цена уже находится минимум на пять пунктов в прибыли, позиция закрывается немедленно.

Параметры

  • Тип свечей — таймфрейм, по которому рассчитываются EMA (по умолчанию 1 минута).
  • Объём — объём заявки для новых входов.
  • MACD Level — период медленной EMA. Дополнительно автоматически создаётся EMA с периодом MACD Level + 1.
  • Trailing Stop — расстояние трейлинг-стопа в пунктах. Ноль отключает функцию.
  • Trailing Updates — максимальное число обновлений трейлинг-стопа для одной сделки.
  • Wait Cycles — количество циклов ожидания перед активацией мягкого стопа.

Дополнительные замечания

  • Перед сменой направления стратегия всегда закрывает текущую позицию.
  • Для перевода пунктов в цены используется шаг цены выбранного инструмента.
  • Реализация построена на высокоуровневом API подписки на свечи StockSharp и не создаёт собственных буферов индикаторов.
using System;
using System.Collections.Generic;

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

namespace StockSharp.Samples.Strategies;

/// <summary>
/// Simple MACD EA strategy using fast/slow EMA difference for trend detection.
/// Buy when EMA difference crosses above zero, sell when it crosses below zero.
/// </summary>
public class SimpleMacdEaStrategy : Strategy
{
	private readonly StrategyParam<int> _fastEmaPeriod;
	private readonly StrategyParam<int> _slowEmaPeriod;
	private readonly StrategyParam<DataType> _candleType;

	private decimal _prevDiff;
	private bool _hasPrev;

	public int FastEmaPeriod
	{
		get => _fastEmaPeriod.Value;
		set => _fastEmaPeriod.Value = value;
	}

	public int SlowEmaPeriod
	{
		get => _slowEmaPeriod.Value;
		set => _slowEmaPeriod.Value = value;
	}

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

	public SimpleMacdEaStrategy()
	{
		_fastEmaPeriod = Param(nameof(FastEmaPeriod), 12)
			.SetDisplay("Fast EMA", "Fast EMA period", "Indicators");

		_slowEmaPeriod = Param(nameof(SlowEmaPeriod), 26)
			.SetDisplay("Slow EMA", "Slow EMA period", "Indicators");

		_candleType = Param(nameof(CandleType), TimeSpan.FromHours(1).TimeFrame())
			.SetDisplay("Candle Type", "Candle timeframe", "General");
	}

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

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

		_prevDiff = 0m;
		_hasPrev = false;
	}

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

		_hasPrev = false;

		var fastEma = new ExponentialMovingAverage { Length = FastEmaPeriod };
		var slowEma = new ExponentialMovingAverage { Length = SlowEmaPeriod };

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

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

		var diff = fast - slow;

		if (!_hasPrev)
		{
			_prevDiff = diff;
			_hasPrev = true;
			return;
		}

		// Buy: MACD crosses above zero
		var longSignal = _prevDiff <= 0 && diff > 0;
		// Sell: MACD crosses below zero
		var shortSignal = _prevDiff >= 0 && diff < 0;

		if (Position <= 0 && longSignal)
		{
			if (Position < 0)
				BuyMarket();
			BuyMarket();
		}
		else if (Position >= 0 && shortSignal)
		{
			if (Position > 0)
				SellMarket();
			SellMarket();
		}

		_prevDiff = diff;
	}
}