龙听期货论坛's Archiver

龙听 发表于 2017-10-30 14:51

[转载]MT4编程入门4-11

MT4编程入门4:自定义函数
自定义函数与数学函数在本质上是一致的
例如:数学函数ft(x,y)=3x+2y (x,y为正整数) 写成程序语句就是:
int ft(int x,int y)
    {
     int temp;
     temp=3*x+2*y;
     return(temp);
    }

一、和自定义变量一样,自定义函数在使用前,也必须先设立,以定义其类型、名称、参数和运算语句。
函数类型、名称、参数位于函数头部(参数间以逗号分隔),
大括号中的是运算语句,也就是函数体部分。
return语句为函数结束语句,return后面括号中的值为函数返回值,即函数运算结果。

上例中,int为函数类型,表示该函数的返回值(运算结果)是整数;
(x,y)为参数,由于自定义变量使用之前都必须先创建,所以要加上类型定义词int
ft为函数名,有了函数名,我们就可以在别处用ft(2,1)的形式来调用此函数;
函数体部分有三句:
第一句,定义一个整形变量temp
第二句,计算temp的值
第三句,函数结束,返回调用点,以temp的值作为本函数返回值。
此函数也可以简写为:int ft(int x,int y)  {return(3*x+2*y);}

二、函数类型与类型符
函数类型就是函数返回值(运算结果)的类型,与自定义变量类型差不多
有整型(int)、双精度型(double)、字符串型(string)、逻辑型(bool)等,
还有一种函数是以函数运行为目的,运行结束后不需要返回值也没有返回值的,则函数类型可以写成void(无值型)

三、函数中如果定义了自定义变量,那么此变量仅在本函数内部存在,函数运行结束时,自定义变量也就自动注销。
上例中,有x、y、temp三个自定义变量,在函数运行时创建,在函数运行结束时自动消失,仅在函数内部有效,与函数外部无关。

四、函数的调用
函数调用非常简单,只要函数名正确、参数个数一致、类型相同就能正确调用
如我们前面创建了ft(x,y)函数,则别处我们可以用ft(4,5)等样式来调用ft(x,y)函数求值。
调用时,传递参数值给ft(x,y)函数(按顺序把4传给x,把5传给y),返回结果23后继续执行主程序。


MT4编程入门5:自定义数组
一、数组是一系列同类型变量的集合,可以同时存储多个数值。
例:int aa[4]={4,3,2,1};
这是一个数据名为aa、包含了4个元素的整数型数组(即数组长度为4),可以同时存储4个数值。

二、自定义数组也必须“先建立,后使用”
创建(定义)时,必须设定类型和名称。
创建(定义)时,数组的元素个数也可以设定,也可以不设定
数组中第一个元素的编号为0,第二个为1,以此类推。编号使用方括号括起来
在这上例中aa[0]的值为4、aa[1]的值为3、aa[2]的值为2、aa[3]的值为1

创建(定义)数组的同时,可以赋初值,也可以不赋初值。
例:
int bb[];   创建数组bb,不设定数组元素个数
int cc[8];  创建数组cc,设定数组元素为8个,不赋初值
int dd[3]={9,8,7};  创建数组dd,设定数组元素为4个,并赋初值
赋值后,dd[0]的值为9、dd[1]的值为8、dd[2]的值为7
int ee[9]={0,1,2,3};   创建数组ee,设定数组元素为9个,但只给前4个元素赋初值

三、数组使用时的赋值
数组的赋值必须指定明确的赋值对象,除了新建数组时可以批量赋值外,数组元素必须一个一个赋值
例如:dd[2]=1;
这是对数组dd的第3个元素dd[2]([0]是第1个,[2]表示第3个)重新赋值为1
在这里,没有整数型变量定义符int,
这样,数组dd原有三个值{9,8,7}就变为{9,8,1}

