(原创:Alex) 一、全局变量简介 除了投资组合交易中的PMM全局变量关键字(PMM全局变量字只有pro版本和精英版才有)之外,还有常规的GV全局变量函数,为了区分这两类全局变量,我们暂且称PMM全局变量关键字和GV全局变量函数;由PMM关键字新建的全局变量,只能由PMM关键字去存储和调用,但是不同投资组合工作中的PMM全局变量与其它投资组合工作区之间的PMM全局变量是相互独立的,也就是投资组合工作区A和投资组合工作区B中有相同名称的PMM全局变量(假设都是”name1”),但是投资组合工作区A中的PMM关键字无法取到投资组合工作区B中的全局变量”name1”中存储的值;当然,也无法在图表或者市场扫描器中取到投资组合工作区中的PMM全局变量的值,总而言之,投资组合工作区A中的PMM全局变量的有效性仅在投资组合工作区A中有效。 与PMM全局变量不同的是,GV全局变量在MC中的任何地方都有效。GV全局变量是指由GV全局变量函数新建的全局变量,而GV全局变量函数是函数,函数的使用需要在MC的两个脚本中(指标和信号)调用函数;我们可以在脚本test1(指标或者信号)中使用GV全局变量函数新建GV全局变量”name2”,然后在脚本test2(指标或者信号)中使用GV全局变量函数调用GV全局变量”name2”中的值,然后我们可以将脚本test1插入到图表上、市场扫描器(市场扫描器中不能使用信号脚本,只可以使用指标脚本)、投资组合工作区中,我们只需要在test1中向GV全局变量”name2”中存储一个值,那么test2中使用GV全局变量函数就可以取到GV全局变量”name2”中的值,这样test1和test2就可以进行“沟通”,也就是信息传递。 二、GV全局变量代码解析 GV全局变量函数按功能可以分为存储和调用两类,按数据类型可以分为整型和浮点型,具体如下表所示: GV全局变量函数 GV全局变量函数 | 功能 | 数据类型 | 用法 | GVSetNamedInt("name1",value) | 存储 | 整型 | 存储整型数值value到全局变量”name1”中 | GVGetNamedInt("name1",-1) | 调用 | 整型 | 从全局变量”name1”中调用整型数值,并且返回该值 | GVSetNamedDouble("name2",value) | 存储 | 浮点型 | 存储浮点数值value到全局变量”name2”中 | GVGetNamedDouble("name2",-2) | 调用 | 浮点型 | 从全局变量”name2”中调用浮点数值,并且返回该值 |
GV全局变量在这里有两点需要注意一下: Ø
GVGetNamedInt("name1",-1)和GVGetNamedDouble("name2",-2)分别是用于调用全局变量”name1”和”name2”中的值,这两个全局变量的初始值分别是-1和-2,我们可以理解成当我们未通过存储类GV全局变量函数向全局变量中存储数值时,通过调用类GV全局变量函数会返回GV全局变量的初始值。 Ø
GV全局变量的命名是通过字符类型进行命名的,需要有两个双引号,如上面的”name1”和”name2”两个GV全局变量。 Ø
整型GV全局变量与浮点型GV全局变量是相互独立的;例如,通过函数GVSetNamedInt("name1",value)将变量值value存储到全局变量”name1”中,但是通过函数GVGetNamedDouble("name1",-2)返回的值是-2,而通过GVGetNamedInt("name1",-1)返回的值是value值。 三、GV全局变量回测 GV全局变量除了可以用于实时行情中,也可以用于历史数据的回测,但是对于历史回测是有限制的。GV全局变量可以用于同一个策略不同信号之间(图表中)历史回测,也可以用于投资组合回测中,而目前投资组合回测我们可以使用最新的PMM全局关键字,所以这里我们仅对GV全局变量在图表中的历史回测进行叙述,GV全局变量在投资组合回测中的原理类似,这里不再赘述。 一个策略是由若干个信号脚本组成的,假设这里信号test3和test4组合成一个图表策略(test3和test4的执行顺序使用默认的排序,即每根bar先执行test3,后执行test4),如下图所示:
图1. 信号test3和test4
//test3 GVSetNamedDouble("gn",close); print("test3 ",",currentbar=",currentbar,",close=",close); //test4 value1=GVGetNamedDouble("gn",-1); print("test4 ",",currentbar=",currentbar,",value1=",value1); 以上是信号test3和test4中的代码,test3中使用浮点GV存储类全局变量函数将收盘价存储到全局变量”gn”中,而test4中使用浮点GV调用类全局变量函数调用全局变量”gn”中的值。这样设置之后,执行回测,变量value1存储每根bar的收盘价,举例,在编号为1的bar上,value1的值是编号为1的bar的收盘价,在编号为2的bar上,value1的值是编号为2的bar的收盘价,以此类推,以下是输出结果: test3 ,currentbar= 1.00,close= 60.24 test4 ,currentbar= 1.00,value1= 60.24 test3 ,currentbar= 2.00,close= 60.25 test4 ,currentbar= 2.00,value1= 60.25 test3 ,currentbar= 3.00,close= 60.59 test4 ,currentbar= 3.00,value1= 60.59 test3 ,currentbar= 4.00,close= 60.64 test4 ,currentbar= 4.00,value1= 60.64 test3 ,currentbar= 5.00,close= 60.58 test4 ,currentbar= 5.00,value1= 60.58 test3 ,currentbar= 6.00,close= 60.58 test4 ,currentbar= 6.00,value1= 60.58 test3 ,currentbar= 7.00,close= 60.62 test4 ,currentbar= 7.00,value1= 60.62 GV全局变量的原理很简单,就是存储GV全局变量函数向GV全局变量中存储最新的值,而调用GV全局变量函数向GV全局变量中调用最新的值,就是在内存中为每个全局变量分配一块特定的存储区域,存储GV全局变量函数向这块区域中存储最新的值,而调用GV全局变量函数在这块特定的区域中取最新的值;若存储了10次之后,再调用全局变量值,那么肯定取的是最近一次存储的数值;若调用了10次全局变量的值,但是这10次中间都没有存储最新的值,那么这10次取的值当然都是一样的数值。之所以同一个策略不同信号之间可以实时传递信息,就是因为test3和test4的执行是依次有序进行的,也就是“存”和“取”是依次有序进行的,“存”一次,再“取”一次,再“存”一次,再“取”一次,依此类推。 以上是对于同一个策略不同信号之间的在历史回测中的运用,当然也可以在实时行情中运用,也可以在回放中使用,原理很简单,就是上面的“存”和“取”的依次有序进行,在此不再赘述。 四、全局变量内存释放 对于GV全局变量,由于是内存中分配一块特定的区域用于存储和调用,那么一个很简单的问题就是GV全局变量的内存空间什么时候被释放。因为GV全局变量是通过GV全局变量函数新建的,而函数是被调用在指标或者信号中的,最后指标或信号被插入图表或者市场扫描器中,也就是当含有GV全局变量函数的指标或者信号从图表或者市场扫描器中停止使用时,全局变量的内存空间就会被释放。 我们接着第三章来举例说明一下全局变量”gn”的内存什么时候被释放: Ø
在图1中的设置对象栏位,将test3的状态由“开启”更改成“关闭”时,”gn”没有被释放 Ø
在图1中的设置对象栏位,将test4的状态由“开启”更改成“关闭”时,”gn”没有被释放 Ø
在图1中的设置对象栏位,将test3和test4的状态由“开启”更改成“关闭”时,”gn”被释放 Ø
在图1中的设置对象栏位,将test3从图表上删除时,”gn”没有被释放 Ø
在图1中的设置对象栏位,将test4从图表上删除时,”gn”没有被释放 Ø
在图1中的设置对象栏位,将test3和test4从图表上删除时时,”gn”被释放 Ø
在图1中的设置对象栏位,将test3中GV函数删除再编译时,”gn”没有被释放 Ø
在图1中的设置对象栏位,将test4中GV函数删除再编译时,”gn”没有被释放 Ø
在图1中的设置对象栏位,将test3和test4中GV函数删除再编译时,”gn”被释放 将图表关闭时,”gn”被释放 |