楼主: kakapdc
6582 3

[交易策略] 传统市值选股策略 VS AI市值策略(Python源码) [推广有奖]

  • 0关注
  • 1粉丝

初中生

80%

还不是VIP/贵宾

-

威望
0
论坛币
56 个
通用积分
0.0001
学术水平
3 点
热心指数
8 点
信用等级
3 点
经验
485 点
帖子
11
精华
0
在线时间
6 小时
注册时间
2017-3-21
最后登录
2017-5-26

楼主
kakapdc 企业认证  发表于 2017-3-21 18:35:34 |只看作者 |坛友微信交流群|倒序 |AI写论文
相似文件 换一批

+2 论坛币
k人 参与回答

经管之家送您一份

应届毕业生专属福利!

求职就业群
赵安豆老师微信:zhaoandou666

经管之家联合CDA

送您一个全额奖学金名额~ !

感谢您参与论坛问题回答

经管之家送您两个论坛币!

+2 论坛币

股票传统策略和股票AI策略的在量化投资中的运用逐渐增多,今天我们就拿市值因子来练手,看看两个策略在2015-01-01到2016-12-31这两年时间各自的收益风险情形。


市值因子是国内股票市场能够带来超额收益的alpha因子,已经被验证为长期有效的因子,也是广大私募基金常用的因子之一,传统的选股策略的股票组合大多在市值因子上有很大的风险暴露。希望了解多因子选股策略的小伙伴可以参考这篇报告:东方证券《因子选股系列研究之十》:Alpha因子库精简与优化-160812


本文所介绍的传统小市值策略思想和操作都比较简单,就是选择市值最小的股票构建组合,可以参考社区文章:《手把手教你写策略之三:浅谈小市值选股策略》


AI市值策略是通过策略生成器构建策略,采用StockRanker排序模型基于市值因子做预测选股,即AI市值策略只有一个特征:市值。创建策略的具体步骤可以参考:第一个人工智能量化投资策略


我们先看传统小市值策略的回测结果图:


1.png


再看看AI市值策略回测结果图:


2.png


我们关注几个常用的指标来比较两个策略:


3.png


关于回测结果各指标的详细计算可以参考这篇文章:策略回测结果指标详解。从总收益来看,AI市值策略收益达到了289.46%,也就是说,如果15年年初你开始按照这个策略交易,期初本金1000元的话,到2017年年初的时候,就增加到了3894.6元,收益达到了289.46%,是不是比自己主观交易强多啦。虽然收益这么高,但是最大回撤也不低啊,最大回撤为35.22%,这个指标可以这样理解,就是严格按照策略系统交易,资金跌得最恨的时候距离资金最高点相差35.22%,如果没有良好的心态和强大的心脏估计是无法继续坚持策略的,比如资金从2000元的高点跌倒了1300元,一般的人可是坐不住的啊。不过,正是做到了坚持,所以坚持到2017年初,最后取到了289.46%的总收益。AI市值策略的最大回撤比传统小市值策略略高。在收益率的波动性方面,两个策略差不多。专业的量化人员关注地比较多的指标是夏普比率,该指标表示每承受一单位总风险,会产生多少的超额报酬,可以同时对策略的收益与风险进行综合考虑,AI市值策略的夏普比率比传统小市值策略高,达到了5.77。


可以看出,虽然传统小市值策略也是一个不错的策略,因为15年初1000元的本金投资在2017年初可以增值到3120元。但是与AI市值策略相比,AI市值策略由于收益更高,而且传统小市值策略用的人比较多,现在大部分的私募公募都暴露在市值因子上,因此策略同质性比较强。再加上,我们还可以在开发AI策略的时候利用自己的专业知识和行业经验构造特征(参考:量化投资中的特征工程),因此AI策略整体上比传统策略更优。


策略源码:

