geek005 发表于 2019-10-28 22:51:59

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]
查看完整版本: MAGICNUMBER: 定单的“魔法“识别符