MAGICNUMBER: 定单的“魔法“识别符
在运用MT4进行量化交易中,如果一个账户中多个策略同时运行,或者一个策略在多品种中运行,如何管理这些策略,魔法号是一个比较好的方法。那么什么是魔法号吗?下面我们看一下下单函数
int OrderSend( string symbol, int cmd, double volume, double price, int slippage, double stoploss, double takeprofit, string comment=NULL, int magic=0, datetime expiration=0, color arrow_color=CLR_NONE)
…
magic - 定单魔法数字。可以使用指定识别符。下单函数中的magic参数既是魔法号。
即,当定单被放置时,可以给定单指定独有的数字。这个数字将用于区分其他定单。当手动交易时,不使用(尽可能)这个特性,但是在智能交易(自动交易)运作时,此特性是不可替代的。
范例1: 在客户端内交易者和智能交易同时运行。
任务: 智能交易的运行必须按照它自己的形式计算,对于手动开仓不会干涉。
解决: 开仓的智能交易必须指定独特的MagicNumber(零除外)。接下来的日子里,智能交易只会管理提前设定的 MagicNumber 的定单。范例 2: 在客户端内两个不同计算方法的智能交易同时运行。
任务: 智能交易只管理自己的定单。.
解决: 当开仓时,每个智能交易必须使用自己的 MagicNumber(零除外)。接下来的日子里,智能交易只会管理提前设定的MagicNumber的定单。范例 3: 在客户端内几个智能交易,交易者和协助智能交易执行的不标准追踪止损同时运行。
任务: 交易的智能交易必须按照自己的形式计算,并且不干涉手动开仓。协助智能交易执行的追踪止损可以在手动开仓处修改,但是其他智能交易不能够开仓。
解决: 智能交易必须使用独有的MagicNumbers并且管理自己的仓位。协助智能交易修改的这些仓位,其 MagicNumber等于 0。以上三个范例都很现实,用户可以提前解决问题。在这三个范例情况中,都是使用MagicNumber来解决问题的。这不是唯一解决问题的途径,但是最简单的途径。
实际应用现在让我们来解决特殊的任务:创建智能交易只管理自己的仓位,不理睬手动仓位和其他智能交易。首先我们编写一个简单的智能交易,当 MACD 指标零线时,智能交易得到信号开仓。智能交易将会运行如下:
//+------------------------------------------------------------------+
//| 魔法号.mq4 |
//| Copyright 2018, MetaQuotes Software Corp. |
//| https://www.geekquant.com |
//+------------------------------------------------------------------+
#property copyright "Copyright 2018, MetaQuotes Software Corp."
#property link "https://www.geekquant.com"
#property version "1.00"
#property strict
//+------------------------------------------------------------------+
//| Expert initialization function |
//+------------------------------------------------------------------+
int OnInit()
{
//--- create timer
EventSetTimer(60);
//---
return(INIT_SUCCEEDED);
}
//+------------------------------------------------------------------+
//| Expert deinitialization function |
//+------------------------------------------------------------------+
void OnDeinit(const int reason)
{
//--- destroy timer
EventKillTimer();
}
//+------------------------------------------------------------------+
//| Expert tick function |
//+------------------------------------------------------------------+
int start()
{
//---- 记住将要分析的指标值
//---- 注意:我们使用第一个和第二个柱。可以是每日的 1-bar 。
//---- (即., 信号晚些显现), 但是保护反复开仓和平仓
//---- 在柱内仓位的
double MACD_1 = iMACD( Symbol(), 0, 12, 26, 9, PRICE_CLOSE, MODE_MAIN, 1 );
double MACD_2 = iMACD( Symbol(), 0, 12, 26, 9, PRICE_CLOSE, MODE_MAIN, 2 );
int _GetLastError = 0, _OrdersTotal = OrdersTotal();
//---- 在开仓中搜索
for ( int z = _OrdersTotal - 1; z >= 0; z -- )
{
//---- 如果在选择仓位中生成错误,转到下一步
if ( !OrderSelect( z, SELECT_BY_POS ) )
{
_GetLastError = GetLastError();
Print( "OrderSelect( ", z, ", SELECT_BY_POS ) - Error #", _GetLastError );
continue;
}
//---- 如果开仓不是当前货币对,略过
if ( OrderSymbol() != Symbol() ) continue;
//---- 如果开仓为BUY,
if ( OrderType() == OP_BUY )
{
//---- 如果 MACD指标看到的零线从上至下,
if ( NormalizeDouble( MACD_1, Digits + 1 ) <0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) >= 0.0 )
{
//---- 平仓
if ( !OrderClose( OrderTicket(), OrderLots(), Bid, 5, Green ) )
{
_GetLastError = GetLastError();
Alert( "错误 OrderClose # ", _GetLastError );
return(-1);
}
}
//---- 如果警报没有改变,退出: 开仓尚早
else return(0);
}
//---- 如果开仓位SELL,
if ( OrderType() == OP_SELL )
{
//----如果MACD指标看到零线从下到上,
if ( NormalizeDouble( MACD_1, Digits + 1 ) >0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) <= 0.0 )
{
//---- 平仓
if ( !OrderClose( OrderTicket(), OrderLots(), Ask, 5, Red ) )
{
_GetLastError = GetLastError();
Alert( "错误 OrderClose # ", _GetLastError );
return(-1);
}
}
//---- 如果警报没有改变,退出:开仓尚早
else return(0);
}
}
//+------------------------------------------------------------------+
//| 如果达到此点,没有仓位开仓 |
//| 检测是否仍有开仓的可能 |
//+------------------------------------------------------------------+
//---- 如果MACD 指标看到零线从下到上,
if ( NormalizeDouble( MACD_1, Digits + 1 ) >0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) <= 0.0 )
{
//---- 开BUY仓
if ( OrderSend( Symbol(), OP_BUY, 0.1, Ask, 5, 0.0, 0.0, "MACD_test", 0, 0, Green ) < 0 )
{
_GetLastError = GetLastError();
Alert( "错误OrderSend # ", _GetLastError );
return(-1);
}
return(0);
}
//----如果MACD 指标看到零线从上至下,
if ( NormalizeDouble( MACD_1, Digits + 1 ) <0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) >= 0.0 )
{
//---- 开 SELL 仓
if ( OrderSend( Symbol(), OP_SELL, 0.1, Bid, 5, 0.0, 0.0, "MACD_test", 0, 0, Red ) < 0 )
{
_GetLastError = GetLastError();
Alert( "错误OrderSend # ", _GetLastError );
return(-1);
}
return(0);
}
return(0);
}
以上代码在运行的时候是没有问题的,但是在执行之后,如果同时自己手动进行交易,这个时候程序就没办法识别是手动单子还是程序下的单子,这个时候就需要对单子进行处理和识别,如何做那,其实就是加一个魔法号数字即可。
//+------------------------------------------------------------------+
//| 魔法号.mq4 |
//| Copyright 2018, MetaQuotes Software Corp. |
//| https://www.geekquant.com |
//+------------------------------------------------------------------+
#property copyright "Copyright 2018, MetaQuotes Software Corp."
#property link "https://www.geekquant.com"
#property version "1.00"
#property strict
extern int Expert_ID = 1234;
//+------------------------------------------------------------------+
//| Expert initialization function |
//+------------------------------------------------------------------+
int OnInit()
{
//--- create timer
EventSetTimer(60);
//---
return(INIT_SUCCEEDED);
}
//+------------------------------------------------------------------+
//| Expert deinitialization function |
//+------------------------------------------------------------------+
void OnDeinit(const int reason)
{
//--- destroy timer
EventKillTimer();
}
//+------------------------------------------------------------------+
//| Expert tick function |
//+------------------------------------------------------------------+
int start()
{
//---- 记住将要分析的指标值
//---- 注意:我们使用第一个和第二个柱。可以是每日的 1-bar 。
//---- (即., 信号晚些显现), 但是保护反复开仓和平仓
//---- 在柱内仓位的
double MACD_1 = iMACD( Symbol(), 0, 12, 26, 9, PRICE_CLOSE, MODE_MAIN, 1 );
double MACD_2 = iMACD( Symbol(), 0, 12, 26, 9, PRICE_CLOSE, MODE_MAIN, 2 );
int _GetLastError = 0, _OrdersTotal = OrdersTotal();
//---- 在全部开仓中搜索
for ( int z = _OrdersTotal - 1; z >= 0; z -- )
{
//---- 如果在搜索仓位中生成错误。转到下一步
if ( !OrderSelect( z, SELECT_BY_POS ) )
{
_GetLastError = GetLastError();
Print( "OrderSelect( ", z, ", SELECT_BY_POS ) -错误 #", _GetLastError );
continue;
}
//---- 如果平仓仓位不是当前货币对,略过
if ( OrderSymbol() != Symbol() ) continue;
//---- 如果MagicNumber 不等于Expert_ID, 忽略此仓位
if ( OrderMagicNumber() != Expert_ID ) continue;
//---- 如果开BUY仓,
if ( OrderType() == OP_BUY )
{
//---- 如果MACD 指标看到零线从上到下,
if ( NormalizeDouble( MACD_1, Digits + 1 ) <0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) >= 0.0 )
{
//---- 平仓
if ( !OrderClose( OrderTicket(), OrderLots(), Bid, 5, Green ) )
{
_GetLastError = GetLastError();
Alert( "错误 OrderClose # ", _GetLastError );
return(-1);
}
}
//---- 如果警报没有改变, 退出: 开仓尚早
else
{ return(0); }
}
//---- 如果开SELL仓,
if ( OrderType() == OP_SELL )
{
//---- 如果MACD指标看到零线从下到上,
if ( NormalizeDouble( MACD_1, Digits + 1 ) >0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) <= 0.0 )
{
//---- 平仓
if ( !OrderClose( OrderTicket(), OrderLots(), Ask, 5, Red ) )
{
_GetLastError = GetLastError();
Alert( "错误 OrderClose № ", _GetLastError );
return(-1);
}
}
//---- 如果警报没有改变,退出: 开仓尚早
else return(0);
}
}
//+------------------------------------------------------------------+
//| 如果执行达到此点,没有仓位开仓 |
//| 检测是否有可能开仓 |
//+------------------------------------------------------------------+
//---- 如果MACD 指标看到零线从下到上,
if ( NormalizeDouble( MACD_1, Digits + 1 ) >0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) <= 0.0 )
{
//---- 开BUY仓
if ( OrderSend( Symbol(), OP_BUY, 0.1, Ask, 5, 0.0, 0.0, "MACD_test",
Expert_ID, 0, Green ) < 0 )
{
_GetLastError = GetLastError();
Alert( "错误OrderSend # ", _GetLastError );
return(-1);
}
return(0);
}
//---- 如果MACD 指标看到零线从上至下,
if ( NormalizeDouble( MACD_1, Digits + 1 ) <0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) >= 0.0 )
{
//---- 开SELL仓
if ( OrderSend( Symbol(), OP_SELL, 0.1, Bid, 5, 0.0, 0.0, "MACD_test",
Expert_ID, 0, Red ) < 0 )
{
_GetLastError = GetLastError();
Alert( "错误 OrderSend # ", _GetLastError );
return(-1);
}
return(0);
}
return(0);
}
添加魔法号之后,当智能交易运行时,用户可以手动开仓。智能交易将不会干涉。
多个相同的智能交易在一种货币对的不同图表上存在这样的状况,相同的智能交易必须在相同货币对的图表中运行,但是需要不同的时间周期。例如,如果我们尝试添加我们的智能交易同时添加到 EURUSD, H1图表和 EURUSD, M30图表,他们将会相互影响:每个都会“考虑”开仓并且会默认管理。这个问题可以通过 Expert_ID 一个协调另一个的方式解决。但是这羊不是很方便。如果很多智能交易在运行,他们的ID必然会混乱。解决这个问题我们需要以 MagicNumber 值作为图表周期。怎样执行呢?如果我们添加图表周期到 Expert_ID, 可能2个不同的智能交易在2个不同的图表上生成相同的 MagicNumber。所以我们最好用 Expert_ID值乘以 10 并且把图表周期(代码从1 到 9, 精确值)放置到最后。如下图表: int Period_ID = 0;
switch ( Period() )
{
case PERIOD_MN1: Period_ID = 9; break;
case PERIOD_W1:Period_ID = 8; break;
case PERIOD_D1:Period_ID = 7; break;
case PERIOD_H4:Period_ID = 6; break;
case PERIOD_H1:Period_ID = 5; break;
case PERIOD_M30: Period_ID = 4; break;
case PERIOD_M15: Period_ID = 3; break;
case PERIOD_M5:Period_ID = 2; break;
case PERIOD_M1:Period_ID = 1; break;
}
_MagicNumber = Expert_ID * 10 + Period_ID;
现在添加这个代码智能交易的 init()函数并且用 Expert_ID替换MagicNumber。
最终版本的智能交易如下://+------------------------------------------------------------------+
//| 魔法号.mq4 |
//| Copyright 2018, MetaQuotes Software Corp. |
//| https://www.geekquant.com |
//+------------------------------------------------------------------+
#property copyright "Copyright 2018, MetaQuotes Software Corp."
#property link "https://www.geekquant.com"
#property version "1.00"
#property strict
extern int Expert_ID = 1234;
int _MagicNumber = 0;
int init()
{
int Period_ID = 0;
switch ( Period() )
{
case PERIOD_MN1: Period_ID = 9; break;
case PERIOD_W1:Period_ID = 8; break;
case PERIOD_D1:Period_ID = 7; break;
case PERIOD_H4:Period_ID = 6; break;
case PERIOD_H1:Period_ID = 5; break;
case PERIOD_M30: Period_ID = 4; break;
case PERIOD_M15: Period_ID = 3; break;
case PERIOD_M5:Period_ID = 2; break;
case PERIOD_M1:Period_ID = 1; break;
}
_MagicNumber = Expert_ID * 10 + Period_ID;
return(0);
}
int start()
{
//---- 记住将要分析的指标值
//---- 注意:我们使用第一个和第二个柱。可以是每日的 1-bar 。
//---- (即., 信号晚些显现), 但是保护反复开仓和平仓
//---- 在柱内仓位的
double MACD_1 = iMACD( Symbol(), 0, 12, 26, 9, PRICE_CLOSE, MODE_MAIN, 1 );
double MACD_2 = iMACD( Symbol(), 0, 12, 26, 9, PRICE_CLOSE, MODE_MAIN, 2 );
int _GetLastError = 0, _OrdersTotal = OrdersTotal();
//---- 在全部开仓中搜索
for ( int z = _OrdersTotal - 1; z >= 0; z -- )
{
//---- 如果在搜索过程中生成错误。转到下一步
if ( !OrderSelect( z, SELECT_BY_POS ) )
{
_GetLastError = GetLastError();
Print( "OrderSelect( ", z, ", SELECT_BY_POS ) - 错误#", _GetLastError );
continue;
}
//---- 如果开仓仓位不是当前货币对,忽略
if ( OrderSymbol() != Symbol() ) continue;
//---- 如果MagicNumber 不等于 _MagicNumber, 忽略此仓位
if ( OrderMagicNumber() != _MagicNumber ) continue;
//----如果开BUY仓,
if ( OrderType() == OP_BUY )
{
//---- 如果MACD指标看见零线从上至下,
if ( NormalizeDouble( MACD_1, Digits + 1 ) <0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) >= 0.0 )
{
//---- 平仓
if ( !OrderClose( OrderTicket(), OrderLots(), Bid, 5, Green ) )
{
_GetLastError = GetLastError();
Alert( "错误 OrderClose # ", _GetLastError );
return(-1);
}
}
//---- 如果警报没有改变,停止: 开新仓位尚早
else return(0);
}
//---- 如果开SELL仓,
if ( OrderType() == OP_SELL )
{
//---- 如果MACD指标看见零线从下到上,
if ( NormalizeDouble( MACD_1, Digits + 1 ) >0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) <= 0.0 )
{
//----平仓
if ( !OrderClose( OrderTicket(), OrderLots(), Ask, 5, Red ) )
{
_GetLastError = GetLastError();
Alert( "错误OrderClose № ", _GetLastError );
return(-1);
}
}
//---- 如果警报没有改变, 停止: 开新仓位尚早
else return(0);
}
}
//+------------------------------------------------------------------+
//| 如果执行达到此点,没有仓位开仓 |
//| 检测是否仍有可能开仓 |
//+------------------------------------------------------------------+
//---- 如果MACD 指标看见零线从下到上,
if ( NormalizeDouble( MACD_1, Digits + 1 ) >0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) <= 0.0 )
{
//---- 开 BUY 仓
if ( OrderSend( Symbol(), OP_BUY, 0.1, Ask, 5, 0.0, 0.0, "MACD_test",
_MagicNumber, 0, Green ) < 0 )
{
_GetLastError = GetLastError();
Alert( "错误 OrderSend # ", _GetLastError );
return(-1);
}
return(0);
}
//---- 如果MACD 指标看到零线从上至下,
if ( NormalizeDouble( MACD_1, Digits + 1 ) <0.0 &&
NormalizeDouble( MACD_2, Digits + 1 ) >= 0.0 )
{
//---- 开 SELL 仓
if ( OrderSend( Symbol(), OP_SELL, 0.1, Bid, 5, 0.0, 0.0, "MACD_test",
_MagicNumber, 0, Red ) < 0 )
{
_GetLastError = GetLastError();
Alert( "错误 OrderSend # ", _GetLastError );
return(-1);
}
return(0);
}
return(0);
}
这样的显示智能交易可以在不同周期的几个图表中使用。
如果需要在相同时间周期和货币对的图表(例如, EURUSD H1 和EURUSD H4)上开启两个智能交易, 此Expert_ID 变量值将被改变。但是这种情况很罕见。
使用以上代码,用户能够改善他的智能交易并且“教会”智能交易区分自己的仓位和他人的仓位。
页:
[1]