巫师3希里杀国王:关于 MT 4 (MQL4) 智能交易(EA)程序编辑简介

来源:百度文库 编辑:偶看新闻 时间:2024/05/04 06:16:09
查看文章   关于 MT 4 (MQL4) 智能交易(EA)程序编辑简介-KD2009-07-13 02:10

EA是什么?及他的工作原理

       EA 即 Expert Advisors 的英文缩写,中文意思专家顾问,俗称智能交易系统,就是由电脑模拟交易员的下单操作进行机器自动交易的过程。

一、人工操盘过程

下面我们就以MT4外汇客户端为例,首先来分析一个外汇交易员手工进行外汇交易的操作过程:其步骤如下:

1.打开外汇交易客户端,选定一种货币对图表;
2.监视该货币对的K线趋势图,俗称盯盘,寻找开仓或者是平仓的时机,即开仓或者是平仓的条件
3.如果条件满足,进行下单开仓(做多或者做空)或者平仓
4.重复第二步,继续盯盘,假定第二步是开仓,就是寻找平仓的条件。
5.如果平仓的条件满足,进行平仓操作,计算盈亏核算。完成一次交易的循环。
6.若继续交易,重复2->3->4->5步
7.若不进行交易,退出外汇客户端。

二、机器操盘过程

基于以上的分析,我们已经知道一个完整的智能交易系统(俗称EA)在运行后必须要实现的基本功能,就是上述的人工操 作的1-5步。 这也就是智能交易系统的基本工作过程,所以智能交易系统的工作原理就是由程序员 借助一门计算机程序设计语言,通过编写程序交易 指令模拟人类交易员的行为进行下单操作,实现机器自动进行交易的过程。主要执行过程可分为:盯盘->开仓->再盯盘->平仓,如此循环 执行的过程。
关于支持机器自动交易的平台,目前外汇市场上流行的就是MetaQuotes公司的MT4平台,由于这个平台中嵌入了一种MQL4语言,它提供了对服务器 端的数据访问并可进行交易操作的接口,程序交易者可以根据自己的交易策略来编写自己的自动交易系统,从而实现让机器自动交易,既可以减轻人类的工作量,又 可以克服人类交易中的一此性格弱点,但目前的EA开发,尚所早期起步阶段,有的还存在缺陷,但相信随着技术的发展,机器自动交易终将会逐步取代人类的手工 操作。届时会给交易者一项新的选择。

三、 相关MQL语言知识:

为了实现机器操作,再来看看所需的MQL4语言的相关知识:

1.掌握MQL4语言的基本语法和程序的构成,及运行流程

有关语法部分,请读者参看相关的资料,这里略去。
关于程序的构成,对于一个智能交易系统EA程序来说:主要由三个函数构成分别是:
init():初始化函数,负责程序变量及数据初始输入;只在程序调入时执行一次,一般不用重写内容。
deinit():反初始化函数,负责程序退出时,将数据从内存中清除;只在程序退出时,执行一次,一般不用重写内容。
start():开始函数,也即程序的主函数,负责EA程序 的全部交易执行过程,实际上他是一个EA的交易管理与执行函数。每隔一定时间,一般几秒之内,执行一次,就是循环执行,起到程序退出时终止
运行流程:启动EA后,程序的INTI()开始执行一次,-->然后 START()循环执行--->最后退出EA时deinit()执行一次

2.mql4中与交易相关的交易函数:

开仓函数:
int OrderSend( string symbol, int cmd, double volume, double price, int slippage, double stoploss, double takeprofit, void comment, void magic, void expiration, void arrow_color)
这个功能主要应用于开仓位置和挂单交易.
参量:
symbol - 交易货币对。
cmd - 购买方式。
volume - 购买手数。
price - 收盘价格。
slippage - 最大允许滑点数。
stoploss - 止损水平。
takeprofit - 赢利水平。
comment - 注解文本。
magic - 定单指定码。可以作为用户指定识别码使用。
expiration - 定单有效时间(只限挂单)。
arrow_color - 图表上箭头颜色。如果参量丢失或存在CLR_NONE价格值不会在图表中画出

