MT4平台自带的EA:Moving Average详解

阅读 :
欢迎访问 外 汇 邦 WWW.WaiHuiBang.com

由于本站模板宽度的制约,看着会比较凌乱,建议复制本文内容,粘贴在记事本中,方便阅读。

 

使用过MT4平台的人都知道,MT4平台自带了一个自动交易的EA指标,那就是Moving Average ,下面我们详细解释一下该指标的语句功能,以便各位汇友学习EA的编程。

            //+------------------------------------------------------------------+
            //|                        Moving Average.mq4                       
               |
            //|      Copyright ?2005, MetaQuotes Software Corp.        |
            //|                  http://www.metaquotes.net/                     
              |
            //+------------------------------------------------------------------+
            #define MAGICMA   20050610
            //定义本EA操作的订单的唯一标识号码,由此可以实现在同一账户上多系统操作,各操作EA的订单标识码不同,就不会互相误操作。凡是EA皆不可缺少,非常非常重要!!!
            //宏定义命令#define用法
            extern double Lots             = 0.1;//每单的交易量
            extern double MaximumRisk        = 0.02;//本系统最大可以动用总资金的2%
            extern double DecreaseFactor     = 3;//作者定义的参数,作用要看程序中的用法
            extern double MovingPeriod    = 10;//EA中使用的均线的周期
            extern double MovingShift        =3;//EA中使用的均线向左的K线偏移量
            //extern 确定从外部程序输入的变量, 会直接显现输入数据窗口。数列本身不能作为外部变量。
            //+------------------------------------------------------------------+
            //| Calculate open positions                                        
                |
            //+------------------------------------------------------------------+
            int CalculateCurrentOrders(string symbol)//函数作用,计算当前持仓订单的数量
            {
            int buys=0,sells=0;//定义两个临时变量,准备用于后面的多空订单的个数计算
            //----
            for(int i=0;i<OrdersTotal();i++)//循环检测当前的订单队列,不包含挂单
            {
            if(OrderSelect(i,SELECT_BY_POS,MODE_TRADES)==false)
            break;//挑出持仓单的每一个订单位置
            if(OrderSymbol()==Symbol() &&
            OrderMagicNumber()==MAGICMA)//根据订单位置,比较是否是当前K线商品
            以及订单唯一标识号是否和本程序设置的一致,即判断这个订单是不是当前EA操作的。(用于避免EA误操作其他程序控制的持仓单)
                   {
            if(OrderType()==OP_BUY)   buys++;//找到符合条件的持仓单后,如果是多单,则临时变量buys增加1
            if(OrderType()==OP_SELL) sells++;//找到符合条件的持仓单后,如果是空单,则临时变量sells增加1
            }
            }
            //---- return orders volume
            if(buys>0) return(buys);
            else    return(-sells);//本函数返回查询计算结束时的持仓单的个数.这种模式返回是假设不存在锁单的。
            }
            //+------------------------------------------------------------------+
            //| Calculate optimal lot size                                      
                  |
            //+------------------------------------------------------------------+
            double LotsOptimized()//函数目的,根据要求 计算出订单交易量
            {
            double lot=Lots;
            int orders=HistoryTotal();     // history orders total 历史出场订单的个数
            int losses=0;                // number of losses orders without a
            break
            //---- select lot size
            lot=NormalizeDouble(AccountFreeMargin()*MaximumRisk/1000.0,1);//通过风险系数的计算获得当前入场单应该采用的交易量,除以1000是因为大多货币对汇价都在这个附近。
            //---- calcuulate number of losses orders without a break
            if(DecreaseFactor>0)
            {
            for(int i=orders-1;i>=0;i--)
            {
            if(OrderSelect(i,SELECT_BY_POS,MODE_HISTORY)==false) { Print("Error
            in history!"); break; }//循环查询出场单队列
            if(OrderSymbol()!=Symbol() || OrderType()>OP_SELL) continue;//
            //----
            if(OrderProfit()>0) break;
            if(OrderProfit()<0) losses++;//循环计算所有出场亏损单的亏损总和
            }
            if(losses>1)
            lot=NormalizeDouble(lot-lot*losses/DecreaseFactor,1);//如果亏损额大于1,则下一入场单的交易量修正为新的计算结果。
            }
            //---- return lot size
            if(lot<0.1) lot=0.1;//如果计算出的交易量小于帐户最小手数0.1,则下一入场单的交易手数使用0.1作为交易量
            return(lot);
            }
            //+------------------------------------------------------------------+
            //| Check for open order conditions                                
|
            //+------------------------------------------------------------------+
            void CheckForOpen()//检查入场条件的情况并作处理
            {
            double ma;
            int res;
            //---- go trading only for first tiks of new
            bar,注意下面采用的Volume[0],Open[1]等可以确保交易是在当前周期下k线 收盘价走完才发生的。
            if(Volume[0]>1)
            return;//如果当前K线持仓量(成交量)大于1,说明不是K线的开盘时间点,即当前k线还没收盘确定,则直接返回
            否则是K线第一个价格,则继续下面的过程
            //---- get Moving Average
            ma=iMA(NULL,0,MovingPeriod,MovingShift,MODE_SMA,PRICE_CLOSE,0);//获得当前的均线数值
            //---- sell conditions
            if(Open[1]>ma && Close[1]<ma)   //如当前K开盘价大于均线,而前一K收盘价小于均线,则发出入场多单
            {
            res=OrderSend(Symbol(),OP_SELL,LotsOptimized(),Bid,3,0,0,"",MAGICMA,0,Red);
            return;
            }
            //---- buy conditions
            if(Open[1]<ma && Close[1]>ma)   //如当前K开盘价小于均线,而前一K收盘价大于均线,则发出入场空单
            {
            res=OrderSend(Symbol(),OP_BUY,LotsOptimized(),Ask,3,0,0,"",MAGICMA,0,Blue);
            return;
            }
            //----
            }
            //+------------------------------------------------------------------+
            //| Check for close order conditions                                
            |
            //+------------------------------------------------------------------+
            void CheckForClose()//检查出场条件的情况并作处理
            {
            double ma;
            //---- 只在一个k收盘另一个新出现时交易
            if(Volume[0]>1) return;
            //---- get Moving Average
            ma=iMA(NULL,0,MovingPeriod,MovingShift,MODE_SMA,PRICE_CLOSE,0);
            //----
            for(int i=0;i<OrdersTotal();i++)
            {
            if(OrderSelect(i,SELECT_BY_POS,MODE_TRADES)==false)        break;
            if(OrderMagicNumber()!=MAGICMA || OrderSymbol()!=Symbol()) continue;
            //----确定是不是自己系统下的单子。
            if(OrderType()==OP_BUY)            //多单平仓
            {
            if(Open[1]>ma && Close[1]<ma)
            OrderClose(OrderTicket(),OrderLots(),Bid,3,White);//如果持仓是多单,则当当前K开盘价小于均
            线,而前一K收盘价大于均线,则发出平仓指令
            break;
            }
            if(OrderType()==OP_SELL)         //空单平仓
            {
            if(Open[1]<ma && Close[1]>ma)
            OrderClose(OrderTicket(),OrderLots(),Ask,3,White););//如果持仓是空单,则当当前K开盘价大于
            均线,而前一K收盘价小于均线,则发出平仓指令
            break;
            }
            }
            //----
            }
            //+------------------------------------------------------------------+
            //| Start function                                                  
                        |
            //+------------------------------------------------------------------+
            void start()//主循环过程
            {
            //---- check for history and trading
            if(Bars<100 || IsTradeAllowed()==false) return;
            //---- calculate open orders by current symbol
            if(CalculateCurrentOrders(Symbol())==0) CheckForOpen();
            else                                  CheckForClose();
            //----
            }

欢迎访问 外 汇 邦 WWW.WaiHuiBang.com
本文标题:MT4平台自带的EA:Moving Average详解 - MT4平台MQL4编程学习
本文地址:http://www.ea666.cn/fxschool/autotrading/mql4/40692.html

相关文章

  • 外汇延迟高频交易EA介绍,剥头皮利器
    外汇延迟高频交易EA介绍,剥头皮利器

    关于延迟交易,这里做大概说明。MT4平台的价格都是来源于伦敦国际金融中心,由于各个外汇交易商服务器网络速度等差异,造成不同平台从伦敦金融中心传输到各自MT4平台的报价时间上并不完全一致,报价慢的平台...

    MQL4编程学习
  • MT4—IndicatorCounted函数解析

    指标刚加载到图上的时候IndicatorCounted()是0 然后程序会自动计算一遍所有K线对应的指标数值并画线。然后每来一个新价格的时候IndicatorCounted();就只是1或者2了(视指标的计算方式决定)。 这时候for循环只需要计算这些有变动的K线对应的指标数值就行了,不需要从头到尾重复计算了。 这就是这段程序的目的,找出for循环中仅需当前计算的K线的个数。...

    MQL4编程学习
  • 解析程序化交易的运用与风险管理

    前言:程序化交易的买卖决策完全决定于自己的交易理念系统化、制度化的逻辑判断规则,透过电脑的辅助,将各种交易理念转化为电脑程序语言的一种交易模式,即由电脑来代替人为发出买卖讯号,再根据系统...

    MQL4编程学习
  • 使用EA交易的几种思路

    使用EA交易的朋友,有以下几种思路可以去尝试一下:(1)小止赢(比如10点),大止损(比如1000点),超短线。这样的话能提高成功率,通过大量的小赢来获取胜利,但是要克服偶尔的大亏。(2)利用5%的交易赚取95%的利润。这就是趋势交易法,在趋...

    MQL4编程学习
  • 经典外汇马丁EA研究分析:依兰 Ilan-TrioKS EA
    经典外汇马丁EA研究分析:依兰 Ilan-TrioKS EA

    依兰 Ilan-TrioKS 在网上有源码流传快要十年了,也是网上经典马丁EA之一。EA 本身的界面看起来就是专业的美工设计。源码因为多次修改而有三个并存模块,代码行数近三千行。用 EURUSD M5 随便复盘测试过去三四个月,可以看...

    MQL4编程学习
你可能感兴趣