四、数组的引用
数组只是一系列变量的集合,所以每次只能使用数组中的一个元素。
数组的引用必须指定明确的引用对象,一次只能引用一个。如果需要使用整列数组,则必须逐个使用。
例如:数组aa[4]的值为{4,3,2,1}
我们用aa来引用数组aa中的值
当i=0时,aa[i]的值为4
当i=1时,aa[i]的值为3
当i=2时,aa[i]的值为2
当i=3时,aa[i]的值为1
这样,我们在处理数列的时候,使用数组就能带来极大的方便

MT4编程入门6:市场数据取值
一、预定义数组(MT4已经定义好的,可以直接使用)
开盘价、最高价、最低价、收盘价、成交量等历史数据,每根K线蜡烛都各有一个,所以必须用数组来解决问题,MT4有几个预定义数组:
开盘价Open[]、最高价High[]、最低价Low[]、收盘价close[]、成交量Volume[]、所属时间Time[]
类型为双精度double型(即精确小数)

这里有一个位置的问题,右边第一根K线蜡烛(即最新的蜡烛)的编号为0,第二根蜡烛编号1,第三根蜡烛编号2,也就是说从右向左倒着数过去。

Open[0]、High[0]、Low[0]、Close[0],表示最新的开盘价、最高价、最低价、收盘价
Open[1]、High[1]、Low[1]、close[1],表示第2根K线蜡烛的开盘价、最高价、最低价、收盘价
Open[2]、High[2]、Low[2]、close[2],表示第3根K线蜡烛的开盘价、最高价、最低价、收盘价
Open[3]、High[3]、Low[3]、close[3],表示第4根K线蜡烛的开盘价、最高价、最低价、收盘价
Open、High、Low、close,表示第i+1根K线蜡烛的开盘价、最高价、最低价、收盘价
以此类推。。。。。。

注意:这些是数组,用的是方括号。

二、预定义变量
买入价、卖出价是实时价格,MT4用预定义变量Ask和Bid表示,数值类型为double双精度
还有一些预定义变量,如:
Bars 表示图表中的蜡烛数,类型为int整数型
Digits 表示当前货币对的小数位,类型为int整数型,无日元币对为4,有日元币对为2,黄金石油等一般也为2
Point 表示当前货币对的点值,类型为双精度double型,无日元币对为0.0001,有日元币对为0.01。与Digits正好相反。

三、指标函数
1、价格、成交量、时间
它们都有三个参数:货币对名、K线周期、位置序号
开盘价:iOpen(symbol,timeframe,shift)    双精度double型
收盘价:iClose(symbol,timeframe,shift)    双精度double型
最高价:iHigh(symbol,timeframe,shift)    双精度double型
最低价:iLow(symbol,timeframe,shift)    双精度double型
成交量:iVolume(symbol,timeframe,shift)  双精度double型
所属时间:iTime(symbol,timeframe,shift)  日期时间datetime型

K线周期为:1分钟图(PERIOD_M1)、5分钟图(PERIOD_M5)、15分钟图(PERIOD_M15)、30分钟图(PERIOD_M30)、
1小时图(PERIOD_H1)、4小时图(PERIOD_H4)、日线图(PERIOD_D1)、周线图(PERIOD_W1)、周线图(PERIOD_W1)、月线图(PERIOD_W1)

例:
iOpen("USDJPY",PERIOD_H1,0)   表示美元兑日元1小时图最新K线蜡烛的开盘价
iClose("EURUSD",PERIOD_H4,2)  表示欧元兑美元4小时图第3根K线蜡烛的收盘价
iClose("GBPUSD",PERIOD_H1,i)  表示英磅兑美元1小时图第i+1根K线蜡烛的收盘价
iHigh(NULL,0,0)               既不指定商品,也不指定K线周期,用在谁就是谁,用在哪就是哪

