MT4平台自带的EA:Moving Average详解
由于本站模板宽度的制约,看着会比较凌乱,建议复制本文内容,粘贴在记事本中,方便阅读。
使用过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();
//----
}
本文地址:http://www.ea666.cn/fxschool/autotrading/mql4/40692.html