Backtrader réalise la véritable stratégie d'opération de combinaison multi-brins, il suffit de lire celle-ci, d'utiliser la minuterie pour entrer dans la logique de la stratégie

Les stratégies suivantes proviennent de tutoriels et de cours vidéo que nous avons écrits .

Le fonctionnement combiné à plusieurs brins est un mode de fonctionnement avancé. Il existe généralement deux modes pour les opérations de portefeuille multi-actions: le premier est le rééquilibrage régulier , c'est-à-dire le rééquilibrage régulier, tel qu'un rééquilibrage une fois par semaine ou un rééquilibrage le 1er de chaque mois. L'autre est un ajustement de position non régulier , tel qu'un jugement et un ajustement quotidiens, ou un ajustement tous les n jours.

L'ajustement de position périodique (rééquilibrage) entre généralement dans la logique d'ajustement de position par le biais d'une minuterie, tandis que l'ajustement de position non périodique entre généralement dans la logique d'ajustement de position par la méthode traditionnelle de stratégie suivante. Bien sûr, ce n'est pas absolu.

Ces deux opérations d'ajustement multi-part ne peuvent pas être effectuées avec la méthode backtrader intégrée de détermination automatique de la période minimale (par exemple, pour trouver la moyenne mobile sur 20 jours, sautez automatiquement les 20 premières barres), car certaines actions ont dates de négociation très tardives, sa période minimale est très longue, et d'autres actions utiliseront également cette période minimale, ce qui entraînera un gaspillage de données d'autres stocks avant la période minimale. Par conséquent, vous devez contrôler vous-même la période minimale, c'est-à-dire que vous devez écrivez self.next () dans la méthode prenext. Passez directement au suivant. Ensuite, si vous utilisez des indicateurs tels que la moyenne mobile sur 5 jours, vous devez juger par vous-même si la longueur de la ligne d'objet de données est suffisamment longue. La stratégie de cas suivante n'utilise pas de tels indicateurs techniques, donc la longueur de la ligne n'est pas jugée. En théorie, la logique peut être exécutée à partir de la première barre.

Bien qu'il existe un cas d'utilisation de backtrader pour effectuer un backtesting de portefeuilles multi-actions sur Internet, certains problèmes de backtesting de multi-actions n'ont pas été bien gérés. Cet article vous donnera une solution complète.

(Veuillez vous référer à nos tutoriels et cours vidéo pour la mise en œuvre de stratégies de rééquilibrage non périodique en fonction de la prochaine)

Cet article présente la mise en œuvre de la stratégie de rééquilibrage périodique multi-brins basée sur le minuteur.

Le but de ce cas est d'introduire quelques points techniques à noter lors de l'utilisation de backtrader pour la gestion de portefeuille, la stratégie elle-même est à titre indicatif. La logique générale de la stratégie est la suivante: chaque année, les 1er mai, 1er septembre et 1er novembre, l'opération de rééquilibrage est réalisée (si le marché est fermé ce jour-là, l'opération de rééquilibrage sera reportée au jour d'ouverture du marché).

Chargez d'abord un groupe d'actions (stock pool). Le jour du rééquilibrage, sélectionnez les actions du stock pool qui sont cotées depuis au moins 3 ans et dont le rendement du ratio d'actif net est supérieur à 0,1 et un ratio cours-bénéfice pe compris entre 0 et 100. Ce groupe d'actions sélectionnées Ensuite, triez les 100 premières actions en fonction du volume des transactions du grand au petit (si les actions sélectionnées sont inférieures à 100, alors en fonction de la situation réelle), toutes les valeurs du compte sont attribuées acheter ces actions dans des proportions égales.

Cette stratégie reflète les points techniques suivants. Si vous comprenez ces points, vous pouvez essentiellement les utiliser en combat réel. Pour une interprétation plus détaillée du code, en particulier l'utilisation du minuteur, veuillez vous référer aux didacticiels et cours vidéo que nous avons écrits:

1 Étendre la classe PandasData

2 Les premières données doivent correspondre à l'index, car la base de temps

3 Prétraitement des données: supprimer les enregistrements sans transactions et indicateurs manquants dans les données d'origine

4 Fermez d'abord la position, puis exécutez les transactions suivantes

5 Méthode de calcul de la quantité commandée

6 Comment vous assurer de vendre d'abord, puis d'acheter pour libérer des fonds

7 Comment calculer la quantité commandée en fonction du prix d'ouverture de demain

8 Fournir des noms pour les objets de données de marché

9 Comment définir la quantité d'achat et de vente sur un multiple entier de 100

10 Mettre en place un modèle de commission en adéquation avec la bourse chinoise et envisager le droit de timbre

11 Traitement de la limite de prix

# 考虑中国佣金,下单量100的整数倍,涨跌停板,滑点
# 考虑一个技术指标,展示怎样处理最小期问题

from datetime import datetime, time
from datetime import timedelta
import pandas as pd
import numpy as np
import backtrader as bt
import os.path  # 管理路径
import sys  # 发现脚本名字(in argv[0])
import glob
from backtrader.feeds import PandasData  # 用于扩展DataFeed