2、移动平均值。双精度double型
iMA(symbol, timeframe, period, ma_shift, ma_method, applied_price, shift)
参数共7个,分别为:商品名称、K线周期、均线周期、均线偏移、平均模式、价格种类、位置
均线周期:10天平均线的均线周期为10,20天均线的均线周期为20
均线偏移:均线位置整体左右移动的位置偏移量
平均模式:简单移动平均(MODE_SMA)、指数移动平均(MODE_EMA)、平滑移动平均线(MODE_SMMA)、线性加权移动平均线(MODE_LWMA)
价格种类:收盘价(PRICE_CLOSE)、开盘价(PRICE_OPEN)、最高价(PRICE_HIGH)、最低价(PRICE_LOW)、中值(PRICE_MEDIAN)、5(PRICE_TYPICAL)、6(PRICE_WEIGHTED)

例1:iMA("EURUSD",PERIOD_H1,20,0,MODE_SMA,PRICE_CLOSE,0)
表示:欧元1小时图上,以收盘价计算的,20小时简单移动平均线,最新K线所对应位置的值
例2:iMA(NULL,0,20,0,MODE_EMA,PRICE_CLOSE,2)
表示:在当前商品、当前K线周期图表上,以收盘价计算的,20(天)指数移动平均线 第3根K线所对应位置的值

其他如MACD指标、威廉指标、RSI、SAR、布林线等指标取值都与移动平均线指标相类似


3、在数组中求元素的移动平均值。双精度double型
iMAOnArray(数组名, 总数, 平均周期, 均线偏移, 平均模式, 位置)
这也与iMA函数差不多,不过数据源变为数组

从数组中不但可以求得移动平均值,还可以求得RSI等指标值


4、求自定义指标的值
我们经常自己编一些自定义指标,可用iCustom函数来取得自定义函数的值
iCustom(商品名,K线周期,自定义指标名称,自定义指标参数1,参数2,参数3,,,自定义指标线编号,位置)
如果自定义指标只有一根指标线,则自定义指标线的编号为0。
如果自定义指标有多根指标线,则第一条自定义指标线的编号为0,第二条为1,第三条为2。。。
例如:iCustom(NULL,0,"mymacd",12,26,9,2,0)   (12,26,9)为自定义指标mymacd的三个参数
表示:求当前图表中,自定义指标mymacd(12,26,9)的第3条指标线在最新位置的值

抛砖引玉,这里只是有代表性地列了几个函数,详细请查阅《MT4编程手册》

MT4编程入门7:判断语句
一、if语句
if语句很常用,也比较简单。

规则:如果小括号中的条件成立,则执行大括号中的语句;如果不成立,则跳过大括号。
例如:
if(a==1)
    {
     b=c+1;
    }
我们在编写报警指标的时候,就经常用到这一语句:
如果“价格向上达到指定价位”,则“报警”
如果“MACD上穿”,则“报警”
如果“均线金叉”,则“报警”。等等


例:

int mark=0;

if( High[1]<1.0000 && High[0]>=1.0000 && mark!=1)
    {
     Alert(symbol(),"价格向上触及1.0000");
     mark=1;
    }
if( Low[1]>1.0000 && Low[0]<=1.0000 && mark!=2)
    {
     Alert(symbol(),"价格向下触及1.0000");
     mark=2;
    }

这是一个价格上、下穿1.0000时报警的判断语句:
上穿报警条件:当第二根K线最高价小于1.0000,并且最新K线最高价大等于1.0000
下穿报警条件:当第二根K线最低价大于1.0000,并且最新K线最低价小等于1.0000



这里,mark是用作报警标记,mark的初值是0,
当上穿报警后,mark的值就改为1;当下穿报警后,mark的值就改为2;

当mark=0时,说明从未报过警,上、下穿都能报警;
当mark=1时,说明已经上穿报过警了,不能再上穿报警了,但仍可下穿报警;
当mark=2时,说明已经下穿报过警了,不能再下穿报警了,但仍可上穿报警。