传统小市值策略

  1. # 获取股票代码
  2. instruments = D.instruments()
  3. # 确定起始时间
  4. start_date = '2015-01-05'
  5. # 确定结束时间
  6. end_date = '2017-01-01'

  7. market_cap_data = D.history_data(instruments,start_date,end_date,
  8.               fields=['market_cap','amount','suspended'])

  9. # 根据是否停牌的字段确定每日选出来的股票
  10. daily_buy_stock = market_cap_data.groupby('date').apply(lambda df:df[(df['amount'] > 0) # 需要有成交量
  11.                                                 & (df['suspended'] == False)  # 是否停牌
  12.                                                 ].sort_values('market_cap')[:30]) # 前三十只

  13. # 回测参数设置,initialize函数只运行一次
  14. def initialize(context):
  15.     # 手续费设置
  16.     context.set_commission(PerOrder(buy_cost=0.0003, sell_cost=0.0013, min_cost=5))
  17.     # 调仓规则(每月的第一天调仓)
  18.     context.schedule_function(rebalance,date_rule=date_rules.month_start(days_offset=0))
  19.     # 传入 整理好的调仓股票数据
  20.     context.daily_buy_stock = daily_buy_stock

  21. # handle_data函数会每天运行一次
  22. def handle_data(context,data):
  23.     pass

  24. # 换仓函数
  25. def rebalance(context, data):
  26.     # 当前的日期
  27.     date = data.current_dt.strftime('%Y-%m-%d')
  28.     # 根据日期获取调仓需要买入的股票的列表
  29.     stock_to_buy = context.daily_buy_stock.ix[date].instrument # 一定要转化为列表
  30.    
  31.     # 通过positions对象,使用列表生成式的方法获取目前持仓的股票列表
  32.     stock_hold_now = [equity.symbol for equity in context.portfolio.positions]
  33.    
  34.     # 继续持有的股票:调仓时,如果买入的股票已经存在于目前的持仓里,那么应继续持有
  35.     no_need_to_sell = [i for i in stock_hold_now if i in stock_to_buy]
  36.     # 需要卖出的股票
  37.     stock_to_sell = [i for i in stock_hold_now if i not in no_need_to_sell]
  38.   
  39.     # 卖出
  40.     for stock in stock_to_sell:
  41.         # 如果该股票停牌,则没法成交。因此需要用can_trade方法检查下该股票的状态
  42.         # 如果返回真值,则可以正常下单,否则会出错
  43.         # 因为stock是字符串格式,我们用symbol方法将其转化成平台可以接受的形式

  44.         if data.can_trade(context.symbol(stock)):
  45.             # order_target_percent是平台的一个下单接口,表明下单使得该股票的权重为0,
  46.             #   即卖出全部股票,可参考回测文档
  47.             context.order_target_percent(context.symbol(stock), 0)
  48.    
  49.     # 如果当天没有买入的股票,就返回
  50.     if len(stock_to_buy) == 0:
  51.         return

  52.     # 等权重买入
  53.     weight =  1 / len(stock_to_buy)
  54.    
  55.     # 买入
  56.     for  stock in stock_to_buy:
  57.         if data.can_trade(context.symbol(stock)):
  58.             # 下单使得某只股票的持仓权重达到weight,因为
  59.             # weight大于0,因此是等权重买入
  60.             context.order_target_percent(context.symbol(stock), weight)
  61.             
  62. # 回测接口
  63. m=M.backtest.v5(
  64.     instruments=instruments,
  65.     start_date=start_date,
  66.     end_date=end_date,
  67.     # 必须传入initialize,只在第一天运行
  68.     initialize=initialize,
  69.     #  必须传入handle_data,每个交易日都会运行
  70.     handle_data=handle_data,
  71.     # 买入以开盘价成交
  72.     order_price_field_buy='open',
  73.     # 卖出也以开盘价成交
  74.     order_price_field_sell='open',
  75.     # 策略本金
  76.     capital_base=float("1.0e7") ,
  77.     # 比较基准:沪深300
  78.     benchmark='000300.INDX',
  79. )
复制代码