平仓函数:
bool OrderClose( int ticket, double lots, double price, int slippage, void Color)
对定单进行平仓操作。如果函数成功,返回的值是真实的。如果函数失败,返回的值是假的。获得详细错误信息,请查看GetLastError()函数。
参量:
ticket - 定单编号。
lots - 手数。
price - 收盘价格。
slippage - 最高划点数。
Color - 图表中标记颜色。如果参量丢失,CLR_NONE值将不会在图表中画出。

定单修改函数:
bool OrderModify( int ticket, double price, double stoploss, double takeprofit, datetime expiration, void arrow_color)
对于先前的开仓或挂单进行特性修改。如果函数成功,返回的值为 TRUE。如果函数失败,返回的值为FALSE。 获得详细的错误信息,查看 GetLastError()函数。

参量:
ticket - 定单编号。
price - 收盘价格
stoploss - 新止损水平。
takeprofit - 新赢利水平。
expiration - 挂单有效时间。
arrow_color - 在图表中允许对止损/赢利颜色进行修改。如果参量丢失或存在CLR_NONE 值,在图表中将不会显示。

四、源码的交易流程分析

下面的源码是一个基于KD指标的智能交易系统的代码,整个程序非常简洁但EA的功能又非常齐全,实现了完全由电脑自动下单和平仓,程序代码分析参看代码中的相关注释

 