二、if ... else语句
规则:如果小括号中的条件成立,则执行if下大括号中的语句1;如果不成立,则执行else下大括号中的语句2。
if(条件)
  {
  语句1;
  }
else
  {
  语句2;
  }


三、注意事项
1、只有语句后面才用到语句结束符“;” 条件、大、小括号后面都不用“;”
2、语句用大括号括起来,但如果只有一行语句,则大括号可以省略

              if(条件)    语句1;

MT4编程入门8:循环
一幅K线图有几千上万条K线,每根K线又各有开收盘价、最高低价等数值,而且还有很多移动平均线、MACD、RSI等指标值。
面对海量数据,我们必须用循环来实现数据的取值和赋值。

一、while循环
while(条件)
   {
   语句1
   语句2
   。。。
   }
规则:当小括号中的条件成立时,就执行大括号中的语句,执行完了再判断条件是否成立,如果条件成立就继续执行大括号中的语句。

只要条件成立,程序就不停地运行大括号中的语句(循环体),直到小括号中的条件不再成立时结束循环。

它与if语句的区别是:if语句是当条件为真时运行一次;而while语句则是只要条件为真,循环体语句就不停地运行,直到条件为假时结束循环。


例:
   extern int 快线=12;
   extern int 慢线=26;
   double buffer[];
   int i=0;

   while(i<1000)
     {
      buffer=  iMA(NULL,0,快线,0,MODE_EMA,PRICE_CLOSE,i)
                 -iMA(NULL,0,慢线,0,MODE_EMA,PRICE_CLOSE,i);
      i++;
     }

这里,循环执行条件是i<1000,循环体中有两个语句,一句是把两条均线的差值赋给数组buffer,另一句“i++;”是每运行一次,i的值在原有基础上增加1。这样,随着循环的不断运行,i的值逐渐增加,循环1000次后,i的值也就从0变为1000,此时“i<1000”就不再成立,循环结束。这样,数组buffer[]中也就有了1000个值。


二、for循环
for循环与while循环在原理上是一致的,只是书写格式上有所区别

把上面的例子改成for语句:

   extern int 快线=12;
   extern int 慢线=26;
   double buffer[];

   for(int i=0; i<1000; i++)
     {
      buffer=  iMA(NULL,0,快线,0,MODE_EMA,PRICE_CLOSE,i)
                 -iMA(NULL,0,慢线,0,MODE_EMA,PRICE_CLOSE,i);
     }

与前面相比,我们注意到:
“int i=0;”语句从“while(i<1000)”上方移到循环条件“i<1000”的前面;
“i++;”语句从循环体中移到了循环条件“i<1000”的后面;

循环执行顺序与while循环一致:

第一步、先执行小括号中的第1句:int i=0;
(此语句用来定义初始变量,在循环中仅执行一次,可以为没有任何表达式的空操作符“;”)

第二步、再判断小括号中的第2句是否成立:i<1000;
如果不成立,则循环结束;如果成立,则循环继续运行,执行第三步

第三步、按顺序执行大括号中的语句

第四步、执行小括号中的第1句:i++ (此语句即:每执行一次,i的值增加1)
然后回到第二步继续执行循环。



再补充一点:如果循环体中(大括号中)只有一行语句,大括号可以省略

MT4编程入门9:MT4自定义指标的结构
MT4自定义指标一般由四个部分构成:

(1)文件头部
(2)参数、变量和数组的定义
(3)初始化函数init()
(4)主函数start()


一、文件头部,也称为预处理程序
预处理程序以“#”开头,行尾无语句结束符“;”
常用的预处理程序有:
1、#property  indicator_chart_window
把指标显示在主图。如:均线、SRA等类指标用到此语句

2、#property indicator_separate_window
把指标显示在副图。如:MACD、RSI、威廉等类指标用到此语句

3、#property indicator_buffers 3
显示3根指标线

