返回列表 发布新帖

QMT版的微盘股策略示例,能不能提供一份QMT版的微盘股策略示例?

1423 1
发表于 2023-12-19 20:44:57 | 显示全部楼层 阅读模式
本帖最后由 *******9406 于 2023-12-20 09:00 编辑

中信证券的版本没有xtdata.reconnect(port=58612)设置,是普通的qmt,能不能提供一份QMT版的微盘股策略示例?




  1. # coding:gbk
  2. # 根据情况指定xtquant的路径
  3. import sys
  4. import numpy as np
  5. import pandas as pd
  6. from xtquant import xtdata
  7. import xtquant
  8. print(xtquant)
  9. print(xtdata.data_dir)
  10. # 指定获取投研端数据(可不指定,默认优先连接投研)
  11. xtdata.reconnect(port=58612)
  12. # xtdata.download_sector_data()

  13. class G():
  14.     pass


  15. g = G()


  16. def init(C):
  17.     # ------------------------参数设定-----------------------------
  18.     g.his_st = {}

  19.     # g.s = C.get_stock_list_in_sector("沪深A股")  # 获取沪深A股股票列表
  20.     g.s = C.get_stock_list_in_sector("沪深300")  # 获取沪深300股票列表
  21.     # g.s = ['000001.SZ']
  22.     g.day = 0
  23.     g.holdings = {i: 0 for i in g.s}
  24.     g.weight = [0.1] * 10
  25.     g.buypoint = {}
  26.     g.money = 1000000  # C.capital
  27.     g.accid = 'test'
  28.     g.profit = 0
  29.     # 因子权重
  30.     g.buy_num = 10  # 买排名前5的股票,在过滤中会用到
  31.     g.per_money = g.money / g.buy_num * 0.95
  32.    



  33. def after_init(C):
  34.     # ------------------------量价数据获取-----------------------------
  35.     data = xtdata.get_market_data_ex([], g.s, period='1d', dividend_type='front_ratio',
  36.                                            fill_data=True)
  37.     close_df = get_df_ex(data,"close")
  38.     open_df = get_df_ex(data,"open")
  39.     low_df = get_df_ex(data,"low")
  40.     high_df = get_df_ex(data,"high")
  41.     volume_df = get_df_ex(data,"volume")
  42.     amount_df = get_df_ex(data,"amount")
  43.     preclose_df = get_df_ex(data,"preClose")

  44.     # ------------------------基础数据获取-----------------------------
  45.     # 将 g.s 中的全部股票的 TotalVolume 都获取出来,组合成一个 DataFrame
  46.     # 例如 g.s 中有 10 个股票,那么下面的代码就会返回一个 1 行 10 列的 DataFrame
  47.     # 该 DataFrame 的 index 是股票代码,columns 是 TotalVolume
  48.     # 该 DataFrame 的数据是每个股票的 TotalVolume,请给出代码:
  49.     # C.get_instrumentdetail('600000.SH')['TotalVolume']
  50.     # 使用字典推导来获取每个股票的TotalVolume
  51.     total_volumes = {stock: C.get_instrumentdetail(stock)['TotalVolume'] for stock in g.s}
  52.     # 将字典转换为DataFrame,但先转化为一个嵌套字典
  53.     df_total_volume = pd.DataFrame({k: v for k, v in total_volumes.items()}, index=['TotalVolume'])


  54.     # ------------------------财务数据获取-----------------------------

  55.     # ------------------------因子1计算及处理--------------------------------
  56.     # 1. 市值因子: 用市值因子 = 股票收盘价 * 股票总股本,要利用好,要求对应列名相乘
  57.     factor = close_df * df_total_volume.loc['TotalVolume']

  58.     # ------------------------因子2计算及处理--------------------------------
  59.     # 判断 close_df 中的 code 上市时间大于120天
  60.     stock_opendate_filter = filter_opendate_qmt(C, close_df, 120)

  61.     # ------------------------上市日期过滤处理--------------------------------
  62.     # 两个布尔值的 DataFrame 对应相乘过滤掉每个交易日上市不足120天的
  63.     factor *= stock_opendate_filter.astype(int).replace(0, np.nan)

  64.     # ------------------------排序处理-----------------------------------
  65.     # 对 factor 每行在一行内进行排序
  66.     factor_sorted = rank_filter(factor, 10, ascending=True, method='min', na_option='keep')


  67.     # ------------------------因子组合得到布尔值信号--------------------------------

  68.     # 确保没有未来数据的影响,将因子数据向后移动一天
  69.     g.factor_df = factor_sorted.shift(1)  #
  70.     g.close_df = close_df.shift(1)  # 为了计算收益率,将收盘价向后移动一天
  71.     g.open_df = open_df
  72.     g.stock_opendate_filter = stock_opendate_filter


  73. def handlebar(C):
  74.     # 获取当前 K 线位置
  75.     d = C.barpos
  76.     # 获取当前 K 线时间
  77.     backtest_time = timetag_to_datetime(C.get_bar_timetag(C.barpos), "%Y%m%d")
  78.     factor_series = g.factor_df.loc[backtest_time]
  79.     buy_list = daily_filter(factor_series, backtest_time)
  80.     print(backtest_time, buy_list)

  81.     # 获取持仓
  82.     hold = get_holdings(g.accid, 'stock')
  83.     need_sell = [s for s in hold if s not in buy_list]
  84.     print('\t\t\t\t\t\t\t', backtest_time, 'sell list', need_sell)

  85.     # 卖出
  86.     for s in need_sell:
  87.         price = g.open_df.loc[backtest_time, s]
  88.         vol = hold[s]['持仓数量']
  89.         passorder(24, 1101, g.accid, s, 11, price, vol, C)

  90.     # 获取持仓
  91.     hold = get_holdings(g.accid, 'stock')
  92.     asset = get_trade_detail_data(g.accid, 'stock', 'account')
  93.     cash = asset[0].m_dAvailable
  94.     buy_num = g.buy_num - len(hold)
  95.     buy_list = [s for s in buy_list if s not in hold]

  96.     # 买入
  97.     if buy_num > 0 and buy_list:
  98.         buy_list = buy_list[:buy_num]
  99.         # money = cash/buy_num
  100.         print(backtest_time, 'buy list', buy_list)
  101.         for s in buy_list:
  102.             price = g.open_df.loc[backtest_time, s]
  103.             if price > 0:
  104.                 passorder(23, 1102, g.accid, s, 11, float(price), g.per_money, C)


  105. def daily_filter(factor_series, backtest_time):
  106.     # 将 factor_series 中值 True 的index,转化成列表
  107.     print(len(factor_series))
  108.     sl = factor_series[factor_series].index.tolist()
  109.     print(len(sl))
  110.     # exit()
  111.     # st过滤
  112.     sl = [s for s in sl if not is_st(s, backtest_time)]
  113.     sl = sorted(sl, key=lambda k: factor_series.loc[k])
  114.     return sl[:g.buy_num]


  115. def is_st(s, date):
  116.     # 判断某日在历史上是不是st *st
  117.     st_dict = g.his_st.get(s, {})
  118.     if not st_dict:
  119.         return False
  120.     else:
  121.         st = st_dict.get('ST', []) + st_dict.get('*ST', [])
  122.         for start, end in st:
  123.             if start <= date <= end:
  124.                 return True


  125. def get_df(dt: dict, df: pd.DataFrame, values_name: str) -> pd.DataFrame:
  126.     '''
  127.     循环从字典里赋值矩阵
  128.     values_name可选字段: ['time', 'stime', 'open', 'high', 'low', 'close', 'volume','amount', 'settelementPrice', 'openInterest', 'preClose', 'suspendFlag']
  129.     '''
  130.     df1 = df.copy()
  131.     df1 = df1.apply(lambda x: dt[x.name][values_name])

  132.     return df1

  133. def get_df_ex(data:dict,field:str) -> pd.DataFrame:
  134.     '''
  135.     ToDo:用于在使用get_market_data_ex的情况下,取到标准df
  136.    
  137.     Args:
  138.         data: get_market_data_ex返回的dict
  139.         field: ['time', 'open', 'high', 'low', 'close', 'volume','amount', 'settelementPrice', 'openInterest', 'preClose', 'suspendFlag']
  140.         
  141.     Return:
  142.         一个以时间为index,标的为columns的df
  143.    
  144.     '''
  145.     _index = data[list(data.keys())[0]].index.tolist()
  146.     _columns = list(data.keys())
  147.     df = pd.DataFrame(index=_index,columns=_columns)
  148.     for i in _columns:
  149.         df[i] = data[i][field]
  150.     return df
  151.         

  152. def rank_filter(df: pd.DataFrame, N: int, axis=1, ascending=False, method="max", na_option="keep") -> pd.DataFrame:
  153.     """
  154.     Args:
  155.         df: 标准数据的df
  156.         N: 判断是否是前N名
  157.         axis: 默认是横向排序
  158.         ascending : 默认是降序排序
  159.         na_option : 默认保留nan值,但不参与排名
  160.     Return:
  161.         pd.DataFrame:一个全是bool值的df
  162.     """
  163.     _df = df.copy()

  164.     _df = _df.rank(axis=axis, ascending=ascending, method=method, na_option=na_option)

  165.     return _df <= N


  166. def filter_opendate_qmt(C, df: pd.DataFrame, n: int) -> pd.DataFrame:
  167.     '''

  168.     ToDo: 判断传入的df.columns中,上市天数是否大于N日,返回的值是一个全是bool值的df

  169.     Args:
  170.         C:contextinfo类
  171.         df:index为时间,columns为stock_code的df,目的是为了和策略中的其他df对齐
  172.         n:用于判断上市天数的参数,如要判断是否上市120天,则填写
  173.     Return:pd.DataFrame

  174.     '''
  175.     local_df = pd.DataFrame(index=df.index, columns=df.columns)
  176.     stock_list = df.columns
  177.     stock_opendate = {i: C.get_instrument_detail(i)["OpenDate"] for i in stock_list}
  178.     # print(type(stock_opendate["000001.SZ"]), stock_opendate["000001.SZ"])
  179.     for stock, date in stock_opendate.items():
  180.         local_df.at[date, stock] = 1
  181.     df_fill = local_df.fillna(method="ffill")

  182.     result = df_fill.expanding().sum() >= n

  183.     return result


  184. def filter_opendate_xt(df: pd.DataFrame, n: int) -> pd.DataFrame:
  185.     '''

  186.     ToDo: 判断传入的df.columns中,上市天数是否大于N日,返回的值是一个全是bool值的df

  187.     Args:
  188.         C:contextinfo类
  189.         df:index为时间,columns为stock_code的df,目的是为了和策略中的其他df对齐
  190.         n:用于判断上市天数的参数,如要判断是否上市120天,则填写
  191.     Return:pd.DataFrame

  192.     '''
  193.     local_df = pd.DataFrame(index=df.index, columns=df.columns)
  194.     stock_list = df.columns
  195.     stock_opendate = {i: xtdata.get_instrument_detail(i)["OpenDate"] for i in stock_list}
  196.     for stock, date in stock_opendate.items():
  197.         local_df.at[date, stock] = 1
  198.     df_fill = local_df.fillna(method="ffill")

  199.     result = df_fill.expanding().sum() >= n

  200.     return result


  201. def get_holdings(accid, datatype):
  202.     '''
  203.     Arg:
  204.         accondid:账户id
  205.         datatype:
  206.             'FUTURE':期货
  207.             'STOCK':股票
  208.             ......
  209.     return:
  210.         {股票名:{'手数':int,"持仓成本":float,'浮动盈亏':float,"可用余额":int}}
  211.     '''
  212.     PositionInfo_dict = {}
  213.     resultlist = get_trade_detail_data(accid, datatype, 'POSITION')
  214.     for obj in resultlist:
  215.         PositionInfo_dict[obj.m_strInstrumentID + "." + obj.m_strExchangeID] = {
  216.             "持仓数量": obj.m_nVolume,
  217.             "持仓成本": obj.m_dOpenPrice,
  218.             "浮动盈亏": obj.m_dFloatProfit,
  219.             "可用余额": obj.m_nCanUseVolume
  220.         }
  221.     return PositionInfo_dict


  222. if __name__ == '__main__':
  223.     import sys
  224.     from xtquant.qmttools import run_strategy_file

  225.     # 参数定义方法一,如果使用方法二定义参数,run_strategy_file的param参数可不传
  226.     param = {
  227.         'stock_code': '000300.SH',  # 驱动handlebar的代码,
  228.         'period': '1d',  # 策略执行周期 即主图周期
  229.         'start_time': '2017-01-01 00:00:00',  # 注意格式,不要写错
  230.         'end_time': '2020-12-31 00:00:00',  # 注意格式,不要写错
  231.         'trade_mode': 'backtest',  # simulation': 模拟, 'trading':实盘, 'backtest':回测
  232.         'quote_mode': 'history',
  233.         # handlebar模式,'realtime':仅实时行情(不调用历史行情的handlebar),'history':仅历史行情, 'all':所有,即history+realtime
  234.     }
  235.     # user_script = os.path.basename(__file__)  # 当前脚本路径,相对路径,绝对路径均可,此处为相对路径的方法
  236.     user_script = sys.argv[0]  # 当前脚本路径,相对路径,绝对路径均可,此处为绝对路径的方法

  237.     print(user_script)
  238.     run_strategy_file(user_script, param=param)
复制代码

评论1

davidfnck
发表于 2023-12-20 13:52:27 | 显示全部楼层
1. 如果在大QMT里面用内置Python跑,可以把
  1. # xtdata.reconnect(port=58612)
复制代码
注释掉,内置Python会自动读取客户端的数据
2. 如果有定制的策略需求,可以看看官网上的策略代写服务:https://xuntou.net/#/strategyvip?choose=D2&id=AFsgJ2

回复

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

客服专线

400-080-8112

用思考的速度交易,用真诚的态度合作,我们是认真的!
  • 关注公众号
  • 添加微信客服
Copyright © 2001-2025 迅投QMT社区 版权所有 All Rights Reserved. 蜀ICP备19002686号-2
关灯 快速发帖
扫一扫添加微信客服
QQ客服返回顶部
快速回复 返回顶部 返回列表