# 创建新的data feed类


class PandasDataExtend(PandasData):
    # 增加线
    lines = ('pe', 'roe', 'marketdays')
    params = (('pe', 15),
              ('roe', 16),
              ('marketdays', 17), )  # 上市天数


class stampDutyCommissionScheme(bt.CommInfoBase):
    '''
    本佣金模式下,买入股票仅支付佣金,卖出股票支付佣金和印花税.    
    '''
    params = (
        ('stamp_duty', 0.005),  # 印花税率
        ('commission', 0.001),  # 佣金率
        ('stocklike', True),
        ('commtype', bt.CommInfoBase.COMM_PERC),
    )

    def _getcommission(self, size, price, pseudoexec):
        '''
        If size is greater than 0, this indicates a long / buying of shares.
        If size is less than 0, it idicates a short / selling of shares.
        '''

        if size > 0:  # 买入,不考虑印花税
            return size * price * self.p.commission
        elif size < 0:  # 卖出,考虑印花税
            return size * price * (self.p.stamp_duty + self.p.commission)
        else:
            return 0  # just in case for some reason the size is 0.


class Strategy(bt.Strategy):
    params = dict(
        rebal_monthday=[1],  # 每月1日执行再平衡
        num_volume=100,  # 成交量取前100名
        period = 5,
    )

    # 日志函数
    def log(self, txt, dt=None):
        # 以第一个数据data0,即指数作为时间基准
        dt = dt or self.data0.datetime.date(0)
        print('%s, %s' % (dt.isoformat(), txt))

    def __init__(self):
        
        self.lastRanks = []  # 上次交易股票的列表
        # 0号是指数,不进入选股池,从1号往后进入股票池
        self.stocks = self.datas[1:]
        # 记录以往订单,在再平衡日要全部取消未成交的订单
        self.order_list = []

        # 移动平均线指标 
        self.sma={d:bt.ind.SMA(d,period=self.p.period) for d in self.stocks}
     
       
        # 定时器
        self.add_timer(
            when= bt.Timer.SESSION_START,            
            monthdays=self.p.rebal_monthday,  # 每月1号触发再平衡
            monthcarry=True,  # 若再平衡日不是交易日,则顺延触发notify_timer
        
            
        )

    def notify_timer(self, timer, when, *args, **kwargs):

   
        # 只在5,9,11月的1号执行再平衡
        if self.data0.datetime.date(0).month in [5,9,11]:
            self.rebalance_portfolio()  # 执行再平衡
       

  
    # def next(self):
        
    #     print('next 账户总值', self.data0.datetime.datetime(0), self.broker.getvalue())
    #     for d in self.stocks:
    #         if(self.getposition(d).size!=0):
    #             print(d._name, '持仓' ,self.getposition(d).size)
     
    def notify_order(self, order):
        if order.status in [order.Submitted, order.Accepted]:
            # 订单状态 submitted/accepted,无动作
            return

        # 订单完成
        if order.status in [order.Completed]:
            if order.isbuy():
                self.log('买单执行,%s, %.2f, %i' % (order.data._name,
                                                order.executed.price, order.executed.size))

            elif order.issell():
                self.log('卖单执行, %s, %.2f, %i' % (order.data._name,
                                                 order.executed.price, order.executed.size))

        else:
            self.log('订单作废 %s, %s, isbuy=%i, size %i, open price %.2f' %
                     (order.data._name, order.getstatusname(), order.isbuy(), order.created.size, order.data.open[0]))

    # 记录交易收益情况
    def notify_trade(self, trade):
        if trade.isclosed:
            print('毛收益 %0.2f, 扣佣后收益 % 0.2f, 佣金 %.2f, 市值 %.2f, 现金 %.2f' %
                  (trade.pnl, trade.pnlcomm, trade.commission, self.broker.getvalue(), self.broker.getcash()))

    def rebalance_portfolio(self):
        # 从指数取得当前日期
        self.currDate = self.data0.datetime.date(0)
        print('rebalance_portfolio currDate', self.currDate, len(self.stocks))

        # 如果是指数的最后一本bar,则退出,防止取下一日开盘价越界错
        if len(self.datas[0]) == self.data0.buflen():
            return     
        
        
        # 取消以往所下订单(已成交的不会起作用)
        for o in self.order_list:
            self.cancel(o)
        self.order_list = []  # 重置订单列表

        
        # for d in self.stocks:
        #     print('sma', d._name, self.sma[d][0],self.sma[d][1], d.marketdays[0])
        
        # 最终标的选取过程
        # 1 先做排除筛选过程
        self.ranks = [d for d in self.stocks if
                      len(d) > 0  # 重要,到今日至少要有一根实际bar
                      and d.marketdays > 3*365  # 到今天至少上市
                      # 今日未停牌 (若去掉此句,则今日停牌的也可能进入,并下订单,次日若复牌,则次日可能成交)(假设原始数据中已删除无交易的记录)
                      and d.datetime.date(0) == self.currDate
                      and d.roe >= 0.1
                      and d.pe < 100
                      and d.pe > 0
                      and len(d) >= self.p.period
                      and d.close[0] > self.sma[d][1]
                      ]

        # 2 再做排序挑选过程
        self.ranks.sort(key=lambda d: d.volume, reverse=True)  # 按成交量从大到小排序
        self.ranks = self.ranks[0:self.p.num_volume]  # 取前num_volume名

        if len(self.ranks) == 0:  # 无股票选中,则返回
            return

        # 3 以往买入的标的,本次不在标的中,则先平仓
        data_toclose = set(self.lastRanks) - set(self.ranks)
        for d in data_toclose:
            print('sell 平仓', d._name, self.getposition(d).size)
            o = self.close(data=d)            
            self.order_list.append(o) # 记录订单

        # 4 本次标的下单
        # 每只股票买入资金百分比,预留2%的资金以应付佣金和计算误差
        buypercentage = (1-0.02)/len(self.ranks)

        # 得到目标市值
        targetvalue = buypercentage * self.broker.getvalue()
        # 为保证先卖后买,股票要按持仓市值从大到小排序
        self.ranks.sort(key=lambda d: self.broker.getvalue([d]), reverse=True)
        self.log('下单, 标的个数 %i, targetvalue %.2f, 当前总市值 %.2f' %
                 (len(self.ranks), targetvalue, self.broker.getvalue()))

        for d in self.ranks:
            # 按次日开盘价计算下单量,下单量是100的整数倍
            size = int(
                abs((self.broker.getvalue([d]) - targetvalue) / d.open[1] // 100 * 100))
            validday = d.datetime.datetime(1)  # 该股下一实际交易日
            if self.broker.getvalue([d]) > targetvalue:  # 持仓过多,要卖
                # 次日跌停价近似值
                lowerprice = d.close[0]*0.9+0.02

                o = self.sell(data=d, size=size, exectype=bt.Order.Limit,
                              price=lowerprice, valid=validday)
            else:  # 持仓过少,要买
                # 次日涨停价近似值
                upperprice = d.close[0]*1.1-0.02
                o = self.buy(data=d, size=size, exectype=bt.Order.Limit,
                             price=upperprice, valid=validday)
                             
            self.order_list.append(o) # 记录订单

        
        self.lastRanks = self.ranks  # 跟踪上次买入的标的


##########################
# 主程序开始
#########################
cerebro = bt.Cerebro(stdstats=False)
cerebro.addobserver(bt.observers.Broker)
cerebro.addobserver(bt.observers.Trades)
# cerebro.broker.set_coc(True)  # 以订单创建日的收盘价成交
# cerebro.broker.set_coo(True) # 以次日开盘价成交


datadir = './dataswind'  # 数据文件位于本脚本所在目录的data子目录中
datafilelist = glob.glob(os.path.join(datadir, '*'))  # 数据文件路径列表

maxstocknum = 20  # 股票池最大股票数目
# 注意,排序第一个文件必须是指数数据,作为时间基准
datafilelist = datafilelist[0:maxstocknum]  # 截取指定数量的股票池
print(datafilelist)
# 将目录datadir中的数据文件加载进系统


for fname in datafilelist:
    
    df = pd.read_csv(
            fname,   
            skiprows=0,  # 不忽略行
            header=0,  # 列头在0行
        )
    # df = df[~df['交易状态'].isin(['停牌一天'])]  # 去掉停牌日记录
    df['date'] = pd.to_datetime(df['date'])  # 转成日期类型   
    df = df.dropna()

    # print(df.info())
    # print(df.head())
 
        
    data = PandasDataExtend(
            dataname=df,
            datetime=0,  # 日期列
            open=2,  # 开盘价所在列
            high=3,  # 最高价所在列
            low=4,  # 最低价所在列
            close=5,  # 收盘价价所在列
            volume=6,  # 成交量所在列
            pe=7,
            roe=8,
            marketdays=9,
            openinterest=-1,  # 无未平仓量列
            fromdate=datetime(2002, 4, 1),  # 起始日2002, 4, 1
            todate=datetime(2015, 12, 31),  # 结束日 2015, 12, 31
            plot=False

        )
    ticker = fname[-13:-4]  # 从文件路径名取得股票代码

    cerebro.adddata(data, name=ticker)


cerebro.addstrategy(Strategy)
startcash = 10000000
cerebro.broker.setcash(startcash)
# 防止下单时现金不够被拒绝。只在执行时检查现金够不够。
cerebro.broker.set_checksubmit(False)
comminfo = stampDutyCommissionScheme(stamp_duty=0.001, commission=0.001)
cerebro.broker.addcommissioninfo(comminfo)
results = cerebro.run()
print('最终市值: %.2f' % cerebro.broker.getvalue())
# cerebro.plot()
复制代码

La stratégie ci-dessus peut également être entrée dans la logique de la stratégie via la méthode suivante. Pour le code et les détails spécifiques, veuillez consulter notre tutoriel.


Auteur: Sweeping Monk 2020
 

Je suppose que tu aimes

Origine blog.csdn.net/qtbgo/article/details/110366241
conseillé
Classement