经典量化交易策略系列——海龟交易系统python源码及策略

0
回复
6148
查看
[复制链接]

294

主题

20

回帖

1380

积分

专栏作家

积分
1380
来源: 2019-7-17 22:10:26 显示全部楼层 |阅读模式
1983年,著名的商品投机家理查德.丹尼斯挑选了13个人进行培训,他教给这13个学员一套完整的交易系统,由于学员们被称为“海龟”,后来人们就把这套系统称为海龟交易系统。

    海龟交易系统简介:

    交易信号:海龟的交易信号其实很简单,当价格创20或50天新高就买入,当价格创10天或20天新低就卖出,时间上具体的参数使用者也可以自己调整。

    头寸管理和风险控制策略:海龟交易系统由总资金风险百分比和N波动的系数策略来决定交易头寸的多少,用N确定什么时候加仓、加多少,同时用2N来确定头寸的保护性损止。N每7天调整一次(五个交易日)。这就是海龟交易系统的交易策略,属于一套完整的交易系统。下面是我写的海龟交易系统测试版的编程源码。

    海龟交易策略:

    1.基本原理:本策略的大框架是按照原版的海龟策略搭建的,但是在具体实现方面有些改动。

    市场:本策略中选取了一只股票,中国平安,为简化起见。

    头寸规模:本策略中最大头寸规模限制为4unit。

  1. #源码:

  2. import numpy as np      
  3. import pandas as pd
  4. from pandas import Series,DataFrame
  5. import random
  6. import math
  7. # 定义一个全局变量, 保存要操作的证券
  8. security='600196.XSHG'
  9. # 设置我们要操作的股票池, 这里我们只操作一支股票
  10. set_universe([security])
  11. set_benchmark('600196.XSHG')
  12. #设置回测条件
  13. set_commission(PerTrade(buy_cost=0.0008, sell_cost=0.0015, min_cost=5))
  14. set_slippage(FixedSlippage(0))
  15. #调整资金规模时的临界损失比例
  16. loss=0.1
  17. #调整资金规模时调整后的资金占当前资金的比例
  18. adjust=0.8
  19. #计算第一个N时取得股票数据的天数
  20. days=20
  21. #系统一入市时股票价格需要高于short_in天内的最高价
  22. short_in=20
  23. #系统二入市时股票价格需要高于long_in天内的最高价
  24. long_in=55
  25. #系统一离市时股票价格需要低于short_out天内的最低价
  26. short_out=10
  27. #系统二离市时股票价格需要低于long_out天内的最低价
  28. long_out=20
  29. #系统一和系统二的资金分配比例,系统一得到ratio*总资金,系统二得到(1-ratio)*总资金
  30. ratio=0.7
  31. #单一市场中的头寸规模限制
  32. limit=4
  33. #记录策略运行了多少天
  34. pdn=0
  35. #记录N值
  36. N=[]
  37. #记录系统一中股票的单位数
  38. sys1=0
  39. #记录系统二中股票的单位数
  40. sys2=0
  41. #判断操作是对系统一还是系统二,值为‘True’是对系统一,‘False’是对系统二
  42. short='False'
  43. #用unit来保存一单位表示多少股票,默认值为1000
  44. unit=1000
  45. #记录系统一形成突破时的股票价格
  46. break_price1=0
  47. #记录系统二形成突破时的股票价格
  48. break_price2=0
  49. #记录分钟
  50. minutes=0

  51. #计算股票的N值
  52. def Calcu_N(context,paused):
  53.     #在策略运行了days-1天时,计算前days-1天的平均实际范围
  54.     if pdn==days-1:
  55.         #取出day-1天来得最高价,最低价,前一天的收盘价
  56.         price=attribute_history(security,days-1,'1d', ('high','low','pre_close'),skip_paused=True)
  57.         #如果不是所有的这day-1天都没有数据,算出这些天的实际范围的平均值
  58.         TR=[]
  59.         for i in range(0,days-1):
  60.             h_l=price['high'][i]-price['low'][i]
  61.             h_pdc=price['high'][i]-price['pre_close'][i]
  62.             pdc_l=price['pre_close'][i]-price['low'][i]
  63.             temp=max(h_l,h_pdc,pdc_l)
  64.             TR.append(temp)
  65.         ATR=np.mean(np.array(TR))
  66.         N.append(ATR)
  67.         
  68.     #如果策略运行天数已经达到了days天
  69.     else:
  70.         #如果股票停牌,则将运行天数减1
  71.         if paused==True:
  72.             global pdn
  73.             pdn=pdn-1
  74.         #如果未停牌,则利用迭代,计算N值,并保存在列表N中
  75.         else:
  76.             price=attribute_history(security,1,'1d', ('high','low','pre_close'),skip_paused=True)
  77.             h_l=price['high'][0]-price['low'][0]
  78.             h_pdc=price['high'][0]-price['pre_close'][0]
  79.             pdc_l=price['pre_close'][0]-price['low'][0]
  80.             temp=max(h_l,h_pdc,pdc_l)
  81.             TR.append(temp)
  82.             ATR=np.mean(np.array(TR))
  83.             N.append(ATR)
  84.             
  85. #止损
  86. def Stop_Loss(current_price):
  87.     #如果对系统一操作,则将突破价设置为系统一的突破价,如果是对系统二,则设置为系统二的
  88.     if short=='True':
  89.         break_price=break_price1
  90.     else:
  91.         break_price=break_price2
  92.     #如果当前价格比上次的突破价低2N,则清空头寸
  93.     #并相应的更改相应系统中的股票单位数
  94.     if current_price
  95.         if short=='True':
  96.             order(security,-sys1)
  97.             global sys1
  98.             sys1=0  
  99.         else:
  100.             order(security,-sys2)
  101.             global sys2
  102.             sys2=0
  103.             
  104. #入市
  105. def Sys_In(highest,day_in,context,current_price,cash):
  106.     #取出day_in天以来的最高价
  107.     price=attribute_history(security,day_in,'1d',('high','open'))
  108.     #如果当前价格高于day_in天的最高价,则形成突破
  109.     if current_price>max(price['high']) and current_price>=highest:
  110.         #计算可以买的股票数量
  111.         num_of_shares=cash/current_price
  112.         #如果可以买的数量不小于一单位,且目前持有的股票数量未达到限制的数量,则买入
  113.         if num_of_shares>=unit:
  114.             if short=='True':
  115.                 if sys1
  116.                     order(security,+int(unit))
  117.                     #买入后,相应的更新持有的股票数及突破价格
  118.                     global sys1
  119.                     sys1=sys1+int(unit)
  120.                     global break_price1
  121.                     break_price1=current_price
  122.             else:
  123.                 if sys2
  124.                     order(security,+int(unit))
  125.                     global sys2
  126.                     sys2=sys2+int(unit)
  127.                     global break_price2
  128.                     break_price2=current_price
  129.    
  130. #增加单位   
  131. def Sys_Add(day_in,context,current_price,cash):
  132.     #根据short的值判断是对哪个系统操作,以对突破价格赋值
  133.     if short=='True':
  134.         break_price=break_price1
  135.     else:
  136.         break_price=break_price2
  137.     #如果当前价格比上次的突破价格高0.5*N,则增加一单位
  138.     if current_price>=break_price+0.5*N[-1]:
  139.         num_of_shares=cash/current_price
  140.         if num_of_shares>=unit:
  141.             if short=='True':
  142.                 if sys1
  143.                     order(security,+int(unit))
  144.                     global sys1
  145.                     sys1=sys1+int(unit)
  146.                     global break_price1
  147.                     break_price1=current_price
  148.             else:
  149.                 if sys2
  150.                     order(security,+int(unit))
  151.                     global sys2
  152.                     sys2=sys2+int(unit)
  153.                     global break_price2
  154.                     break_price2=current_price
  155.    
  156. # 离市
  157. def Sys_Out(day_out,current_price,context):
  158.     #取出day_out天以来的最低价
  159.     price=attribute_history(security,day_out,'1d',('high','low'),skip_paused=True)
  160.     #如果股票当前价格比day_out天的最低价低,则清空系统内的头寸
  161.     #并将相应系统的头寸数量置为0
  162.     if current_price
  163.         if short=='True':
  164.             if sys1>0:
  165.                 order(security,-sys1)
  166.                 global sys1
  167.                 sys1=0
  168.         else:
  169.             if sys2>0:
  170.                 order(security,-sys2)
  171.                 global sys2
  172.                 sys2=0

  173. #每个单位时间(如果按天回测,则每天调用一次,如果按分钟,则每分钟调用一次)调用一次
  174. def handle_data(context, data):
  175.     global minutes
  176.     minutes=minutes+1
  177.     price=attribute_history(security,minutes,'1m',('high','price','open'),skip_paused=True)
  178.     #取得从今天开盘为止的最高价
  179.     highest=max(price['open'])
  180.     #用paused保存股票是否停牌
  181.     paused=data[security].paused
  182.     #用dt保存当前时间
  183.     dt=context.current_dt
  184.     #保存当前股票价格
  185.     current_price=data[security].price
  186.     #保存资产组合的总值
  187.     value=context.portfolio.portfolio_value
  188.     #在每天开市时将策略运行时间加1
  189.     if dt.hour==9 and dt.minute==30:
  190.         global minutes
  191.         minutes=0
  192.         global pdn
  193.         pdn=pdn+1
  194.         #在运行时间达到days-1天时计算头寸单位
  195.         if pdn==days-1:
  196.             Calcu_N(context,paused)
  197.     #运行时间达到days天时开始执行各种买卖操作
  198.     if pdn>=days:
  199.         if pdn==days:
  200.             global break_price1
  201.             break_price1=current_price*5+1
  202.         #如果股票不停牌
  203.         if paused==False:
  204.             #取得当前现金
  205.             cash=context.portfolio.cash
  206.             #如果空仓了
  207.             if sys1==0 and sys2==0:
  208.                 #调整资金规模
  209.                 if context.portfolio.portfolio_value<(1-loss)*context.portfolio.starting_cash:
  210.                     cash=adjust*cash
  211.                     value=adjust*value
  212.             #每点价值量(yuan per point)
  213.             ypp=1.0
  214.             #价值量波动性 value volatility
  215.             vv=ypp*N[-1]
  216.             #计算一单位的数量
  217.             global unit
  218.             unit=value*0.01/vv
  219.             #将short置为‘True’,对系统一进行操作
  220.             global short
  221.             short='True'
  222.             #如果系统一没有头寸,并且当前价格比上次的突破价低,入市
  223.             if sys1==0:
  224.                 #入市
  225.                 Sys_In(highest,short_in,context,current_price,ratio*cash)
  226.             #如果已经有了头寸,则进行止损或者增加单位
  227.             else:
  228.                 #止损
  229.                 Stop_Loss(current_price)
  230.                 #增加单位
  231.                 Sys_Add(short_in,context,current_price,ratio*cash)   
  232.             #离市
  233.             Sys_Out(short_out,current_price,context)
  234.             #将short置为‘False’,对系统二进行操作
  235.             global short
  236.             short='False'
  237.             if sys2==0:
  238.                 #入市
  239.                 Sys_In(highest,long_in,context,current_price,(1-ratio)*cash)
  240.             else:
  241.                 Stop_Loss(current_price)
  242.                 Sys_Add(long_in,context,current_price,(1-ratio)*cash)
  243.             #离市
  244.             Sys_Out(long_out,current_price,context)   
