声明:本内容为个人业余研究,所有标的代码仅做为示例,回测收益不代表未来,不做为投资建议。
在上一篇文章中十年73倍的菜场大妈量化策略与风险平价模型,是在聚宽平台上回测的,但是有球友希望在ptrade上也能回测,为后面模拟盘做准备。
按照日线回测,月度调仓,一年的回测结果如下,回测期间总收益为 60.07%,策略的最大回撤为 22.7%。原始股票池变成了中小板,收益率比之前提高了一点。回测速度比聚宽要慢很多,五年回测要将近20分钟。
在每日运行中通过包装函数判断是否为月度首个交易日,若是,则触发选股流程。选股过程(select_stocks)是策略的阿尔法来源,它通过多层级过滤剔除不合格标的(如科创板/北交所、ST、停牌、涨跌停股),随后运用财务指标(股息率、PEG、股价)进行正面筛选,得到最终备选池。 得到股票池后,交易流程(trade_stocks)启动。首先,策略获取备选股票的历史收益率数据,计算其协方差矩阵,并输入到风险平价模型(get_risk_parity_weights)中求解最优权重。该模型的目标是使组合中每个资产对整体风险的贡献度相等,从而实现真正的风险分散。最后,策略根据计算出的目标权重执行调仓操作:卖出不在新组合中的持仓,并调整现有持仓至目标权重。
select_stocks(context)函数详解date_str = context.previous_date.strftime('%Y%m%d')log.info(f"获取股票列表,日期: {date_str}")# stock_list = get_Ashares(date_str)stock_list = get_index_stocks('399101.XSHE')log.info(f"初始股票数量: {len(stock_list)}")get_Ashares()获取全市场数据,此处为示例简化处理。stock_list = filter_kcbj_stock(stock_list)log.info(f"剔除科创北交后股票数量: {len(stock_list)}")stock_list = filter_st_stock(stock_list, context.previous_date)log.info(f"剔除ST股票后数量: {len(stock_list)}")stock_list = filter_paused_stock(stock_list, context.previous_date)log.info(f"剔除停牌股票后数量: {len(stock_list)}")stock_list_before_limit = stock_list.copy()stock_list = filter_limitup_stock(context, stock_list)log.info(f"剔除涨停股票后数量: {len(stock_list)}")stock_list = filter_limitdown_stock(context, stock_list)log.info(f"剔除跌停股票后数量: {len(stock_list)}")# PEG指标筛选stock_list = get_peg_filter_list(context, stock_list, g.peg_dn, g.peg_up)log.info(f"PEG筛选后股票数量: {len(stock_list)}")# 价格区间约束stock_list = get_price_filter_list(context, stock_list, g.price_dn, g.price_up)log.info(f"价格筛选后股票数量: {len(stock_list)}")trade_stocks(context)函数详解ifnot hasattr(g, 'select_list') ornot g.select_list: log.info("本月无选股,不进行交易")returndf_list = []for stock in select_list: hist_data = get_history( count=g.cov_days+30, frequency='1d', field=['close'], security_list=[stock], fq='pre', include=False )ifnot hist_data.empty and stock in hist_data['code'].values: stock_data = hist_data[hist_data['code']==stock]ifnot stock_data.empty: close_prices = pd.to_numeric(stock_data['close'], errors='coerce') df_list.append(pd.DataFrame({stock: close_prices.values}, index=stock_data.index))df = pd.concat(df_list, axis=1)returns = df.pct_change().dropna().tail(g.cov_days)g.cov_days天的观测值用于建模。weights = get_risk_parity_weights(returns, thre_down=g.pos_dn, thre_up=g.pos_up)weights = weights.sort_values()# 卖出不在新名单中的持仓for s in context.portfolio.positions:if (s notin weights.index): order_target(s, 0)# 买入目标权重对应的股票total_value = context.portfolio.portfolio_valuefor s in weights.index: w = weights[s] value = total_value * w order_target_value(s, value)get_risk_parity_weights(df, thre_down=None, thre_up=None)函数详解cov_matrix = df.cov()defrisk_parity(weights): sigma = np.sqrt(np.dot(np.dot(weights, cov_matrix), weights)) mrc = np.dot(cov_matrix, weights) / sigma trc = weights * mrc diff_trc_list = [sum((v - trc) ** 2) for v in trc] target = 10000 * sum(diff_trc_list)return targetweights0 = np.ones(df.shape[1]) / df.shape[1]cons = [{'type': 'eq', 'fun': lambda w: np.sum(w) - 1}]bounds = tuple((thre_down, thre_up) for w in weights0)if thre_down isNone: res = minimize(risk_parity, weights0, method='SLSQP', constraints=cons, options={'maxiter': 100})else: res = minimize(risk_parity, weights0, method='SLSQP', constraints=cons, bounds=bounds, options={'maxiter': 100})get_peg_filter_list(context, stocks, thre1, thre2)函数详解df_pe = get_fundamentals(stocks, 'valuation', fields='pe_dynamic', date=date_str)df_growth = get_fundamentals(stocks, 'growth_ability', fields='np_parent_company_yoy', date=date_str)# 统一索引格式if'secu_code'notin df_pe.columns and df_pe.index.name == 'secu_code': df_pe = df_pe.reset_index()df_pe = df_pe.rename(columns={'secu_code': 'code'}).set_index('code')# 同理处理df_growth...# 内连接合并df_merged = pd.concat([df_pe, df_growth], axis=1, join='inner')df_merged['peg'] = df_merged['pe_dynamic'] / df_merged['np_parent_company_yoy']df_filtered = df_merged[(df_merged['peg'] > thre1) & (df_merged['peg'] < thre2)]peg_stocks = list(df_filtered.index)df_cap = get_fundamentals(peg_stocks, 'valuation', fields='total_value', date=date_str)if df_cap isNoneor df_cap.empty:return peg_stocks # 保持原有顺序df_cap = df_cap.rename(columns={'secu_code': 'code'}).sort_values(by='total_value', ascending=True)return list(df_cap['code'])
PS: 源码下载,请移步知识星球!
星球中整理了文章中涉及到的源码,加入后即可以看到之前发的源码,三天内不满意可以退款。也会将收集到的机构研报,优秀策略源码,群友常咨询的问题整理到不同标签下,做为知识库沉淀下来。
所有策略仅用于学习和研究,不保证交易收益,不作为投资建议,风险自负,所有收益仅表示历史回测收益,不表示未来收益。大QMT和ptrade请充分使用模拟盘测试,miniQMT使用模拟账号测试。
听了群友的建议,年化收益达到了70%,增加了动态仓位权重调整后的全球核心资产轮动策略(含python代码解析)
欢迎扫描下方二维码,备注【开通】,开通QMT或Ptrade;备注【加群】,加入量化交易交流群;备注【电子书】,赠送QMT或Ptrade入门操作指南。

仅做知识整理,本公众号对这些信息的准确性和完整性不作任何保证,本材料不构成任何投资意见。投资有风险,入市需谨慎。