在 GitHub 上查看
Simple FX 趋势交叉策略
摘要
- 基于 MetaTrader 4 专家顾问 simplefx2.mq4(Simple FX 2.0)的高层 API 移植版本。
- 在收盘完成的K线数据上交易快/慢简单移动平均线的金叉与死叉。
- 始终只保持一个方向的仓位,在趋势反转时立即翻转持仓。
交易逻辑
- 按照可配置的时间框架生成K线。
- 使用K线收盘价计算一条快线和一条慢线简单移动平均值。
- 当当前与上一根K线都显示快线高于慢线时确认多头趋势;当两根K线都显示快线低于慢线时确认空头趋势。
- 如果确认的趋势与策略保存的趋势状态不同,则先平掉反向持仓,再按照设定手数以市价开仓跟随新趋势。
- 可选的止损/止盈设置使用价格步长表示,并通过 StockSharp 的保护服务执行,以贴近原始 MT4 机器人。
策略仅处理完整收盘的K线,不对单个Tick作出反应,从而保持与原程序一致的节奏。每次开仓都会写入日志,便于复盘每个交叉信号。
参数
| 名称 |
描述 |
默认值 |
优化范围 |
ShortPeriod |
快速简单移动平均线周期。 |
50 |
10 → 150 步长 5 |
LongPeriod |
慢速简单移动平均线周期。 |
200 |
50 → 400 步长 10 |
Volume |
每次市价单的交易手数。 |
0.1 |
0.1 → 2 步长 0.1 |
StopLossPoints |
以价格步长表示的止损距离(0 表示关闭)。 |
0 |
— |
TakeProfitPoints |
以价格步长表示的止盈距离(0 表示关闭)。 |
0 |
— |
CandleType |
用于分析的K线时间框架。 |
1 小时 |
— |
与 MT4 版本的差异
- 不再需要 MT4 的
simplefx.dat 持久化文件,趋势方向在策略状态中直接维护。
- MT4 中的滑点、订单备注、魔术编号和箭头颜色等选项未提供,对应功能由 StockSharp 以不同方式处理。
- 止损与止盈距离以价格步长解释,部署前请根据交易商的点值调整。
- 策略始终只有一笔仓位,翻转时通过
ClosePosition() 清理旧仓后再开新仓。
使用步骤
- 选择交易品种并设置所需的K线时间框架。
- 根据需要调整两条移动平均线周期以及风控参数。
- 启动策略后,它会订阅K线、维护趋势状态,并在连续两根K线确认交叉时发出市价单。
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 FX crossover strategy.
/// Uses fast and slow SMA crossover for trend detection.
/// Buys on golden cross, sells on death cross.
/// </summary>
public class SimpleFxCrossoverStrategy : Strategy
{
private readonly StrategyParam<int> _shortPeriod;
private readonly StrategyParam<int> _longPeriod;
private readonly StrategyParam<DataType> _candleType;
private decimal _prevShort;
private decimal _prevLong;
private bool _hasPrev;
public int ShortPeriod { get => _shortPeriod.Value; set => _shortPeriod.Value = value; }
public int LongPeriod { get => _longPeriod.Value; set => _longPeriod.Value = value; }
public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }
public SimpleFxCrossoverStrategy()
{
_shortPeriod = Param(nameof(ShortPeriod), 10)
.SetDisplay("Fast SMA", "Fast SMA period", "Indicators");
_longPeriod = Param(nameof(LongPeriod), 30)
.SetDisplay("Slow SMA", "Slow SMA 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();
_prevShort = 0m;
_prevLong = 0m;
_hasPrev = false;
}
protected override void OnStarted2(DateTime time)
{
base.OnStarted2(time);
_hasPrev = false;
var fast = new SimpleMovingAverage { Length = ShortPeriod };
var slow = new SimpleMovingAverage { Length = LongPeriod };
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)
{
_prevShort = fast;
_prevLong = slow;
_hasPrev = true;
return;
}
var crossUp = _prevShort <= _prevLong && fast > slow;
var crossDown = _prevShort >= _prevLong && fast < slow;
if (crossUp && Position <= 0)
{
if (Position < 0)
BuyMarket();
BuyMarket();
}
else if (crossDown && Position >= 0)
{
if (Position > 0)
SellMarket();
SellMarket();
}
_prevShort = fast;
_prevLong = slow;
}
}
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 SimpleMovingAverage
from StockSharp.Algo.Strategies import Strategy
class simple_fx_crossover_strategy(Strategy):
def __init__(self):
super(simple_fx_crossover_strategy, self).__init__()
self._short_period = self.Param("ShortPeriod", 10) \
.SetDisplay("Fast SMA", "Fast SMA period", "Indicators")
self._long_period = self.Param("LongPeriod", 30) \
.SetDisplay("Slow SMA", "Slow SMA period", "Indicators")
self._candle_type = self.Param("CandleType", DataType.TimeFrame(TimeSpan.FromHours(1))) \
.SetDisplay("Candle Type", "Candle timeframe", "General")
self._prev_short = 0.0
self._prev_long = 0.0
self._has_prev = False
@property
def short_period(self):
return self._short_period.Value
@property
def long_period(self):
return self._long_period.Value
@property
def candle_type(self):
return self._candle_type.Value
def OnReseted(self):
super(simple_fx_crossover_strategy, self).OnReseted()
self._prev_short = 0.0
self._prev_long = 0.0
self._has_prev = False
def OnStarted2(self, time):
super(simple_fx_crossover_strategy, self).OnStarted2(time)
self._has_prev = False
fast = SimpleMovingAverage()
fast.Length = self.short_period
slow = SimpleMovingAverage()
slow.Length = self.long_period
subscription = self.SubscribeCandles(self.candle_type)
subscription.Bind(fast, slow, self.process_candle).Start()
def process_candle(self, candle, fast, slow):
if candle.State != CandleStates.Finished:
return
fast_val = float(fast)
slow_val = float(slow)
if not self._has_prev:
self._prev_short = fast_val
self._prev_long = slow_val
self._has_prev = True
return
cross_up = self._prev_short <= self._prev_long and fast_val > slow_val
cross_down = self._prev_short >= self._prev_long and fast_val < slow_val
if cross_up and self.Position <= 0:
if self.Position < 0:
self.BuyMarket()
self.BuyMarket()
elif cross_down and self.Position >= 0:
if self.Position > 0:
self.SellMarket()
self.SellMarket()
self._prev_short = fast_val
self._prev_long = slow_val
def CreateClone(self):
return simple_fx_crossover_strategy()