复制代码




    入市:策略中有系统一和系统二两个入市系统。

    止损:使用以价格为基础的止损条件,当股价比买入时股价低2N时将持有的头寸清空。

    离市:在当前股价比day_out天以来的最低价还低时把持有的头寸清空。

    策略:执行限价指令以避免对实际市场价格造成大的波动是原版的海龟交易策略,由于本策略采用以前的数据进行回测,并没有干扰市场的问题,因为股票价格已经给定,所以都执行的是市价指令。

    2.变量计算及操作:

    下文把各个变量的计算及一些操作出来进行说明,因为涉及到的变量计算和操作较多。

    1.N的计算方式:N=(19×PDN+TR)/20,其中,PDN是指前一天的N,
      TR(实际范围)=max(H-L,H-PDC,PDC-L)
      式中:
      H-当日最高价
      L-当日最低价
      PDC-前个交易日的收盘价
      在本策略中,用于迭代的第一个N是采用days天内数据可用的那些天的TR的平均值。
    2.Unit的计算方式:unit=帐户净值的1%/(N×每点价值量)
      每点价值量 是指一手合约的价格每变动一个最小单位,整个合约变动的价格。在中国股市中,股票价格变动的最小价格一般是0.01元,一手股票是100股,因此每点价值量就是0.01*100=1。
    3.入市:价格形成突破,即当前股票价格>day_in天中股票的最高价格
      对于系统一,day_in=short_in,对于系统二,day_in=long_in
      对于系统一,如果上一次的突破实现了盈利,那么下一次就忽视掉系统一的突破信号
      入市以后的增加单位:如果在持有头寸之后,股票价格在上次突破时的价格的基础上又上涨了0.5N,则再买入一单位股票
      注:short_in是系统一入市时的参数,表示取多少天内的股票价格,例如,如果设置short_in=20,则当前股票价格>20天中股票的最高价格时算一次突破。
      Long_in是系统二的入市参数,含义同short_in
    4.增加单位:当股票价格>上次突破价格+0.5N时,在原来持有的头寸的基础上继续增加一单位
    5.止损:如果股票价格相对于上次买入的价格下降了2N,则把持有的头寸清仓
    6.离市:如果股票价格小于day_out天来得最低价,则清空持有的头寸
