SAR RSI MTS 策略
概览
SAR RSI MTS 策略是 MetaTrader 5 专家顾问“SAR RSI MTS”的 StockSharp 高级 API 版本。系统依靠抛物线 SAR 指标来识别趋势方向,并使用相对强弱指数(RSI)确认信号。策略只在已经收盘的 K 线(默认 1 小时)上运行,并对净持仓规模设置了可配置的上限。
指标与数据
- 抛物线 SAR:
Acceleration = SarStep、AccelerationStep = SarStep、AccelerationMax = SarMax。 - RSI:可配置周期与中性水平(默认 50)。
- K 线类型:由
CandleType参数决定,默认订阅 1 小时数据。
策略会根据交易品种的价格步长和小数位数计算“点值”。当标的价格保留 3 或 5 位小数时,会自动把步长乘以 10,从而复制原始 MQL 程序中对“点”的处理方式。
入场逻辑
在每根完成的 K 线收盘时,当两个指标都给出有效结果后评估新的交易机会:
做多条件
- 前一根 K 线的 SAR 位于当前收盘价之下,并且当前的 SAR 值高于上一根。
- RSI 高于中性阈值,并且相较上一根有所上升。
- 如果当前为净空头头寸,策略会先买入足够的数量将仓位翻转,再按照
Volume参数设定的数量建立新的多单,并保证不超过MaxPosition上限。
做空条件
- 前一根 K 线的 SAR 位于当前收盘价之上,并且当前的 SAR 值低于上一根。
- RSI 低于中性阈值,并且相较上一根有所下降。
- 若已有净多头,则先平掉现有多头,再按计划建立新的空单。空头仓位的绝对值同样受到
MaxPosition限制。
所有比较都会按照品种的小数精度进行,以对应 MQL 版本中 CompareDoubles 函数的效果。
出场与风控
每根 K 线在检测新信号之前都会先执行风控逻辑:
- 固定止损:以点数配置,换算成价格距离后作用于当前平均持仓价。
- 固定止盈:同样以点数配置,逻辑与止损对称。
- 追踪止损:仅在浮动盈利超过
TrailingStop + TrailingStep时激活,并按离散步长移动,重现原 MQL 程序中的Trailing()行为。 - 当持仓归零时会自动清空追踪状态。
所有退出操作都会一次性平掉全部净头寸。当某个保护条件触发时,策略会跳过同一根 K 线的入场判断,模拟原始系统中经纪商侧止损单的效果。
参数说明
| 参数 | 说明 |
|---|---|
StopLossPips |
以点数表示的止损距离,设为 0 表示关闭。 |
TakeProfitPips |
以点数表示的止盈距离,设为 0 表示关闭。 |
TrailingStopPips |
追踪止损的基础距离,设为 0 表示关闭。 |
TrailingStepPips |
每次调整追踪止损所需的最小价格改进幅度。 |
SarStep |
抛物线 SAR 的加速度步长,同时作为初始加速度。 |
SarMax |
抛物线 SAR 的最大加速度。 |
RsiPeriod |
RSI 指标的计算周期。 |
RsiNeutralLevel |
用于区分多空的 RSI 中性水平(默认 50)。 |
CandleType |
用于计算的 K 线类型,默认 1 小时。 |
MaxPosition |
策略允许的净持仓绝对值上限。 |
其他说明
- 默认配置与原始 EA 保持一致:10 点止损、40 点止盈、15/5 点追踪止损、SAR 参数
0.05/0.5、RSI 周期 14。 - 下单数量由基础的
Strategy.Volume属性控制。策略在加仓或反向时会自动遵守MaxPosition限制。 - 全部指标绑定与交易执行均使用 StockSharp 高级 API,未直接访问原始数据缓冲,从而完全符合项目规范。
using System;
using System.Linq;
using System.Collections.Generic;
using Ecng.Common;
using Ecng.Collections;
using Ecng.Serialization;
using StockSharp.Algo.Indicators;
using StockSharp.Algo.Strategies;
using StockSharp.BusinessEntities;
using StockSharp.Messages;
namespace StockSharp.Samples.Strategies;
/// <summary>
/// Parabolic SAR and RSI strategy translated from the original MQL implementation.
/// </summary>
public class SarRsiMtsStrategy : Strategy
{
private readonly StrategyParam<decimal> _stopLossPips;
private readonly StrategyParam<decimal> _takeProfitPips;
private readonly StrategyParam<decimal> _trailingStopPips;
private readonly StrategyParam<decimal> _trailingStepPips;
private readonly StrategyParam<decimal> _sarStep;
private readonly StrategyParam<decimal> _sarMax;
private readonly StrategyParam<int> _rsiPeriod;
private readonly StrategyParam<decimal> _rsiNeutralLevel;
private readonly StrategyParam<DataType> _candleType;
private readonly StrategyParam<decimal> _maxPosition;
private decimal? _previousSar;
private decimal? _previousRsi;
private decimal? _longTrailingStop;
private decimal? _shortTrailingStop;
private decimal _pipSize;
private decimal _entryPrice;
private DateTimeOffset _lastTradeTime;
/// <summary>
/// Stop loss distance expressed in pips.
/// </summary>
public decimal StopLossPips
{
get => _stopLossPips.Value;
set => _stopLossPips.Value = value;
}
/// <summary>
/// Take profit distance expressed in pips.
/// </summary>
public decimal TakeProfitPips
{
get => _takeProfitPips.Value;
set => _takeProfitPips.Value = value;
}
/// <summary>
/// Trailing stop distance expressed in pips.
/// </summary>
public decimal TrailingStopPips
{
get => _trailingStopPips.Value;
set => _trailingStopPips.Value = value;
}
/// <summary>
/// Trailing step distance expressed in pips.
/// </summary>
public decimal TrailingStepPips
{
get => _trailingStepPips.Value;
set => _trailingStepPips.Value = value;
}
/// <summary>
/// Parabolic SAR acceleration step.
/// </summary>
public decimal SarStep
{
get => _sarStep.Value;
set => _sarStep.Value = value;
}
/// <summary>
/// Parabolic SAR maximum acceleration.
/// </summary>
public decimal SarMax
{
get => _sarMax.Value;
set => _sarMax.Value = value;
}
/// <summary>
/// RSI lookback period.
/// </summary>
public int RsiPeriod
{
get => _rsiPeriod.Value;
set => _rsiPeriod.Value = value;
}
/// <summary>
/// RSI neutral level used for bullish or bearish confirmation.
/// </summary>
public decimal RsiNeutralLevel
{
get => _rsiNeutralLevel.Value;
set => _rsiNeutralLevel.Value = value;
}
/// <summary>
/// Candle type used for indicator calculations.
/// </summary>
public DataType CandleType
{
get => _candleType.Value;
set => _candleType.Value = value;
}
/// <summary>
/// Maximum absolute net position allowed by the strategy.
/// </summary>
public decimal MaxPosition
{
get => _maxPosition.Value;
set => _maxPosition.Value = value;
}
/// <summary>
/// Initializes a new instance of the <see cref="SarRsiMtsStrategy"/> class.
/// </summary>
public SarRsiMtsStrategy()
{
_stopLossPips = Param(nameof(StopLossPips), 10m)
.SetNotNegative()
.SetDisplay("Stop Loss (pips)", "Stop loss distance in pips", "Risk");
_takeProfitPips = Param(nameof(TakeProfitPips), 40m)
.SetNotNegative()
.SetDisplay("Take Profit (pips)", "Take profit distance in pips", "Risk");
_trailingStopPips = Param(nameof(TrailingStopPips), 15m)
.SetNotNegative()
.SetDisplay("Trailing Stop (pips)", "Trailing stop distance in pips", "Risk");
_trailingStepPips = Param(nameof(TrailingStepPips), 5m)
.SetNotNegative()
.SetDisplay("Trailing Step (pips)", "Trailing step distance in pips", "Risk");
_sarStep = Param(nameof(SarStep), 0.05m)
.SetGreaterThanZero()
.SetDisplay("SAR Step", "Parabolic SAR acceleration step", "Indicators");
_sarMax = Param(nameof(SarMax), 0.5m)
.SetGreaterThanZero()
.SetDisplay("SAR Maximum", "Parabolic SAR maximum acceleration", "Indicators");
_rsiPeriod = Param(nameof(RsiPeriod), 14)
.SetGreaterThanZero()
.SetDisplay("RSI Period", "Lookback period for RSI", "Indicators");
_rsiNeutralLevel = Param(nameof(RsiNeutralLevel), 50m)
.SetDisplay("RSI Neutral", "Neutral RSI threshold separating bullish and bearish bias", "Indicators");
_candleType = Param(nameof(CandleType), TimeSpan.FromMinutes(5).TimeFrame())
.SetDisplay("Candle Type", "Candle type for indicator calculations", "General");
_maxPosition = Param(nameof(MaxPosition), 5m)
.SetGreaterThanZero()
.SetDisplay("Max Position", "Maximum absolute net position allowed", "Risk");
}
/// <inheritdoc />
public override IEnumerable<(Security sec, DataType dt)> GetWorkingSecurities()
{
return [(Security, CandleType)];
}
protected override void OnReseted()
{
base.OnReseted();
_previousSar = null;
_previousRsi = null;
_longTrailingStop = null;
_shortTrailingStop = null;
_pipSize = 0;
_entryPrice = 0;
_lastTradeTime = default;
}
/// <inheritdoc />
protected override void OnStarted2(DateTime time)
{
base.OnStarted2(time);
_pipSize = CalculatePipSize();
var parabolicSar = new ParabolicSar
{
Acceleration = SarStep,
AccelerationStep = SarStep,
AccelerationMax = SarMax
};
var rsi = new RelativeStrengthIndex
{
Length = RsiPeriod
};
var subscription = SubscribeCandles(CandleType);
subscription
.Bind(parabolicSar, rsi, ProcessCandle)
.Start();
var area = CreateChartArea();
if (area != null)
{
DrawCandles(area, subscription);
DrawIndicator(area, parabolicSar);
DrawIndicator(area, rsi);
DrawOwnTrades(area);
}
}
private void ProcessCandle(ICandleMessage candle, decimal sarValue, decimal rsiValue)
{
if (candle.State != CandleStates.Finished)
return;
if (ManageRisk(candle))
return;
if (sarValue == 0m || rsiValue == 0m)
return;
if (!_previousSar.HasValue || !_previousRsi.HasValue)
{
_previousSar = sarValue;
_previousRsi = rsiValue;
return;
}
if (!IsFormedAndOnlineAndAllowTrading())
{
_previousSar = sarValue;
_previousRsi = rsiValue;
return;
}
// Cooldown: skip if a trade was placed within the last ~240 candles (5-min candles = ~1200 min)
if (_lastTradeTime != default && (candle.OpenTime - _lastTradeTime) < TimeSpan.FromMinutes(1200))
{
_previousSar = sarValue;
_previousRsi = rsiValue;
return;
}
var sarPrev = _previousSar.Value;
var rsiPrev = _previousRsi.Value;
var price = candle.ClosePrice;
var buySignal = sarPrev < price
&& !AreClose(sarPrev, price)
&& sarValue > sarPrev
&& rsiValue > RsiNeutralLevel
&& rsiValue > rsiPrev
&& !AreClose(rsiValue, rsiPrev);
if (buySignal)
{
EnterLong(candle);
}
else
{
var sellSignal = sarPrev > price
&& !AreClose(sarPrev, price)
&& sarValue < sarPrev
&& rsiValue < RsiNeutralLevel
&& rsiValue < rsiPrev
&& !AreClose(rsiValue, rsiPrev);
if (sellSignal)
EnterShort(candle);
}
_previousSar = sarValue;
_previousRsi = rsiValue;
}
private void EnterLong(ICandleMessage candle)
{
var tradeVolume = Volume;
if (tradeVolume <= 0m)
return;
var maxPosition = MaxPosition;
if (maxPosition <= 0m)
return;
var current = Position;
var target = current < 0 ? Math.Min(maxPosition, tradeVolume) : Math.Min(maxPosition, current + tradeVolume);
var required = target - current;
if (required <= 0m)
return;
BuyMarket(required);
_longTrailingStop = null;
_shortTrailingStop = null;
_lastTradeTime = candle.OpenTime;
}
private void EnterShort(ICandleMessage candle)
{
var tradeVolume = Volume;
if (tradeVolume <= 0m)
return;
var maxPosition = MaxPosition;
if (maxPosition <= 0m)
return;
var current = Position;
var target = current > 0 ? -Math.Min(maxPosition, tradeVolume) : Math.Max(-maxPosition, current - tradeVolume);
var required = current - target;
if (required <= 0m)
return;
SellMarket(required);
_longTrailingStop = null;
_shortTrailingStop = null;
_lastTradeTime = candle.OpenTime;
}
private bool ManageRisk(ICandleMessage candle)
{
if (Position > 0m)
{
var entryPrice = _entryPrice;
if (entryPrice <= 0m)
return false;
var trailingTriggered = UpdateLongTrailing(candle, entryPrice);
if (trailingTriggered)
return true;
var stopDistance = GetPriceOffset(StopLossPips);
if (stopDistance > 0m)
{
var stopPrice = entryPrice - stopDistance;
if (candle.LowPrice <= stopPrice)
{
SellMarket(Position);
ResetTrailing();
return true;
}
}
var takeDistance = GetPriceOffset(TakeProfitPips);
if (takeDistance > 0m)
{
var takePrice = entryPrice + takeDistance;
if (candle.HighPrice >= takePrice)
{
SellMarket(Position);
ResetTrailing();
return true;
}
}
}
else if (Position < 0m)
{
var entryPrice = _entryPrice;
if (entryPrice <= 0m)
return false;
var trailingTriggered = UpdateShortTrailing(candle, entryPrice);
if (trailingTriggered)
return true;
var stopDistance = GetPriceOffset(StopLossPips);
if (stopDistance > 0m)
{
var stopPrice = entryPrice + stopDistance;
if (candle.HighPrice >= stopPrice)
{
BuyMarket(Math.Abs(Position));
ResetTrailing();
return true;
}
}
var takeDistance = GetPriceOffset(TakeProfitPips);
if (takeDistance > 0m)
{
var takePrice = entryPrice - takeDistance;
if (candle.LowPrice <= takePrice)
{
BuyMarket(Math.Abs(Position));
ResetTrailing();
return true;
}
}
}
else
{
ResetTrailing();
}
return false;
}
private bool UpdateLongTrailing(ICandleMessage candle, decimal entryPrice)
{
var trailingDistance = GetPriceOffset(TrailingStopPips);
if (trailingDistance <= 0m)
{
_longTrailingStop = null;
return false;
}
var trailingStep = GetPriceOffset(TrailingStepPips);
var profit = candle.ClosePrice - entryPrice;
if (profit >= trailingDistance + trailingStep)
{
var candidate = candle.ClosePrice - trailingDistance;
var threshold = candle.ClosePrice - (trailingDistance + trailingStep);
if (!_longTrailingStop.HasValue || _longTrailingStop.Value < threshold)
_longTrailingStop = candidate;
}
if (_longTrailingStop.HasValue && candle.LowPrice <= _longTrailingStop.Value)
{
SellMarket(Position);
ResetTrailing();
return true;
}
return false;
}
private bool UpdateShortTrailing(ICandleMessage candle, decimal entryPrice)
{
var trailingDistance = GetPriceOffset(TrailingStopPips);
if (trailingDistance <= 0m)
{
_shortTrailingStop = null;
return false;
}
var trailingStep = GetPriceOffset(TrailingStepPips);
var profit = entryPrice - candle.ClosePrice;
if (profit >= trailingDistance + trailingStep)
{
var candidate = candle.ClosePrice + trailingDistance;
var threshold = candle.ClosePrice + (trailingDistance + trailingStep);
if (!_shortTrailingStop.HasValue || _shortTrailingStop.Value > threshold)
_shortTrailingStop = candidate;
}
if (_shortTrailingStop.HasValue && candle.HighPrice >= _shortTrailingStop.Value)
{
BuyMarket(Math.Abs(Position));
ResetTrailing();
return true;
}
return false;
}
private decimal GetPriceOffset(decimal pips)
{
if (pips <= 0m)
return 0m;
var pip = _pipSize;
if (pip <= 0m)
pip = Security?.PriceStep ?? 1m;
return pip * pips;
}
private decimal CalculatePipSize()
{
var priceStep = Security?.PriceStep ?? 0m;
if (priceStep <= 0m)
priceStep = 1m;
var decimals = Security?.Decimals;
var adjust = decimals == 3 || decimals == 5 ? 10m : 1m;
return priceStep * adjust;
}
private void ResetTrailing()
{
_longTrailingStop = null;
_shortTrailingStop = null;
}
/// <inheritdoc />
protected override void OnOwnTradeReceived(MyTrade trade)
{
base.OnOwnTradeReceived(trade);
if (trade?.Trade == null) return;
if (Position != 0 && _entryPrice == 0m)
_entryPrice = trade.Trade.Price;
if (Position == 0)
_entryPrice = 0m;
}
private bool AreClose(decimal value1, decimal value2)
{
var decimals = Security?.Decimals ?? 4;
return Math.Round(value1 - value2, decimals) == 0m;
}
}
import clr
clr.AddReference("StockSharp.Messages")
clr.AddReference("StockSharp.Algo")
clr.AddReference("StockSharp.Algo.Indicators")
clr.AddReference("StockSharp.Algo.Strategies")
from System import TimeSpan, Math
from StockSharp.Messages import DataType, CandleStates
from StockSharp.Algo.Indicators import ParabolicSar, RelativeStrengthIndex
from StockSharp.Algo.Strategies import Strategy
from datatype_extensions import *
from indicator_extensions import *
class sar_rsi_mts_strategy(Strategy):
def __init__(self):
super(sar_rsi_mts_strategy, self).__init__()
self._sl_pips = self.Param("StopLossPips", 10.0).SetNotNegative().SetDisplay("Stop Loss (pips)", "SL distance", "Risk")
self._tp_pips = self.Param("TakeProfitPips", 40.0).SetNotNegative().SetDisplay("Take Profit (pips)", "TP distance", "Risk")
self._trailing_pips = self.Param("TrailingStopPips", 15.0).SetNotNegative().SetDisplay("Trailing Stop (pips)", "Trailing stop distance", "Risk")
self._trailing_step_pips = self.Param("TrailingStepPips", 5.0).SetNotNegative().SetDisplay("Trailing Step (pips)", "Trailing step distance", "Risk")
self._sar_step = self.Param("SarStep", 0.05).SetGreaterThanZero().SetDisplay("SAR Step", "Parabolic SAR acceleration step", "Indicators")
self._sar_max = self.Param("SarMax", 0.5).SetGreaterThanZero().SetDisplay("SAR Maximum", "Parabolic SAR maximum acceleration", "Indicators")
self._rsi_period = self.Param("RsiPeriod", 14).SetGreaterThanZero().SetDisplay("RSI Period", "Lookback period for RSI", "Indicators")
self._rsi_neutral = self.Param("RsiNeutralLevel", 50.0).SetDisplay("RSI Neutral", "Neutral RSI threshold", "Indicators")
self._candle_type = self.Param("CandleType", DataType.TimeFrame(TimeSpan.FromMinutes(5))).SetDisplay("Candle Type", "Candle type", "General")
@property
def CandleType(self): return self._candle_type.Value
@CandleType.setter
def CandleType(self, value): self._candle_type.Value = value
def OnReseted(self):
super(sar_rsi_mts_strategy, self).OnReseted()
self._prev_sar = None
self._prev_rsi = None
self._entry_price = 0
self._long_trailing = None
self._short_trailing = None
def OnStarted2(self, time):
super(sar_rsi_mts_strategy, self).OnStarted2(time)
self._prev_sar = None
self._prev_rsi = None
self._entry_price = 0
self._long_trailing = None
self._short_trailing = None
self._pip_size = 1.0
if self.Security is not None and self.Security.PriceStep is not None and self.Security.PriceStep > 0:
self._pip_size = float(self.Security.PriceStep)
sar = ParabolicSar()
sar.Acceleration = self._sar_step.Value
sar.AccelerationStep = self._sar_step.Value
sar.AccelerationMax = self._sar_max.Value
rsi = RelativeStrengthIndex()
rsi.Length = self._rsi_period.Value
sub = self.SubscribeCandles(self.CandleType)
sub.Bind(sar, rsi, self.OnProcess).Start()
area = self.CreateChartArea()
if area is not None:
self.DrawCandles(area, sub)
self.DrawIndicator(area, sar)
self.DrawIndicator(area, rsi)
self.DrawOwnTrades(area)
def OnProcess(self, candle, sar_val, rsi_val):
if candle.State != CandleStates.Finished:
return
close = float(candle.ClosePrice)
# Manage risk first
if self._manage_risk(candle, close):
self._prev_sar = sar_val
self._prev_rsi = rsi_val
return
if self._prev_sar is None or self._prev_rsi is None:
self._prev_sar = sar_val
self._prev_rsi = rsi_val
return
prev_sar = self._prev_sar
prev_rsi = self._prev_rsi
# Buy signal: SAR below price, SAR rising, RSI above neutral and rising
buy_signal = (prev_sar < close
and sar_val > prev_sar
and rsi_val > self._rsi_neutral.Value
and rsi_val > prev_rsi)
if buy_signal:
if self.Position < 0:
self.BuyMarket()
if self.Position <= 0:
self.BuyMarket()
self._entry_price = close
self._long_trailing = None
self._short_trailing = None
else:
# Sell signal: SAR above price, SAR falling, RSI below neutral and falling
sell_signal = (prev_sar > close
and sar_val < prev_sar
and rsi_val < self._rsi_neutral.Value
and rsi_val < prev_rsi)
if sell_signal:
if self.Position > 0:
self.SellMarket()
if self.Position >= 0:
self.SellMarket()
self._entry_price = close
self._long_trailing = None
self._short_trailing = None
self._prev_sar = sar_val
self._prev_rsi = rsi_val
def _manage_risk(self, candle, close):
pip = self._pip_size
if self.Position > 0 and self._entry_price > 0:
# Trailing
trail_dist = self._trailing_pips.Value * pip
trail_step = self._trailing_step_pips.Value * pip
if trail_dist > 0:
profit = close - self._entry_price
if profit >= trail_dist + trail_step:
candidate = close - trail_dist
threshold = close - (trail_dist + trail_step)
if self._long_trailing is None or self._long_trailing < threshold:
self._long_trailing = candidate
if self._long_trailing is not None and float(candle.LowPrice) <= self._long_trailing:
self.SellMarket()
self._reset_trailing()
return True
# Stop loss
sl_dist = self._sl_pips.Value * pip
if sl_dist > 0:
if float(candle.LowPrice) <= self._entry_price - sl_dist:
self.SellMarket()
self._reset_trailing()
return True
# Take profit
tp_dist = self._tp_pips.Value * pip
if tp_dist > 0:
if float(candle.HighPrice) >= self._entry_price + tp_dist:
self.SellMarket()
self._reset_trailing()
return True
elif self.Position < 0 and self._entry_price > 0:
# Trailing
trail_dist = self._trailing_pips.Value * pip
trail_step = self._trailing_step_pips.Value * pip
if trail_dist > 0:
profit = self._entry_price - close
if profit >= trail_dist + trail_step:
candidate = close + trail_dist
threshold = close + (trail_dist + trail_step)
if self._short_trailing is None or self._short_trailing > threshold:
self._short_trailing = candidate
if self._short_trailing is not None and float(candle.HighPrice) >= self._short_trailing:
self.BuyMarket()
self._reset_trailing()
return True
# Stop loss
sl_dist = self._sl_pips.Value * pip
if sl_dist > 0:
if float(candle.HighPrice) >= self._entry_price + sl_dist:
self.BuyMarket()
self._reset_trailing()
return True
# Take profit
tp_dist = self._tp_pips.Value * pip
if tp_dist > 0:
if float(candle.LowPrice) <= self._entry_price - tp_dist:
self.BuyMarket()
self._reset_trailing()
return True
else:
self._reset_trailing()
return False
def _reset_trailing(self):
self._long_trailing = None
self._short_trailing = None
self._entry_price = 0
def CreateClone(self):
return sar_rsi_mts_strategy()