“还在为MT4和MT5代码不兼容而烦恼吗?还在重复编写两套代码吗?今天分享一个实用技巧,让你的交易策略代码同时支持MT4和MT5平台!
📌 痛点:为什么需要通用交易函数?
作为量化交易开发者,你是否遇到过这样的困扰:
- 开发时间翻倍:同一个策略,MT4写一套,MT5再写一套,工作量直接翻倍
- 维护成本高:两套代码需要分别维护,修改一个功能要改两处
- 调试困难:不同平台的代码逻辑分散,找bug就像大海捞针
- 代码混乱:状态和行为混在一起,逻辑不清晰,难以扩展
如果你也有这些烦恼,那么通用交易函数就是你的救星!
💡 核心思想:用宏实现跨平台编译
什么是跨平台编译?
简单来说,就是一份代码,同时支持MT4和MT5两个平台。
MT4和MT5的交易函数互不兼容,但我们可以利用编译器内置的宏来实现:
通过条件编译,让编译器自动选择对应平台的代码,实现"一套代码,双平台运行"。
设计原则
1. 按功能模块划分
将交易操作按单一功能划分:
2. 状态与行为分离
这样设计的好处是:
🚀 实战:多空交易函数实现
Buy函数:做多交易
函数功能:
- 传入参数:成交量、止损、止盈、魔术编号、产品符号、订单注释
核心代码逻辑:
ulong Buy(double volume, double sl, double tp, ulong magic, string symbol, string comment){// 1. 检查手数是否合法if (!CheckVolumeValue(volume, symbol, error))return (0);// 2. 获取当前卖价double ask = SymbolInfoDouble(symbol, SYMBOL_ASK);// 3. 根据平台选择对应代码#ifdef __MQL4__// MT4平台的代码int ticket = OrderSend(...);#endif#ifdef __MQL5__// MT5平台的代码 MqlTradeRequest request = {}; OrderSend(request, result);#endifreturn (ticket);}
关键点:
Sell函数:做空交易
Sell函数的实现逻辑与Buy函数类似,只是方向相反:
手数验证:CheckVolumeValue函数
这是一个重要的辅助函数,用于验证交易手数是否合法:
检查项:
这样可以避免因手数错误导致的交易失败。
📦 批量平仓:高效管理仓位
设计思路
平仓操作通常是批量处理的,比如:
因此,平仓函数设计为批量平仓模式。
核心实现
1. 宏定义循环
通过宏定义统一循环方式,实现跨平台:
#ifdef __MQL4__#define ForEachPositionDown(i) for (int i=OrdersTotal()-1; i>=0; i--)#endif#ifdef __MQL5__#define ForEachPositionDown(i) for (int i=PositionsTotal()-1; i>=0; i--)#endif
2. 批量平仓函数
boolBuyClose(ulong magic, string symbol){bool ret = true;// 遍历所有仓位 ForEachPositionDown(i) {if (CheckPositionSelectedByPos(i)) ret = ret && SelectedBuyClose(magic, symbol); }return (ret);}
设计亮点:
🎯 进阶:多功能平仓函数设计
为什么需要更灵活的平仓函数?
基础的平仓函数只能按单个魔术编号和产品符号过滤,但在实际交易中,我们经常需要:
- 多策略管理:同时运行多个策略,每个策略有不同的魔术编号
- 多品种交易:同时交易多个货币对,需要按品种分别平仓
- 分类管理:通过订单注释区分不同类型的交易,需要按注释筛选平仓
MT4和MT5提供的交易函数接口简单,无法满足这些复杂需求。因此,我们需要设计一个更通用、更灵活的平仓函数。
设计思路:参数化多条件筛选
核心思想:将所有筛选条件抽象为函数参数,通过数组传递多个条件值。
设计特点:
- ✅ 支持数组参数:可以传入多个魔术编号、多个产品符号、多个订单注释
- ✅ 空数组=全局:如果数组为空,表示不限制该条件,实现灵活组合
- ✅ 多条件组合:同时支持按魔术编号、产品符号、订单注释、仓位类型筛选
- ✅ 跨平台兼容:使用宏定义实现MT4和MT5统一接口
核心代码实现
1. 多功能平仓函数
// 根据指定魔术编号数组、产品符号数组、订单注释数组平仓多头仓位// 数组为空时,默认全局,如此便实现了各种条件的组合筛选boolBuyClose(ulong &magic[], string &symbol[], string &comment[]){bool ret = true;// 从最新时间到过去,遍历所有仓位 ForEachPositionDown(i) {if (CheckPositionSelectedByPos(i)) {// 判断是否符合魔术编号匹配if (!CheckPositionSelectedMagic(magic)) continue;// 判断是否符合产品符号匹配if (!CheckPositionSelectedSymbol(symbol)) continue;// 判断是否符合订单注释匹配if (!CheckPositionSelectedComment(comment)) continue;// 判断是否是多头仓位if (!CheckSelectedBuy()) continue;// 所有条件符合后,执行平仓 ret = ret && SelectedPositionClose(); } }return (ret);}
2. 筛选函数实现
// 筛选仓位是否符合魔术编号数组boolCheckPositionSelectedMagic(ulong &magic[]){int arr_len = ArraySize(magic);if (arr_len == 0) return (true); // 空数组表示不限制#ifdef __MQL4__if (MathQuery(magic, OrderMagicNumber()) == -1) return (false);#endif#ifdef __MQL5__ ulong pos_magic = PositionGetInteger(POSITION_MAGIC);if (MathQuery(magic, pos_magic) == -1) return (false);#endifreturn (true);}// 筛选仓位是否符合产品符号数组boolCheckPositionSelectedSymbol(string &symbol[]){int arr_len = ArraySize(symbol);if (arr_len == 0) return (true); // 空数组表示不限制string sym = "";#ifdef __MQL4__ sym = OrderSymbol();#endif#ifdef __MQL5__ sym = PositionGetString(POSITION_SYMBOL);#endifif (!ArrayContains(symbol, sym)) return (false);return (true);}// 筛选仓位是否符合订单注释数组boolCheckPositionSelectedComment(string &comment[]){int arr_len = ArraySize(comment);if (arr_len == 0) return (true); // 空数组表示不限制if (!ArrayContains(comment, SelectedPositionComment())) return (false);return (true);}
3. 通用平仓函数
// 平仓指定选择的仓位(支持多单和空单)intSelectedPositionClose(void){#ifdef __MQL4__double close_price;if (OrderType() == OP_BUY) close_price = SymbolInfoDouble(OrderSymbol(), SYMBOL_BID);elseif (OrderType() == OP_SELL) close_price = SymbolInfoDouble(OrderSymbol(), SYMBOL_ASK);elsereturn (-1);if (!OrderClose(OrderTicket(), OrderLots(), close_price, _Deviation, clrNONE)) { Print("OrderClose failed with error #", GetLastError());return (0); }#endif#ifdef __MQL5__ MqlTradeRequest request; MqlTradeResult result; ZeroMemory(request); ZeroMemory(result); ENUM_POSITION_TYPE type = (ENUM_POSITION_TYPE)PositionGetInteger(POSITION_TYPE);string symbol = PositionGetString(POSITION_SYMBOL); request.action = TRADE_ACTION_DEAL; request.position = PositionGetInteger(POSITION_TICKET); request.symbol = symbol; request.volume = PositionGetDouble(POSITION_VOLUME); request.deviation = _Deviation;if (type == POSITION_TYPE_BUY) { request.price = SymbolInfoDouble(symbol, SYMBOL_BID); request.type = ORDER_TYPE_SELL; }elseif (type == POSITION_TYPE_SELL) { request.price = SymbolInfoDouble(symbol, SYMBOL_ASK); request.type = ORDER_TYPE_BUY; }if (!FillingCheck(symbol, request, result)) return (0);if (!OrderSend(request, result)) { PrintFormat("OrderSend error %d", GetLastError());return (0); }#endifreturn (1);}
关键设计点:
- ✅ 空数组=全局:如果传入空数组,表示不限制该条件,实现灵活筛选
- ✅ 多条件组合:可以同时按魔术编号、产品符号、订单注释筛选
- ✅ 统一接口:BuyClose和SellClose使用相同的设计思路
💼 多功能平仓函数的实际应用
场景1:多策略管理
假设你同时运行3个策略,每个策略有不同的魔术编号:
// 只平仓策略1和策略2的多单ulong magic_array[] = {1001, 1002};string symbol_array[] = {}; // 空数组,不限制品种string comment_array[] = {}; // 空数组,不限制注释BuyClose(magic_array, symbol_array, comment_array);
场景2:多品种分类平仓
只平仓特定品种的多单:
// 只平仓EURUSD和GBPUSD的多单ulong magic_array[] = {}; // 空数组,不限制魔术编号string symbol_array[] = {"EURUSD", "GBPUSD"};string comment_array[] = {}; // 空数组,不限制注释BuyClose(magic_array, symbol_array, comment_array);
场景3:按订单注释筛选
通过订单注释区分不同类型的交易:
// 只平仓标记为"趋势跟踪"的多单ulong magic_array[] = {}; // 空数组,不限制魔术编号string symbol_array[] = {}; // 空数组,不限制品种string comment_array[] = {"趋势跟踪"};BuyClose(magic_array, symbol_array, comment_array);
场景4:复杂组合条件
同时满足多个条件:
// 平仓策略1在EURUSD上的"趋势跟踪"多单ulong magic_array[] = {1001};string symbol_array[] = {"EURUSD"};string comment_array[] = {"趋势跟踪"};BuyClose(magic_array, symbol_array, comment_array);
场景5:风险控制 - 按亏损平仓
虽然函数本身不支持按盈亏筛选,但可以结合其他逻辑:
// 遍历所有仓位,只平仓亏损超过阈值的多单ForEachPositionDown(i){if (CheckPositionSelectedByPos(i)) {double profit = PositionGetDouble(POSITION_PROFIT);if (profit < -100.0 && CheckSelectedBuy()) // 亏损超过100美元 { SelectedPositionClose(); } }}
🔄 基础版 vs 进阶版对比
| | |
|---|
| 魔术编号筛选 | | |
| 产品符号筛选 | | |
| 订单注释筛选 | | |
| 条件组合 | | |
| 使用场景 | | |
| 代码复杂度 | | |
选择建议:
🎯 基础版应用场景
以下是基础版交易函数的典型应用场景:
场景1:策略信号反转
// 当策略信号从做多转为做空时if (新信号 == 做空 && 当前有做多仓位){ BuyClose(magic, symbol); // 平掉所有做多仓位 Sell(volume, sl, tp, magic, symbol, "信号反转");}
场景2:风险控制
// 当账户亏损超过阈值时if (账户亏损 > 风险阈值){ BuyClose(magic, symbol); // 平掉所有做多仓位 SellClose(magic, symbol); // 平掉所有做空仓位}
场景3:多品种交易
// 同时交易多个品种string symbols[] = {"EURUSD", "GBPUSD", "USDJPY"};for (int i = 0; i < ArraySize(symbols); i++){ Buy(0.1, 0, 0, magic, symbols[i], "多品种策略");}
✨ 使用通用交易函数的好处
1. 开发效率提升
2. 代码质量提升
3. 维护成本降低
📝 总结
通用交易函数的设计核心在于:
- 利用宏实现跨平台编译:
__MQL4__和__MQL5__条件编译
功能层次
基础功能:
进阶功能:
通过这种方式,你可以:
💬 写在最后
量化交易开发是一个不断优化的过程。本文介绍了从基础到进阶的通用交易函数设计:
基础版适合简单策略,代码简洁,易于理解。
进阶版适合复杂场景,功能强大,支持多策略、多品种的灵活管理。
后续优化方向
- ✅ 挂单功能:BuyLimit、SellLimit等挂单函数的通用实现
如果你也在开发MT4/MT5交易策略,不妨试试这个方案:
相信会让你的开发效率大大提升!
📢 关注「超哥量化」公众号,获取更多量化交易开发干货!