AI市值策略:

  1. # 基础参数配置
  2. class conf:
  3.     start_date = '2010-01-01'
  4.     end_date='2017-01-20'
  5.     # split_date 之前的数据用于训练,之后的数据用作效果评估
  6.     split_date = '2015-01-01'
  7.     # D.instruments: https://bigquant.com/docs/data_instruments.html
  8.     instruments = D.instruments(start_date, end_date)

  9.     # 机器学习目标标注函数
  10.     # 如下标注函数等价于 max(min((持有期间的收益 * 100), -20), 20) + 20 (后面的M.fast_auto_labeler会做取整操作)
  11.     # 说明:max/min这里将标注分数限定在区间[-20, 20],+20将分数变为非负数 (StockRanker要求标注分数非负整数)
  12.     label_expr = ['return * 100', 'where(label > {0}, {0}, where(label < -{0}, -{0}, label)) + {0}'.format(20)]
  13.     # 持有天数,用于计算label_expr中的return值(收益)
  14.     hold_days = 30

  15.     # 特征 https://bigquant.com/docs/data_features.html,你可以通过表达式构造任何特征
  16.     features = [
  17.         'market_cap_0',  # 总市值
  18.     ]

  19. # 给数据做标注:给每一行数据(样本)打分,一般分数越高表示越好
  20. m1 = M.fast_auto_labeler.v5(
  21.     instruments=conf.instruments, start_date=conf.start_date, end_date=conf.end_date,
  22.     label_expr=conf.label_expr, hold_days=conf.hold_days,
  23.     benchmark='000300.SHA', sell_at='open', buy_at='open')
  24. # 计算特征数据
  25. m2 = M.general_feature_extractor.v5(
  26.     instruments=conf.instruments, start_date=conf.start_date, end_date=conf.end_date,
  27.     features=conf.features)
  28. # 数据预处理:缺失数据处理,数据规范化,T.get_stock_ranker_default_transforms为StockRanker模型做数据预处理
  29. m3 = M.transform.v2(
  30.     data=m2.data, transforms=T.get_stock_ranker_default_transforms(),
  31.     drop_null=True, astype='int32', except_columns=['date', 'instrument'],
  32.     clip_lower=0, clip_upper=200000000)
  33. # 合并标注和特征数据
  34. m4 = M.join.v2(data1=m1.data, data2=m3.data, on=['date', 'instrument'], sort=True)

  35. # 训练数据集
  36. m5_training = M.filter.v2(data=m4.data, expr='date < "%s"' % conf.split_date)
  37. # 评估数据集
  38. m5_evaluation = M.filter.v2(data=m4.data, expr='"%s" <= date' % conf.split_date)
  39. # StockRanker机器学习训练
  40. m6 = M.stock_ranker_train.v2(training_ds=m5_training.data, features=conf.features)
  41. # 对评估集做预测
  42. m7 = M.stock_ranker_predict.v2(model_id=m6.model_id, data=m5_evaluation.data)


  43. ## 量化回测 https://bigquant.com/docs/strategy_backtest.html
  44. # 回测引擎:初始化函数,只执行一次
  45. def initialize(context):
  46.     # 系统已经设置了默认的交易手续费和滑点,要修改手续费可使用如下函数
  47.     context.set_commission(PerOrder(buy_cost=0.0003, sell_cost=0.0013, min_cost=5))
  48.     # 预测数据,通过options传入进来,使用 read_df 函数,加载到内存 (DataFrame)
  49.     context.ranker_prediction = context.options['ranker_prediction'].read_df()
  50.     # 设置买入的股票数量,这里买入预测股票列表排名靠前的5只
  51.     stock_count = 5
  52.     # 每只的股票的权重,如下的权重分配会使得靠前的股票分配多一点的资金,[0.339160, 0.213986, 0.169580, ..]
  53.     context.stock_weights = T.norm([1 / math.log(i + 2) for i in range(0, stock_count)])
  54.     # 设置每只股票占用的最大资金比例
  55.     context.max_cash_per_instrument = 0.2

  56. # 回测引擎:每日数据处理函数,每天执行一次
  57. def handle_data(context, data):
  58.     # 按日期过滤得到今日的预测数据
  59.     ranker_prediction = context.ranker_prediction[context.ranker_prediction.date == data.current_dt.strftime('%Y-%m-%d')]

  60.     # 1. 资金分配
  61.     # 平均持仓时间是hold_days,每日都将买入股票,每日预期使用 1/hold_days 的资金
  62.     # 实际操作中,会存在一定的买入误差,所以在前hold_days天,等量使用资金;之后,尽量使用剩余资金(这里设置最多用等量的1.5倍)
  63.     is_staging = context.trading_day_index < context.options['hold_days'] # 是否在建仓期间(前 hold_days 天)
  64.     cash_avg = context.portfolio.portfolio_value / context.options['hold_days']
  65.     cash_for_buy = min(context.portfolio.cash, (1 if is_staging else 1.5) * cash_avg)
  66.     cash_for_sell = cash_avg - (context.portfolio.cash - cash_for_buy)
  67.     positions = {e.symbol: p.amount * p.last_sale_price         for e, p in context.perf_tracker.position_tracker.positions.items()}

  68.     # 2. 生成卖出订单:hold_days天之后才开始卖出;对持仓的股票,按StockRanker预测的排序末位淘汰
  69.     if not is_staging and cash_for_sell > 0:
  70.         equities = {e.symbol: e for e, p in context.perf_tracker.position_tracker.positions.items()}
  71.         instruments = list(reversed(list(ranker_prediction.instrument[ranker_prediction.instrument.apply(
  72.                 lambda x: x in equities and not context.has_unfinished_sell_order(equities[x]))])))
  73.         # print('rank order for sell %s' % instruments)
  74.         for instrument in instruments:
  75.             context.order_target(context.symbol(instrument), 0)
  76.             cash_for_sell -= positions[instrument]
  77.             if cash_for_sell <= 0:
  78.                 break

  79.     # 3. 生成买入订单:按StockRanker预测的排序,买入前面的stock_count只股票
  80.     buy_cash_weights = context.stock_weights
  81.     buy_instruments = list(ranker_prediction.instrument[:len(buy_cash_weights)])
  82.     max_cash_per_instrument = context.portfolio.portfolio_value * context.max_cash_per_instrument
  83.     for i, instrument in enumerate(buy_instruments):
  84.         cash = cash_for_buy * buy_cash_weights[i]
  85.         if cash > max_cash_per_instrument - positions.get(instrument, 0):
  86.             # 确保股票持仓量不会超过每次股票最大的占用资金量
  87.             cash = max_cash_per_instrument - positions.get(instrument, 0)
  88.         if cash > 0:
  89.             context.order_value(context.symbol(instrument), cash)

  90. # 调用回测引擎
  91. m8 = M.backtest.v5(
  92.     instruments=m7.instruments,
  93.     start_date=m7.start_date,
  94.     end_date='2017-01-01',
  95.     initialize=initialize,
  96.     handle_data=handle_data,
  97.     order_price_field_buy='open',       # 表示 开盘 时买入
  98.     order_price_field_sell='close',     # 表示 收盘 前卖出
  99.     capital_base=1000000,               # 初始资金
  100.     benchmark='000300.SHA',             # 比较基准,不影响回测结果
  101.     # 通过 options 参数传递预测数据和参数给回测引擎
  102.     options={'ranker_prediction': m7.predictions, 'hold_days': conf.hold_days}
  103. )
