复制链接
克隆策略

价值选股策略

版本 v1.0

目录

  • ### 价值选股策略的交易规则

  • ### 策略构建步骤

  • ### 策略的实现

正文

一、价值选股策略的交易规则

  • 每隔30个交易日,以开盘价买入当日0<PB<1.5且0<PE<15且有成交量的股票;
  • 每隔30个交易日,将不符合上述标准的持仓股票在第二天以收盘价卖出。

二、策略构建步骤

1、确定股票池和回测时间

  • 通过证券代码列表输入要回测的单只/多只股票,以及回测的起止日期

2、确定买卖条件信号

  • 在输入特征列表中通过表达式引擎定义 buy_condition=where((pb_lf_0<1.5) & (pe_ttm_0<15) & (amount_0>0) & (pb_lf_0>0) & (pe_ttm_0>0), 1, 0) 实现买入条件信号定义。
  • 通过基础特征和衍生特征抽取模块实现买卖条件指标 buy_condition 数据的抽取。
  • 通过缺失数据处理模块删去有缺失值的数据。
  • 通过排序模块实现每日股票数据按 pe_ttm_0 和 pb_lf_0 两个指标降序排列

3、确定买卖原则

  • 每隔30个交易日换仓一次;
  • 已有持仓中不满足买入条件的股票为卖出股票列表,需执行卖出操作
  • 满足买入条件且没有持仓的股票为买入股票列表,需执行买入操作
  • 满足买入条件且已有持仓的股票为调仓股票列表,需执行调整仓位操作
  • 本策略中将买入股票列表和调仓股票列表中的所有股票统一调整为等资金比例仓位。

4、模拟回测

  • 通过 trade 模块中的初始化函数定义交易手续费和滑点;
  • 通过 trade 模块中的准备函数定义 context.daily_stock_buy 变量来获取并存放每日买卖交易信号;
  • 通过 trade 模块中的主函数(handle函数)查看每日的买卖交易信号,按照买卖原则执行相应的买入/卖出/调仓操作。

三、策略的实现