4、#property indicator_color1 Red
第1根指标线的颜色为Red

5、#property  indicator_width1  1
第1根指标线的粗细分别为1

6、#property indicator_level1   0.00
在0.00值位置横划1条虚线


二、参数、变量和数组的定义

全局性的参数、变量、数组在此定义,局部变量可在start()函数中定义


三、初始化函数init()
init()在自定义指标加载时运行一次。

初始化函数的功能是“设置”。如果自定义指标需要划线,则必然用到此函数


四、主函数start()

当数据有变动时,start()就被触发。数据变动一次,start()就运行一次。
自定义指标的编程主要依靠此函数进行。

start()函数的作用主要是取值和给指标线赋值,报警也在此函数内发起。


另外,还有一个反初始化函数deinit()
deinit()在自定义卸载时运行一次,可用以去除指标加载时init()所做的初始化操作。

MT4编程入门10:画一条指标线
要画指标线,只要在程序中写明以下几点就可以了:
第一、明确指标线所在窗口,是主图还是副图
第二、要建立数组,用以保存指标线在各个位置的值。
第三、要建立指标线与数组的对应关系,哪个数组对应哪条指标线
第四、要明确指标线的线型,是曲线还是柱线或者是箭头
第五、如果指标线是箭头,还要说明是哪种箭头
第六、给数组赋值
其中:
第一、二条写在文件头部中,
第三、四、五条写在init()函数中(init函数仅在指标加载时运行一次)
第六条写在start()函数中(start函数在数据发动变动时运行,变动一次运行一次)

下面以MACD为例说明
我们知道,MACD指标由二条曲线和一组红绿柱线组成。(下图一)
其中:
白线是二根均线的差;
紫线是白线的移动平均线;
红绿柱线则是白线和紫线的差,白线上穿紫线,出现红柱,下穿则出现绿柱。

我们从简单入手,先去除紫线和红绿柱线,仅保留其中的那根白线,来看白线是怎样画出来的。

下面是全部语句:

#property indicator_separate_window
#property indicator_color1  White
#property indicator_level1  0
extern int FMA=12;
extern int SMA=26;
double     buf[];
int init()
  {
   SetIndexBuffer(0,buf);
   SetIndexStyle(0,DRAW_LINE);
   return(0);
  }
int start()
  {
   int limit=Bars-IndicatorCounted();

   for(int i=0; i<limit; i++)
    {
    buf=
        iMA(NULL,0,FMA,0,1,0,i)
       -iMA(NULL,0,SMA,0,1,0,i);
    }
   return(0);
  }
说明如下:



==============================================

==============================================

以下为上述语句的简要说明

#property indicator_separate_window
指标放在副图

#property indicator_color1  White
第一条指标线为白色

#property indicator_level1  0
在副图中零值位置上画一条水平横线,

extern int FMA=12;
extern int SMA=26;
设立二个整数型变量,默认值为12和26,允许外部修改值

double     buf[];
设立一个数组

int init()
初始化函数。该函数在指标加载时只运行一次。init是系统默认的函数名,但使用时仍需要进行创设,所以要加定义符int
  {
   SetIndexBuffer(0,buf);
   设置数组buf为第一条指标线

   SetIndexStyle(0,DRAW_LINE);
   设置第一条指标线线型为连续曲线

   return(0);
   函数结束语句
  }

