Fortrader 10 Pips — порт советника MetaTrader 4 10pips.mq4 (ID 8074) на платформу StockSharp. Робот одновременно держит одну длинную и одну короткую позицию. Для каждой стороны используются фиксированные уровни тейк-профита, стоп-лосса и трейлинг-стопа, заданные в пунктах инструмента.
Реализация повторяет хеджирующее поведение исходного эксперта с помощью высокоуровневого API StockSharp. Сразу после запуска стратегия отправляет рыночные заявки на покупку и продажу. Если защитный ордер закрывает одну из ног, стратегия мгновенно открывает новую позицию в том же направлении, сохраняя две противоположные позиции в рынке.
Параметры
Название
Описание
Take Profit Buy
Дистанция тейк-профита для длинной позиции (в пунктах).
Stop Loss Buy
Дистанция стоп-лосса для длинной позиции (в пунктах).
Trailing Stop Buy
Дистанция трейлинг-стопа для длинной позиции (в пунктах). Значение 0 отключает трейлинг.
Take Profit Sell
Дистанция тейк-профита для короткой позиции (в пунктах).
Stop Loss Sell
Дистанция стоп-лосса для короткой позиции (в пунктах).
Trailing Stop Sell
Дистанция трейлинг-стопа для короткой позиции (в пунктах). Значение 0 отключает трейлинг.
Volume
Объём каждой рыночной заявки в лотах.
Все расстояния умножаются на PriceStep инструмента для перевода пунктов в абсолютную цену. Каждый параметр реализован через StrategyParam<T>, поэтому его можно менять и оптимизировать в интерфейсе.
Логика торговли
Старт — метод OnStarted подписывается на Level 1, чтобы отслеживать лучшие цены Bid/Ask, и сразу отправляет рыночные заявки на покупку и продажу.
Защитные ордера — после каждой сделки (OnNewMyTrade) выставляются соответствующие ордера стоп-лосс и тейк-профит, если их дистанция больше нуля. Цены округляются к ближайшему шагу цены.
Повторный вход — при срабатывании стоп-лосса или тейк-профита стратегия немедленно переоткрывает позицию в том же направлении, поддерживая двухстороннее присутствие в рынке.
Трейлинг-стоп — при обновлении котировок вызывается UpdateTrailingStops. Если прибыль превысила заданную дистанцию от цены входа, стоп-лосс сдвигается по направлению движения, как в оригинальном советнике.
Особенности реализации
В оригинальном MT4 коде между первой покупкой и продажей была пауза 10 секунд. В StockSharp задержка не требуется, поэтому обе сделки размещаются сразу.
Поскольку по умолчанию StockSharp работает с нетто-позицией, фактическая возможность держать встречные позиции зависит от брокера/коннектора. Стратегия самостоятельно отслеживает каждую ногу и переоткрывает её при закрытии.
Метод StartProtection() вызывается в OnStarted, чтобы активировать глобальную защиту портфеля, если она настроена в платформе.
Рекомендации по использованию
Убедитесь, что используемый коннектор поддерживает одновременные длинные и короткие позиции, если требуется хеджирование.
Установите значение трейлинг-стопа в 0, чтобы отключить его для выбранного направления.
Оптимизируйте параметры тейк-профита, стоп-лосса и трейлинг-стопа на исторических данных под свой инструмент и таймфрейм.
using System;
using StockSharp.Algo.Indicators;
using StockSharp.Algo.Strategies;
using StockSharp.BusinessEntities;
using StockSharp.Messages;
namespace StockSharp.Samples.Strategies;
public class Fortrader10PipsStrategy : Strategy
{
private readonly StrategyParam<int> _fastPeriod;
private readonly StrategyParam<int> _slowPeriod;
private readonly StrategyParam<DataType> _candleType;
private decimal _prevFast; private decimal _prevSlow; private bool _hasPrev;
public int FastPeriod { get => _fastPeriod.Value; set => _fastPeriod.Value = value; }
public int SlowPeriod { get => _slowPeriod.Value; set => _slowPeriod.Value = value; }
public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }
public Fortrader10PipsStrategy()
{
_fastPeriod = Param(nameof(FastPeriod), 7).SetDisplay("Fast EMA", "Fast EMA period", "Indicators");
_slowPeriod = Param(nameof(SlowPeriod), 21).SetDisplay("Slow EMA", "Slow EMA period", "Indicators");
_candleType = Param(nameof(CandleType), TimeSpan.FromHours(1).TimeFrame()).SetDisplay("Candle Type", "Candle timeframe", "General");
}
protected override void OnStarted2(DateTime time)
{
base.OnStarted2(time);
_hasPrev = false;
var fast = new ExponentialMovingAverage { Length = FastPeriod };
var slow = new ExponentialMovingAverage { Length = SlowPeriod };
var subscription = SubscribeCandles(CandleType);
subscription.Bind(fast, slow, ProcessCandle).Start();
}
private void ProcessCandle(ICandleMessage candle, decimal fast, decimal slow)
{
if (candle.State != CandleStates.Finished) return;
if (!_hasPrev) { _prevFast = fast; _prevSlow = slow; _hasPrev = true; return; }
if (_prevFast <= _prevSlow && fast > slow && Position <= 0)
{ if (Position < 0) BuyMarket(); BuyMarket(); }
else if (_prevFast >= _prevSlow && fast < slow && Position >= 0)
{ if (Position > 0) SellMarket(); SellMarket(); }
_prevFast = fast; _prevSlow = slow;
}
protected override void OnReseted()
{
base.OnReseted();
_prevFast = 0;
_prevSlow = 0;
_hasPrev = false;
}
}
import clr
clr.AddReference("StockSharp.Messages")
clr.AddReference("StockSharp.Algo")
clr.AddReference("StockSharp.Algo.Indicators")
clr.AddReference("StockSharp.Algo.Strategies")
from System import TimeSpan
from StockSharp.Messages import DataType, CandleStates
from StockSharp.Algo.Indicators import ExponentialMovingAverage
from StockSharp.Algo.Strategies import Strategy
class fortrader_10_pips_strategy(Strategy):
"""
Fortrader 10 Pips: Simple EMA crossover (fast 7, slow 21).
Buys when fast crosses above slow, sells on reverse.
"""
def __init__(self):
super(fortrader_10_pips_strategy, self).__init__()
self._fast_period = self.Param("FastPeriod", 7) \
.SetDisplay("Fast EMA", "Fast EMA period", "Indicators")
self._slow_period = self.Param("SlowPeriod", 21) \
.SetDisplay("Slow EMA", "Slow EMA period", "Indicators")
self._candle_type = self.Param("CandleType", DataType.TimeFrame(TimeSpan.FromMinutes(5))) \
.SetDisplay("Candle Type", "Candle timeframe", "General")
self._prev_fast = 0.0
self._prev_slow = 0.0
self._has_prev = False
@property
def candle_type(self):
return self._candle_type.Value
def OnReseted(self):
super(fortrader_10_pips_strategy, self).OnReseted()
self._prev_fast = 0.0
self._prev_slow = 0.0
self._has_prev = False
def OnStarted2(self, time):
super(fortrader_10_pips_strategy, self).OnStarted2(time)
self._has_prev = False
fast = ExponentialMovingAverage()
fast.Length = self._fast_period.Value
slow = ExponentialMovingAverage()
slow.Length = self._slow_period.Value
subscription = self.SubscribeCandles(self.candle_type)
subscription.Bind(fast, slow, self._process_candle).Start()
def _process_candle(self, candle, fast_val, slow_val):
if candle.State != CandleStates.Finished:
return
fast_val = float(fast_val)
slow_val = float(slow_val)
if not self._has_prev:
self._prev_fast = fast_val
self._prev_slow = slow_val
self._has_prev = True
return
if self._prev_fast <= self._prev_slow and fast_val > slow_val and self.Position <= 0:
if self.Position < 0:
self.BuyMarket()
self.BuyMarket()
elif self._prev_fast >= self._prev_slow and fast_val < slow_val and self.Position >= 0:
if self.Position > 0:
self.SellMarket()
self.SellMarket()
self._prev_fast = fast_val
self._prev_slow = slow_val
def CreateClone(self):
return fortrader_10_pips_strategy()