Backtrader(二十二)- 混合时间粒度回测

简述

当要使用不同时间粒度的行情数据时,backtrader有两种处理方式,第一种是读入不同的行情bar数据,第二种是将细粒度数据合并成粗粒度数据(resample)

读入不同的行情bar数据
规则
  • 最小粒度的数据必须被第一个加入到Cerebro实例中
  • 数据必须按照日期时间升序排列
  • 大粒度数据的使用有可能使得策略的最小周期变大
  • 若使用不同时间粒度的指标进行运算,则要采用cerebro.run(runonce=False) 若策略中使用了不同时间粒度的指标进行运算,不能用cerebro.run()
  • init方法中不同粒度的指标执行混合运算时,大粒度的指标后要加括号()
使用resample合成大时间粒度数据
用法:

cerebro.resampledata(data, timeframe=bt.TimeFrame.Minutes, compression=1)
data: bt数据源
timeframe:用于指定合成结果(Ticks, MicroSeconds, Seconds, Minutes, Days, Weeks, Months, Years, NoTimeFrame)
compression:设置bar的条数进一步压缩合并,如resmaple初次合成为周数据后,每根bar是周线。如设置compression=2,则进一步将两条周数据bar压缩成 一根bar为两周数据。

注意

如果采用了resmaple方法,则数据不会预加载preload。

1、日线合并成周线

'''重采样合成大粒度数据 日线合成周线'''

import datetime
import backtrader as bt
from feed import feed, feed_weekly
from logger import lg
import backtrader.indicators as btind


class SmaCross(bt.Strategy):

    def __init__(self):
        # 5日移动平均线指标
        self.move_average = bt.ind.MovingAverageSimple(
            self.data.close,
            period=5
        )
        # 2周线移动平均
        self.move_average2 = bt.ind.MovingAverageSimple(
            self.datas[1].close,
            period=2
        )
        # 日线交叉信号指标
        self.crossover = bt.ind.CrossOver(
            self.data.close,
            self.move_average
        )

    def log(self, txt, dt=None):
        dt = dt or self.data.datetime.date(0)
        lg.info(
            '%s, %s' % (dt.isoformat(), txt)
        )

    def notify_order(self, order):
        if order.status in [order.Submitted, order.Accepted]:
            return

        if order.status in [order.Completed]:
            if order.isbuy():
                self.log(
                    f'Buy {order.executed.price}'
                )
            elif order.issell():
                self.log(
                    f'Sell {order.executed.price}'
                )

    def notify_trade(self, trade):
        if trade.isclosed:
            self.log(
                f'''
            毛收益 {trade.pnl},
            扣佣后收益 {trade.pnlcomm},
            佣金 {trade.commission}
            '''
            )

    def next(self):
        if not self.position:
            if self.crossover > 0 and self.move_average[0] > self.move_average2[0]:
                self.buy()

        elif self.crossover < 0 and self.move_average[0] < self.move_average2[0]:
            self.close()

if __name__ == '__main__':
    cerebro = bt.Cerebro()
    cerebro.adddata(feed)

    cerebro.addstrategy(SmaCross)

    cerebro.resampledata(
        feed,
        timeframe=bt.TimeFrame.Weeks,
        compression=1
    )

    cerebro.broker.setcash(10000.0)
    cerebro.broker.setcommission(0.001)
    cerebro.broker.set_slippage_fixed(0.05)
    lg.error(f'市值 {cerebro.broker.getvalue()}')
    cerebro.run(stdstats=False, runonce=False)
    lg.error(f'市值 {cerebro.broker.getvalue()}')
    cerebro.plot()

2、5分钟线合并成10分钟线

'''重采样合成大粒度数据 5分钟线合成10分钟线'''
from datetime import datetime
import backtrader as bt