int start()
指标触发函数。与init函数不同,该函数在有数据变化时被触发,如果数据被不断更新,则该函数将不断执行。start也是系统默认的函数名,但使用时也仍然需要进行创设,所以也要加定义符int

  {
   int limit=Bars-IndicatorCounted();
   自定义一个变量limit,并赋值
   Bars是图表中的柱数
   IndicatorCounted()缓存中的柱数,就是已经计算过的有值的柱数
   这样limit的值就是未经计算的柱数,这样就可以起到优化程序的作用。

   for(int i=0; i<limit; i++)
   循环语句。
   循环从i=0开始,每循环一次i值增加1,一直循环到i<limit不满足时结束循环
   由于循环变量i为一个新变量,所以要先定义,加上整型变量定义符int
   下面大括中为循环体,此例中只一条语句
    {
    buf=
        iMA(NULL,0,FMA,0,1,0,i)
       -iMA(NULL,0,SMA,0,1,0,i);
    }
   给数组buf赋值,其值分别为相应位置上两条均线的差
   i是水平位置序号值,即烛柱从右到左的序号,右边第一个烛柱序号为0

   return(0);
   start函数结束
  }
MT4编程入门11:MT4的报警
报警功能是MT4的一大特色。它可以在预定的条件达到时,发出警报。


与指标画线相比,报警语句显得非常简单,
只要在判断语句中加一个报警语句即可

报警方式有:弹出窗口报警、音乐报警、邮件报警等。
如果邮箱开通了手机短信通知,则邮件报警的内容会即时转发到手机上。

1、弹出窗口报警:
当(条件达到)执行此语句时,以弹出窗口警告。
格式:Alert(内容1,内容2,内容3,内容4);
报警内容为字符串型,内容之间加逗号
例如:
Alert( Symbol(),"4小时图MACD上穿零轴");


2、音乐报警:
当(条件达到)执行此语句时,播放一段音乐。
格式:PlaySound("音乐文件名.wav");
文件类型为wav格式,并且保存在C:Program FilesMetaTrader4sounds目录中
文件名加引号

3、邮件报警:
当(条件达到)执行此语句时,发送一个邮件。
(收发件人地址在MT4系统中设置详见《MT4编程实例1:一个简单的小程序,让你的手机摇身变成外汇行情接收机》)
格式:SendMail(标题1+标题2, 内容1+内容2);
标题之间以加号连接,内容之间也以加号连接
邮件标题和邮件内容以逗号间隔



下面是《价格穿越某均线报警》举例


+---------------------------------
#property indicator_chart_window
extern int 警戒均线=20;
int mark=0;
int start()
{
    if(   iHigh(0,0,0) >= iMA(0,0,警戒均线,0,MODE_SMA,PRICE_CLOSE,0)
       && iHigh(0,0,1) <  iMA(0,0,警戒均线,0,MODE_SMA,PRICE_CLOSE,1)
       && mark != 1   )
          {
            Alert(Symbol(),"向上触及30均线");
            mark = 1;
          }

    if(   iLow(0,0,0) <= iMA(0,0,警戒均线,0,MODE_SMA,PRICE_CLOSE,0)
       && iLow(0,0,1) >  iMA(0,0,警戒均线,0,MODE_SMA,PRICE_CLOSE,1)
       && mark != 2   )
          {
            Alert(Symbol(),"向下触及",警戒均线,"均线");
            mark = 2;
          }
    return(0);
}

+---------------------------------

部分语句说明:


#property indicator_chart_window
此句是把程序放在主图,当然这此例中放在副图也一样

extern int    定义一个外部参数变量,整数型,允许外部值修改
int                定义一个整数型变量
int start()    定义触发函数
if()                判断
iHigh()        最高价值函数
iLow()         最低价值函数
iMA()           移动平均线值函数
Alert()         报警函数
Symbol()    商品名称函数
&&              逻辑运算符“并且”
!=                逻辑运算符“不等于”
MODE_SMA     简单移动平均模式
PRICE_CLOSE  以收盘价计算

再说一下自定义变量mark的作用:

mark的初值是0,当上穿报警时给mark赋值1,当下穿报警时给mark赋值2。

这样当mark的值为1时,说明已经对上穿报过警了,就不能再次对上穿报警;

当mark的值为2时,说明已经对下穿报过警了,就不能再次对下穿报警。

这样就起到了消除重复报警的作用。

页: [1]