可视化策略实现如下:

    {"description":"实验创建于2017/8/26","graph":{"edges":[{"to_node_id":"-2326:features","from_node_id":"287d2cb0-f53c-4101-bdf8-104b137c8601-24:data"},{"to_node_id":"-2333:features","from_node_id":"287d2cb0-f53c-4101-bdf8-104b137c8601-24:data"},{"to_node_id":"-2326:instruments","from_node_id":"287d2cb0-f53c-4101-bdf8-104b137c8601-62:data"},{"to_node_id":"-2171:instruments","from_node_id":"287d2cb0-f53c-4101-bdf8-104b137c8601-62:data"},{"to_node_id":"-2333:input_data","from_node_id":"-2326:data"},{"to_node_id":"-104:input_data","from_node_id":"-2333:data"},{"to_node_id":"-2171:options_data","from_node_id":"-50:sorted_data"},{"to_node_id":"-1699:input_data","from_node_id":"-104:data"},{"to_node_id":"-50:input_ds","from_node_id":"-1699:data"}],"nodes":[{"node_id":"287d2cb0-f53c-4101-bdf8-104b137c8601-24","module_id":"BigQuantSpace.input_features.input_features-v1","parameters":[{"name":"features","value":"# #号开始的表示注释\n# 多个特征,每行一个,可以包含基础特征和衍生特征\nbuy_condition=where((pb_lf_0<1.5) & (pe_ttm_0<15) & (amount_0>0) & (pb_lf_0>0) & (pe_ttm_0>0), 1, 0)","type":"Literal","bound_global_parameter":null}],"input_ports":[{"name":"features_ds","node_id":"287d2cb0-f53c-4101-bdf8-104b137c8601-24"}],"output_ports":[{"name":"data","node_id":"287d2cb0-f53c-4101-bdf8-104b137c8601-24"}],"cacheable":true,"seq_num":1,"comment":"","comment_collapsed":true},{"node_id":"287d2cb0-f53c-4101-bdf8-104b137c8601-62","module_id":"BigQuantSpace.instruments.instruments-v2","parameters":[{"name":"start_date","value":"2017-05-01","type":"Literal","bound_global_parameter":"交易日期"},{"name":"end_date","value":"2017-10-11","type":"Literal","bound_global_parameter":"交易日期"},{"name":"market","value":"CN_STOCK_A","type":"Literal","bound_global_parameter":null},{"name":"instrument_list","value":"","type":"Literal","bound_global_parameter":null},{"name":"max_count","value":"0","type":"Literal","bound_global_parameter":null}],"input_ports":[{"name":"rolling_conf","node_id":"287d2cb0-f53c-4101-bdf8-104b137c8601-62"}],"output_ports":[{"name":"data","node_id":"287d2cb0-f53c-4101-bdf8-104b137c8601-62"}],"cacheable":true,"seq_num":2,"comment":"预测数据,用于回测和模拟","comment_collapsed":false},{"node_id":"-2326","module_id":"BigQuantSpace.general_feature_extractor.general_feature_extractor-v7","parameters":[{"name":"start_date","value":"","type":"Literal","bound_global_parameter":null},{"name":"end_date","value":"","type":"Literal","bound_global_parameter":null},{"name":"before_start_days","value":"60","type":"Literal","bound_global_parameter":null}],"input_ports":[{"name":"instruments","node_id":"-2326"},{"name":"features","node_id":"-2326"}],"output_ports":[{"name":"data","node_id":"-2326"}],"cacheable":true,"seq_num":7,"comment":"","comment_collapsed":true},{"node_id":"-2333","module_id":"BigQuantSpace.derived_feature_extractor.derived_feature_extractor-v3","parameters":[{"name":"date_col","value":"date","type":"Literal","bound_global_parameter":null},{"name":"instrument_col","value":"instrument","type":"Literal","bound_global_parameter":null},{"name":"drop_na","value":"False","type":"Literal","bound_global_parameter":null},{"name":"remove_extra_columns","value":"False","type":"Literal","bound_global_parameter":null},{"name":"user_functions","value":"","type":"Literal","bound_global_parameter":null}],"input_ports":[{"name":"input_data","node_id":"-2333"},{"name":"features","node_id":"-2333"}],"output_ports":[{"name":"data","node_id":"-2333"}],"cacheable":true,"seq_num":8,"comment":"","comment_collapsed":true},{"node_id":"-50","module_id":"BigQuantSpace.sort.sort-v4","parameters":[{"name":"sort_by","value":"pe_ttm_0,pb_lf_0","type":"Literal","bound_global_parameter":null},{"name":"group_by","value":"date","type":"Literal","bound_global_parameter":null},{"name":"keep_columns","value":"--","type":"Literal","bound_global_parameter":null},{"name":"ascending","value":"True","type":"Literal","bound_global_parameter":null}],"input_ports":[{"name":"input_ds","node_id":"-50"},{"name":"sort_by_ds","node_id":"-50"}],"output_ports":[{"name":"sorted_data","node_id":"-50"}],"cacheable":true,"seq_num":3,"comment":"","comment_collapsed":true},{"node_id":"-2171","module_id":"BigQuantSpace.trade.trade-v4","parameters":[{"name":"start_date","value":"","type":"Literal","bound_global_parameter":null},{"name":"end_date","value":"","type":"Literal","bound_global_parameter":null},{"name":"initialize","value":"# 回测引擎:初始化函数,只执行一次\ndef bigquant_run(context):\n\n # 系统已经设置了默认的交易手续费和滑点,要修改手续费可使用如下函数\n context.set_commission(PerOrder(buy_cost=0.0003, sell_cost=0.0013, min_cost=5))\n","type":"Literal","bound_global_parameter":null},{"name":"handle_data","value":"# 回测引擎:每日数据处理函数,每天执行一次\ndef bigquant_run(context, data):\n #周期控制\n if context.trading_day_index % 30 != 0:#以30天(交易日)换一次仓为例\n return\n # 获取今日的日期\n today = data.current_dt.strftime('%Y-%m-%d') \n # 通过positions对象,使用列表生成式的方法获取目前持仓的股票列表\n stock_hold_now = {e.symbol: p.amount * p.last_sale_price\n for e, p in context.perf_tracker.position_tracker.positions.items()} \n \n # 记录用于买入股票的可用现金\n cash_for_buy = context.portfolio.cash \n \n try:\n buy_stock = context.daily_stock_buy[today] # 当日符合买入条件的股票\n except:\n buy_stock=[] # 如果没有符合条件的股票,就设置为空\n \n try:\n sell_stock = context.daily_stock_sell[today] # 当日符合卖出条件的股票\n except:\n sell_stock=[] # 如果没有符合条件的股票,就设置为空\n \n # 需要卖出的股票:已有持仓中符合卖出条件的股票\n stock_to_sell = [ i for i in stock_hold_now if i in sell_stock ]\n # 需要买入的股票:没有持仓且符合买入条件的股票\n stock_to_buy = [ i for i in buy_stock if i not in stock_hold_now ] \n # 需要调仓的股票:已有持仓且不符合卖出条件的股票\n stock_to_adjust=[ i for i in stock_hold_now if i not in sell_stock ]\n \n # 如果有卖出信号\n if len(stock_to_sell)>0:\n for instrument in stock_to_sell:\n sid = context.symbol(instrument) # 将标的转化为equity格式\n cur_position = context.portfolio.positions[sid].amount # 持仓\n if cur_position > 0 and data.can_trade(sid):\n context.order_target_percent(sid, 0) # 全部卖出\n # 因为是早盘买尾盘卖,所以卖出时不需更新可用现金,因为尾盘卖出股票所得现金无法用来买股票\n # cash_for_buy += stock_hold_now[instrument]\n \n # 如果有买入信号/有持仓\n if len(stock_to_buy)+len(stock_to_adjust)>0:\n weight = 1/(len(stock_to_buy)+len(stock_to_adjust)) # 每只股票的比重为等资金比例持有\n for instrument in stock_to_buy+stock_to_adjust:\n sid = context.symbol(instrument) # 将标的转化为equity格式\n if data.can_trade(sid):\n context.order_target_value(sid, weight*cash_for_buy) # 买入","type":"Literal","bound_global_parameter":null},{"name":"prepare","value":"# 回测引擎:准备数据,只执行一次\ndef bigquant_run(context):\n # 加载预测数据\n history_data = context.options['data'].read_df()\n\n #获取每日买入股票列表\n context.daily_stock_buy = history_data.groupby('date').apply(lambda x:x.instrument[:30])\n","type":"Literal","bound_global_parameter":null},{"name":"before_trading_start","value":"","type":"Literal","bound_global_parameter":null},{"name":"volume_limit","value":0.025,"type":"Literal","bound_global_parameter":null},{"name":"order_price_field_buy","value":"open","type":"Literal","bound_global_parameter":null},{"name":"order_price_field_sell","value":"close","type":"Literal","bound_global_parameter":null},{"name":"capital_base","value":1000000,"type":"Literal","bound_global_parameter":null},{"name":"auto_cancel_non_tradable_orders","value":"True","type":"Literal","bound_global_parameter":null},{"name":"data_frequency","value":"daily","type":"Literal","bound_global_parameter":null},{"name":"price_type","value":"后复权","type":"Literal","bound_global_parameter":null},{"name":"product_type","value":"股票","type":"Literal","bound_global_parameter":null},{"name":"plot_charts","value":"True","type":"Literal","bound_global_parameter":null},{"name":"backtest_only","value":"False","type":"Literal","bound_global_parameter":null},{"name":"benchmark","value":"000300.HIX","type":"Literal","bound_global_parameter":null}],"input_ports":[{"name":"instruments","node_id":"-2171"},{"name":"options_data","node_id":"-2171"},{"name":"history_ds","node_id":"-2171"},{"name":"benchmark_ds","node_id":"-2171"},{"name":"trading_calendar","node_id":"-2171"}],"output_ports":[{"name":"raw_perf","node_id":"-2171"}],"cacheable":false,"seq_num":9,"comment":"","comment_collapsed":true},{"node_id":"-104","module_id":"BigQuantSpace.filter.filter-v3","parameters":[{"name":"expr","value":"buy_condition>0","type":"Literal","bound_global_parameter":null},{"name":"output_left_data","value":"False","type":"Literal","bound_global_parameter":null}],"input_ports":[{"name":"input_data","node_id":"-104"}],"output_ports":[{"name":"data","node_id":"-104"},{"name":"left_data","node_id":"-104"}],"cacheable":true,"seq_num":4,"comment":"","comment_collapsed":true},{"node_id":"-1699","module_id":"BigQuantSpace.dropnan.dropnan-v2","parameters":[],"input_ports":[{"name":"input_data","node_id":"-1699"},{"name":"features","node_id":"-1699"}],"output_ports":[{"name":"data","node_id":"-1699"}],"cacheable":true,"seq_num":5,"comment":"","comment_collapsed":true}],"node_layout":"<node_postions><node_position Node='287d2cb0-f53c-4101-bdf8-104b137c8601-24' Position='1298,45,200,200'/><node_position Node='287d2cb0-f53c-4101-bdf8-104b137c8601-62' Position='801,35,200,200'/><node_position Node='-2326' Position='970,185,200,200'/><node_position Node='-2333' Position='1135,302,200,200'/><node_position Node='-50' Position='1122,543,200,200'/><node_position Node='-2171' Position='997,646,200,200'/><node_position Node='-104' Position='1107,383,200,200'/><node_position Node='-1699' Position='1133,459,200,200'/></node_postions>"},"nodes_readonly":false,"studio_version":"v2"}
    In [1]:
    # 本代码由可视化策略环境自动生成 2021年12月6日 22:15
    # 本代码单元只能在可视化模式下编辑。您也可以拷贝代码,粘贴到新建的代码单元或者策略,然后修改。
    
    
    # 回测引擎:初始化函数,只执行一次
    def m9_initialize_bigquant_run(context):
    
        # 系统已经设置了默认的交易手续费和滑点,要修改手续费可使用如下函数
        context.set_commission(PerOrder(buy_cost=0.0003, sell_cost=0.0013, min_cost=5))
    
    # 回测引擎:每日数据处理函数,每天执行一次
    def m9_handle_data_bigquant_run(context, data):
         #周期控制
        if context.trading_day_index % 30 != 0:#以30天(交易日)换一次仓为例
            return
        # 获取今日的日期
        today = data.current_dt.strftime('%Y-%m-%d')  
        # 通过positions对象,使用列表生成式的方法获取目前持仓的股票列表
        stock_hold_now = {e.symbol: p.amount * p.last_sale_price
                          for e, p in context.perf_tracker.position_tracker.positions.items()} 
        
        # 记录用于买入股票的可用现金
        cash_for_buy = context.portfolio.cash     
        
        try:
            buy_stock = context.daily_stock_buy[today]  # 当日符合买入条件的股票
        except:
            buy_stock=[]  # 如果没有符合条件的股票,就设置为空
        
        try:
            sell_stock = context.daily_stock_sell[today]  # 当日符合卖出条件的股票
        except:
            sell_stock=[] # 如果没有符合条件的股票,就设置为空
        
        # 需要卖出的股票:已有持仓中符合卖出条件的股票
        stock_to_sell = [ i for i in stock_hold_now if i in sell_stock ]
        # 需要买入的股票:没有持仓且符合买入条件的股票
        stock_to_buy = [ i for i in buy_stock if i not in stock_hold_now ]  
        # 需要调仓的股票:已有持仓且不符合卖出条件的股票
        stock_to_adjust=[ i for i in stock_hold_now if i not in sell_stock ]
        
        # 如果有卖出信号
        if len(stock_to_sell)>0:
            for instrument in stock_to_sell:
                sid = context.symbol(instrument) # 将标的转化为equity格式
                cur_position = context.portfolio.positions[sid].amount # 持仓
                if cur_position > 0 and data.can_trade(sid):
                    context.order_target_percent(sid, 0) # 全部卖出
                    # 因为是早盘买尾盘卖,所以卖出时不需更新可用现金,因为尾盘卖出股票所得现金无法用来买股票
                    # cash_for_buy += stock_hold_now[instrument]
        
        # 如果有买入信号/有持仓
        if len(stock_to_buy)+len(stock_to_adjust)>0:
            weight = 1/(len(stock_to_buy)+len(stock_to_adjust)) # 每只股票的比重为等资金比例持有
            for instrument in stock_to_buy+stock_to_adjust:
                sid = context.symbol(instrument) # 将标的转化为equity格式
                if  data.can_trade(sid):
                    context.order_target_value(sid, weight*cash_for_buy) # 买入
    # 回测引擎:准备数据,只执行一次
    def m9_prepare_bigquant_run(context):
        # 加载预测数据
        history_data = context.options['data'].read_df()
    
        #获取每日买入股票列表
        context.daily_stock_buy = history_data.groupby('date').apply(lambda x:x.instrument[:30])
    
    
    m1 = M.input_features.v1(
        features="""# #号开始的表示注释
    # 多个特征,每行一个,可以包含基础特征和衍生特征
    buy_condition=where((pb_lf_0<1.5) & (pe_ttm_0<15) & (amount_0>0) & (pb_lf_0>0) & (pe_ttm_0>0), 1, 0)"""
    )
    
    m2 = M.instruments.v2(
        start_date=T.live_run_param('trading_date', '2017-05-01'),
        end_date=T.live_run_param('trading_date', '2017-10-11'),
        market='CN_STOCK_A',
        instrument_list='',
        max_count=0
    )
    
    m7 = M.general_feature_extractor.v7(
        instruments=m2.data,
        features=m1.data,
        start_date='',
        end_date='',
        before_start_days=60
    )
    
    m8 = M.derived_feature_extractor.v3(
        input_data=m7.data,
        features=m1.data,
        date_col='date',
        instrument_col='instrument',
        drop_na=False,
        remove_extra_columns=False
    )
    
    m4 = M.filter.v3(
        input_data=m8.data,
        expr='buy_condition>0',
        output_left_data=False
    )
    
    m5 = M.dropnan.v2(
        input_data=m4.data
    )
    
    m3 = M.sort.v4(
        input_ds=m5.data,
        sort_by='pe_ttm_0,pb_lf_0',
        group_by='date',
        keep_columns='--',
        ascending=True
    )
    
    m9 = M.trade.v4(
        instruments=m2.data,
        options_data=m3.sorted_data,
        start_date='',
        end_date='',
        initialize=m9_initialize_bigquant_run,
        handle_data=m9_handle_data_bigquant_run,
        prepare=m9_prepare_bigquant_run,
        volume_limit=0.025,
        order_price_field_buy='open',
        order_price_field_sell='close',
        capital_base=1000000,
        auto_cancel_non_tradable_orders=True,
        data_frequency='daily',
        price_type='后复权',
        product_type='股票',
        plot_charts=True,
        backtest_only=False,
        benchmark='000300.HIX'
    )
    
    • 收益率5.96%
    • 年化收益率14.18%
    • 基准收益率13.46%
    • 阿尔法-0.06
    • 贝塔0.62
    • 夏普比率1.19
    • 胜率0.61
    • 盈亏比2.42
    • 收益波动率9.03%
    • 信息比率-0.13
    • 最大回撤4.91%
    bigcharts-data-start/{"__type":"tabs","__id":"bigchart-8f8a7817f36a48949a43effa9d45fc76"}/bigcharts-data-end