复制代码



感兴趣的朋友可以点此查看更多策略细节:原文链接

二维码

扫码加我 拉你入群

请注明:姓名-公司-职位

以便审核进群资格,未注明则拒绝


2.png (39.2 KB)

2.png

1.png (36.77 KB)

1.png

已有 2 人评分经验 学术水平 热心指数 信用等级 收起 理由
xugonglei + 5 精彩帖子
accumulation + 100 + 1 + 1 + 1 精彩帖子

总评分: 经验 + 100  学术水平 + 1  热心指数 + 6  信用等级 + 1   查看全部评分

本帖被以下文库推荐

沙发
ydc129 发表于 2017-3-29 22:36:15 |只看作者 |坛友微信交流群
thanks

使用道具

藤椅
panstrgh 在职认证  发表于 2018-3-3 23:19:59 |只看作者 |坛友微信交流群
感谢分享

使用道具

板凳
xugonglei 发表于 2018-3-4 23:21:25 |只看作者 |坛友微信交流群
谢谢分享

使用道具

您需要登录后才可以回帖 登录 | 我要注册

本版微信群
加好友,备注jr
拉您进交流群

京ICP备16021002-2号 京B2-20170662号 京公网安备 11010802022788号 论坛法律顾问:王进律师 知识产权保护声明   免责及隐私声明

GMT+8, 2024-4-20 07:42