//+------------------------------------------------------------------+
//|                                                           KD.mq4 |
//|                                                    QQ: 412968847 |
//|                                   http://hi.baidu.com/qianlima88 |
//+------------------------------------------------------------------+
#property copyright "大熊EA系统-Sto-KD"
#property link      "大熊@大熊.com"
#define MAGICKDJ    20090710
//---- input parameters
extern int whichmethod = 2;//1;//4;    //1: no S/L,no T/P   不设止赢也不设止损
//2: no S/L,has T/P 设止损,但不设止赢
//3: has S/L,no T/P 设止赢,不设止损
//4: has T/P has S/L,设止赢也设止损
extern double    Lots=0.1;             //开仓量0.1
extern double    MaximumRisk=0.6;      //开仓占可用资金比例0.5
extern double    DecreaseFactor=0.3;   //开仓亏损减少下单手数0.3
extern int        tp=80;               //最大赢利 止赢点数200
extern int        sl=40;               //最大损失 止损点数100
extern int       Leverage=100;         //交易倍数1:100
//+------------------------------------------------------------------+
//| expert start function                                            |
//+------------------------------------------------------------------+
int 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();
return(0);
}
//+------------------------------------------------------------------+
//| Calculate open positions                                         |
//+------------------------------------------------------------------+
int CalculateCurrentOrders(string symbol)
{
int buys=0,sells=0;
//----
for(int i=0;i{
if(OrderSelect(i,SELECT_BY_POS,MODE_TRADES)==false) break;
if(OrderSymbol()==Symbol() && OrderMagicNumber()==MAGICKDJ)
{
if(OrderType()==OP_BUY) buys++;
if(OrderType()==OP_SELL) sells++;
}
}
//---- return orders volume
if(buys>0) return(buys);
else       return(-sells);
}

//+------------------------------------------------------------------+
//| Check for open order conditions                                  |
//+------------------------------------------------------------------+
void CheckForOpen()
{
//double ma;
int    res;
double point =MarketInfo(Symbol(),MODE_POINT);
Print("point=",point);
//---- go trading only for first tiks of new bar
if(Volume[0]>1) return;
double valK=iStochastic(NULL,0,5,3,3,MODE_SMA,0,MODE_MAIN,0); //当周期的K值
double valD=iStochastic(NULL,0,5,3,3,MODE_SMA,0,MODE_SIGNAL,0); //当周期的D值
//---- sell conditions
if(valD > 80 && valK < valD) //k下穿D K{
switch (whichmethod)
{
case 1:   res=OrderSend(Symbol(),OP_SELL,LotsOptimized(),Bid,3,0,0,"开空仓",MAGICKDJ,0,Red); break;
case 2:   res=OrderSend(Symbol(),OP_SELL,LotsOptimized(),Bid,3,Bid+sl*point,0,"开空仓",MAGICKDJ,0,Red);break;
case 3:   res=OrderSend(Symbol(),OP_SELL,LotsOptimized(),Bid,3,0,Bid-tp*point,"开空仓",MAGICKDJ,0,Red); break;
case 4:   res=OrderSend(Symbol(),OP_SELL,LotsOptimized(),Bid,0,Bid+sl*point,Bid- tp*point,"开空仓",MAGICKDJ,0,Red); break;
default : res=OrderSend(Symbol(),OP_SELL,LotsOptimized(),Bid,3,0,0,"",MAGICKDJ,0,Red); break;
}
if (res <=0)
{
int error=GetLastError();
if(error==134) Print("Received 134 Error after OrderSend() !! "); // not enough money
if(error==135) RefreshRates();                                     // prices have changed
if(error==131) Print("Received 131 Error after OrderSend() !! "); // not enough money

}
//Sleep(5000);
return;
}
//---- buy conditions
if(valD < 20 && valK > valD) //k上穿D k>d 开多仓
{
switch (whichmethod)
{
case 1:   res=OrderSend(Symbol(),OP_BUY,LotsOptimized(),Ask,3,0,0,"开多仓",MAGICKDJ,0,Blue);break;
case 2:   res=OrderSend(Symbol(),OP_BUY,LotsOptimized(),Ask,3,Ask-sl*point,0,"开多仓",MAGICKDJ,0,Blue);break;
case 3:   res=OrderSend(Symbol(),OP_BUY,LotsOptimized(),Ask,3,0,Ask+tp*point,"开多仓",MAGICKDJ,0,Blue);break;
case 4:   res=OrderSend(Symbol(),OP_BUY,LotsOptimized(),Ask,0,Ask-sl*point,Ask+tp*point,"开多仓",MAGICKDJ,0,Blue);break;
default : res=OrderSend(Symbol(),OP_BUY,LotsOptimized(),Ask,3,0,0,"开多仓",MAGICKDJ,0,Blue);break;
}

if (res <=0)
{
error=GetLastError();
if(error==134) Print("Received 134 Error after OrderSend() !! "); // not enough money
if(error==135) RefreshRates();   // prices have changed
}
Sleep(5000);
return;
}
//----
}
//+------------------------------------------------------------------+
//| Check for close order conditions                                 |
//+------------------------------------------------------------------+
void CheckForClose()
{
double ma;
//---- go trading only for first tiks of new bar
if(Volume[0]>1) return;
double valK=iStochastic(NULL,0,5,3,3,MODE_SMA,0,MODE_MAIN,0); //当周期的K值
double valD=iStochastic(NULL,0,5,3,3,MODE_SMA,0,MODE_SIGNAL,0); //当周期的D值
//----
for(int i=0;i{
if(OrderSelect(i,SELECT_BY_POS,MODE_TRADES)==false)        break;
if(OrderMagicNumber()!=MAGICKDJ || OrderSymbol()!=Symbol()) continue;
//---- check order type
if(OrderType()==OP_BUY)
{
if(valK >80 && valK < valD) //k下穿D K
OrderClose(OrderTicket(),OrderLots(),Bid,3,White); Sleep(5000);
break;
}
if(OrderType()==OP_SELL)
{
if(valK <20 && valK > valD) //k上穿D or k>d 空方盘平仓
OrderClose(OrderTicket(),OrderLots(),Ask,3,White); Sleep(5000);
break;
}
}
//----
}
//+------------------------------------------------------------------+
//| 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*Leverage/1000000.0,1);
//---- 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);
}
//---- return lot size
if(lot<0.1) lot=0.1;
return(lot);

}
//+------------------------------------------------------------------+
//| 请勿用于真实帐户使用,并非稳定盈利交易系统,技术交流:           |
//+------------------------------------------------------------------+

五、本文结论

从以上的分析,可以知道,所谓EA,就是由电脑模拟交易员的下单操作进行机器交易的过程, 具体步骤如下:

1.当用户打开外汇客户端程序后,由客户端程序调入用户在系统内已预置好的EA交易系统程序。
2.当EA程序启动后便开始对图表中货币对的K线趋势图,进行监视,寻找开仓的条件;
3.如果条件满足,进行下单开仓(做多或者做空);
4.重复第二步,继续盯盘,假定第二步是开仓,就是寻找平仓的条件。
5.如果平仓的条件满足,进行平仓操作,计算盈亏核算。完成一次交易的循环。
6.若继续交易,EA重复2->3->4->5步
7.用户若不想让EA进行交易,可通过相关菜单操作设置禁用EA,或者退出外汇客户端。