对于系统一,day_out=short_out。对于系统二,day_out=long_out
      注:short_out是系统一的离市参数,如果股票价格小于short_out天中股票的最低价,则离市,long__out同理
    7.调整资金规模:在将头寸全部清空之后,判断如果当前资金小于(1-loss)初始资金,则将可调用的资金缩减为adjust当前资金
注:loss,adjust是调整资金规模的参数,loss表示调整资金规模时的损失比例,即如果损失的资金占初始资金得比例达到loss时,需要调整资金规模。
      Adjust表示资金规模的调整系数,即每次调整时,将可用资金缩减到adjust*当前资金
      注:系统一和系统二是两个独立的系统,这两个系统的资金分配由ratio控制,系统一得到的资金占总资金的比例为ratio,系统二所占比例为1-ratio。此外,在购买股票和清空持有头寸时,两系统也是独立的,清空不是指把所有股票都清空,而是哪个系统需要清空就清空该系统内的股票。

    3.海龟策略流程

    1.计算N,unit,
    2.判断是否调整资金规模,如果需要,则进行调整
    3.进行系统一,根据当前价格,系统一中的头寸判断应该入市,还是增加单位,还是止损,还是离市
    4.进行系统二,同样,根据当前价格,系统一中的头寸判断应该入市,还是增加单位,还是止损,还是离市

黄金 4.jpg


回复

使用道具 举报

您需要登录后才可以回帖 登录 | 免费注册
关注微信