class TestStrategy(bt.Strategy):
    def next(self):
        print(
            f'''
        {self.data0.datetime.datetime(0)},
        {len(self.data0)},
        {self.data0.open[0]},
        {self.data0.high[0]},
        {self.data0.low[0]},
        {self.data0.close[0]},
        '''
        )

if __name__ == '__main__':
    cerebro = bt.Cerebro(stdstats=False)
    cerebro.addstrategy(TestStrategy)

    feed = bt.feeds.BacktraderCSVData(
        dataname='min.csv',
        timeframe=bt.TimeFrame.Minutes,
        todate=datetime(2016, 1, 3)
    )

    cerebro.resampledata(
        feed,
        timeframe=bt.TimeFrame.Minutes,
        compression=2  # 5min整合10分钟的关键
    )

    cerebro.run()

3、Tick数据合成 1分钟数据

'''Tick数据合成 1分钟数据'''

from datetime import datetime
import backtrader as bt


class TestStrategy(bt.Strategy):
    def next(self):
        print(
            f'''
        {self.data0.datetime.datetime(0)},
        {len(self.data0)},
        {self.data0.open[0]},
        {self.data0.high[0]},
        {self.data0.low[0]},
        {self.data0.close[0]},
        '''
        )

if __name__ == '__main__':
    cerebro = bt.Cerebro(stdstats=False)
    cerebro.addstrategy(TestStrategy)

    feed = bt.feeds.GenericCSVData(
        dataname='tick.csv',
        dtformat='%Y-%m-%dT%H:%M:%S.%f',
        timeframe=bt.TimeFrame.Ticks
    )

    cerebro.resampledata(
        feed,
        timeframe=bt.TimeFrame.Minutes,
        compression=1
    )

    cerebro.run()

  • 1
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
设计模式是在软件设计过程中,对于常见问题的解决方案的总结和模板化的提炼。Java语言中广泛应用的设计模式有23种,分为三个大类:创建型模式、结构型模式和行为型模式。 首先是创建型模式,包括单例模式、简单工厂模式、工厂方法模式、抽象工厂模式、建造者模式和原型模式。单例模式确保一个类只有一个实例化对象,简单工厂模式通过一个工厂类来创建对象,工厂方法模式通过使用工厂接口来创建对象,抽象工厂模式为一个产品族提供创建对象的接口,建造者模式逐步构建一个复杂对象,原型模式通过复制现有对象来创建新对象。 其次是结构型模式,包括适配器模式、桥接模式、装饰器模式、组合模式、外观模式、享元模式和代理模式。适配器模式用于将一个接口转换成客户端所期待的接口,桥接模式将抽象和实现解耦,装饰器模式动态地给对象添加额外职责,组合模式将对象组合成树形结构以表示“部分-整体”的层次结构,外观模式提供了一个简化客户端调用复杂子系统的接口,享元模式通过共享对象来有效地支持大量细粒度的对象,代理模式为其他对象提供一种代理以控制对这个对象的访问。 最后是行为型模式,包括模板方法模式、命令模式、迭代器模式、观察者模式、中介者模式、备忘录模式、解释器模式、状态模式、策略模式和访问者模式。模板方法模式定义一个操作中的算法的骨架,而将一些步骤延迟到子类中实现,命令模式将请求封装成一个对象,以使不同的请求能够参数化客户端对象,迭代器模式提供一种访问容器对象中各个元素的方式,观察者模式定义了对象之间的一对多依赖,中介者模式用一个中介对象来封装一系列对象的交互,备忘录模式在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态,解释器模式给定一个语言,定义它的文法的一种表示,并定义一个解释器,状态模式允许一个对象在其内部状态改变时改变它的行为,策略模式定义了算法家族,分别封装起来,让它们之间可以互相替换,以使得算法的变化不影响到使用算法的客户,访问者模式可以在不改变数据结构的前提下,定义作用于这个结构元素的新操作。通过使用这些设计模式,可以提高软件系统的可维护性、可扩展性和可重用性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值