backtrader指标

68 篇文章 14 订阅
13 篇文章 22 订阅

添加分析指标

# 添加分析指标
# 返回年初至年末的年度收益率
cerebro.addanalyzer(bt.analyzers.AnnualReturn, _name='_AnnualReturn')
# 计算最大回撤相关指标
cerebro.addanalyzer(bt.analyzers.DrawDown, _name='_DrawDown')
# 计算年化收益:日度收益
cerebro.addanalyzer(bt.analyzers.Returns, _name='_Returns', tann=252)
# 计算年化夏普比率:日度收益
cerebro.addanalyzer(bt.analyzers.SharpeRatio, _name='_SharpeRatio', timeframe=bt.TimeFrame.Days, annualize=True, riskfreerate=0) # 计算夏普比率
cerebro.addanalyzer(bt.analyzers.SharpeRatio_A, _name='_SharpeRatio_A')
# 返回收益率时序
cerebro.addanalyzer(bt.analyzers.TimeReturn, _name='_TimeReturn')
# 启动回测
result = cerebro.run()

# 提取结果
print("--------------- AnnualReturn -----------------")
print(result[0].analyzers._AnnualReturn.get_analysis())
print("--------------- DrawDown -----------------")
print(result[0].analyzers._DrawDown.get_analysis())
print("--------------- Returns -----------------")
print(result[0].analyzers._Returns.get_analysis())
print("--------------- SharpeRatio -----------------")
print(result[0].analyzers._SharpeRatio.get_analysis())
print("--------------- SharpeRatio_A -----------------")
print(result[0].analyzers._SharpeRatio_A.get_analysis())

# 常用指标提取
analyzer = {}
# 提取年化收益
analyzer['年化收益率'] = result[0].analyzers._Returns.get_analysis()['rnorm']
analyzer['年化收益率(%)'] = result[0].analyzers._Returns.get_analysis()['rnorm100']
# 提取最大回撤
analyzer['最大回撤(%)'] = result[0].analyzers._DrawDown.get_analysis()['max']['drawdown'] * (-1)
# 提取夏普比率
analyzer['年化夏普比率'] = result[0].analyzers._SharpeRatio_A.get_analysis()['sharperatio']

# 日度收益率序列
ret = pd.Series(result[0].analyzers._TimeReturn.get_analysis())

下面是在 Backtrader 社区中找到的自定义分析器,用于查看每笔交易盈亏情况:

地址:

https://community.backtrader.com/topic/1274/closed-trade-list-including-mfe-mae-analyzer;

该案例涉及到 trade 对象的相关属性,具体可以参考官方文档:https://www.backtrader.com/docu/trade/ 。

class trade_list(bt.Analyzer):
    def __init__(self):

        self.trades = []
        self.cumprofit = 0.0

    def notify_trade(self, trade):

        if trade.isclosed:
            brokervalue = self.strategy.broker.getvalue()

            dir = 'short'
            if trade.history[0].event.size > 0: dir = 'long'

            pricein = trade.history[len(trade.history)-1].status.price
            priceout = trade.history[len(trade.history)-1].event.price
            datein = bt.num2date(trade.history[0].status.dt)
            dateout = bt.num2date(trade.history[len(trade.history)-1].status.dt)
            if trade.data._timeframe >= bt.TimeFrame.Days:
                datein = datein.date()
                dateout = dateout.date()

            pcntchange = 100 * priceout / pricein - 100
            pnl = trade.history[len(trade.history)-1].status.pnlcomm
            pnlpcnt = 100 * pnl / brokervalue
            barlen = trade.history[len(trade.history)-1].status.barlen
            pbar = pnl / barlen
            self.cumprofit += pnl

            size = value = 0.0
            for record in trade.history:
                if abs(size) < abs(record.status.size):
                    size = record.status.size
                    value = record.status.value

            highest_in_trade = max(trade.data.high.get(ago=0, size=barlen+1))
            lowest_in_trade = min(trade.data.low.get(ago=0, size=barlen+1))
            hp = 100 * (highest_in_trade - pricein) / pricein
            lp = 100 * (lowest_in_trade - pricein) / pricein
            if dir == 'long':
                mfe = hp
                mae = lp
            if dir == 'short':
                mfe = -lp
                mae = -hp

            self.trades.append({'ref': trade.ref,
             'ticker': trade.data._name,
             'dir': dir,
             'datein': datein,
             'pricein': pricein,
             'dateout': dateout,
             'priceout': priceout,
             'chng%': round(pcntchange, 2),
             'pnl': pnl, 'pnl%': round(pnlpcnt, 2),
             'size': size,
             'value': value,
             'cumpnl': self.cumprofit,
             'nbars': barlen, 'pnl/bar': round(pbar, 2),
             'mfe%': round(mfe, 2), 'mae%': round(mae, 2)})
            
    def get_analysis(self):
        return self.trades


# 添加自定义的分析指标
cerebro.addanalyzer(trade_list, _name='tradelist')

# 启动回测
result = cerebro.run(tradehistory=True)

# 返回结果
ret = pd.DataFrame(result[0].analyzers.tradelist.get_analysis())

参数优化:

该功能只需通过 cerebro.optstrategy() 方法往大脑添加策略。

class TestStrategy(bt.Strategy):
  
    params=(('period1',5),
            ('period2',10),) #全局设定均线周期
    ......

    
# 实例化大脑
cerebro1= bt.Cerebro(optdatas=True, optreturn=True)
# 设置初始资金
cerebro1.broker.set_cash(10000000)
# 加载数据
datafeed1 = bt.feeds.PandasData(dataname=data1, fromdate=datetime.datetime(2019,1,2), todate=datetime.datetime(2021,1,28))
cerebro1.adddata(datafeed1, name='600466.SH')

# 添加优化器
cerebro1.optstrategy(TestStrategy, period1=range(5, 25, 5), period2=range(10, 41, 10))

# 添加分析指标
# 返回年初至年末的年度收益率
cerebro1.addanalyzer(bt.analyzers.AnnualReturn, _name='_AnnualReturn')
# 计算最大回撤相关指标
cerebro1.addanalyzer(bt.analyzers.DrawDown, _name='_DrawDown')
# 计算年化收益
cerebro1.addanalyzer(bt.analyzers.Returns, _name='_Returns', tann=252)
# 计算年化夏普比率
cerebro1.addanalyzer(bt.analyzers.SharpeRatio_A, _name='_SharpeRatio_A')
# 返回收益率时序
cerebro1.addanalyzer(bt.analyzers.TimeReturn, _name='_TimeReturn')

# 启动回测
result = cerebro1.run()

# 打印结果
def get_my_analyzer(result):
    analyzer = {}
    # 返回参数
    analyzer['period1'] = result.params.period1
    analyzer['period2'] = result.params.period2
    # 提取年化收益
    analyzer['年化收益率'] = result.analyzers._Returns.get_analysis()['rnorm']
    analyzer['年化收益率(%)'] = result.analyzers._Returns.get_analysis()['rnorm100']
    # 提取最大回撤(习惯用负的做大回撤,所以加了负号)
    analyzer['最大回撤(%)'] = result.analyzers._DrawDown.get_analysis()['max']['drawdown'] * (-1)
    # 提取夏普比率
    analyzer['年化夏普比率'] = result.analyzers._SharpeRatio_A.get_analysis()['sharperatio']
    
    return analyzer

ret = []
for i in result:
    ret.append(get_my_analyzer(i[0]))
    
pd.DataFrame(ret)

  • 1
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: 在 backtrader 库中,可以使用 `bt.indicators.Distance` 指标来计算价格和另一个指标之间的距离,该指标可以用于表示价格与均线之间的背离程度。 以下是使用 `bt.indicators.Distance` 指标来表示收盘价和 20 日简单移动平均线之间的距离的示例代码: ```python import backtrader as bt class MyStrategy(bt.Strategy): def __init__(self): # 创建收盘价指标和 20 日简单移动平均线指标 self.close = self.datas[0].close self.sma20 = bt.indicators.SMA(self.close, period=20) # 创建收盘价与 20 日简单移动平均线之间的距离指标 self.dist = bt.indicators.Distance(self.close, self.sma20) def next(self): # 打印收盘价与 20 日简单移动平均线之间的距离 print('Distance:', self.dist[0]) ``` 在上面的代码中,我们创建了 `MyStrategy` 类来演示如何使用 `bt.indicators.Distance` 指标来计算收盘价和 20 日简单移动平均线之间的距离。在 `__init__` 方法中,我们创建了 `self.close` 和 `self.sma20` 两个指标,分别表示收盘价和 20 日简单移动平均线。然后,我们使用 `bt.indicators.Distance` 指标创建了 `self.dist` 指标,表示收盘价与 20 日简单移动平均线之间的距离。 在 `next` 方法中,我们打印了最新的收盘价与 20 日简单移动平均线之间的距离,即 `self.dist[0]`。 如果 `self.dist[0]` 为正数,则表示收盘价高于 20 日简单移动平均线,如果为负数,则表示收盘价低于 20 日简单移动平均线。距离越大,背离程度越大。 ### 回答2: backtrader指标库中可以使用的指标来表示背离是Divergence指标。Divergence指标可以通过计算价格和指标之间的差异来检测背离现象。背离是指价格走势和指标走势不一致的情况,通常用来预测趋势的反转或调整。 要表示量价背离,可以先使用Volume指标来衡量成交量的变化情况。然后再与价格指标相比较,观察价格和成交量之间的关系。 以Divergence指标为例,如果价格创新高而Divergence指标在同一时间内没有创新高,就可以认为是一种量价背离的现象。这种背离意味着虽然价格上升,但买压不足,可能出现价格的反转或调整。 另一种情况是,价格创新低而Divergence指标在同一时间内没有创新低,也可以视为一种量价背离的现象。这种情况下,虽然价格下降,但卖压不足,可能预示着价格反转或调整的机会。 为了使用Divergence指标来表示量价背离,需要先选择合适的价格指标和Divergence指标进行计算,并关注它们之间的差异和趋势。通过观察价格和成交量之间的背离现象,可以帮助我们更好地理解市场的供需关系,提高交易决策的准确性。 ### 回答3: backtrader指标库中可以使用macd指标来表示背离。MACD是指数平滑移动平均线(Exponential Moving Average)与其自身的差值,通过观察MACD的高点和底点以及价格曲线的高点和底点之间的关系,可以判断出背离的存在。 量价背离可以通过使用OBV(On Balance Volume)指标来表示。OBV指标是以成交量作为基础,通过对成交量的累加与减去的方式判断买卖的力量,从而判断价格趋势的背离情况。当价格上升但OBV指标下降时,表示出现了量价背离,可能预示着价格即将反转。 在backtrader中,可以通过导入相应的指标库并使用相应的参数来创建并应用MACD和OBV指标,然后通过编写相应的条件判断语句来判断背离和量价背离的情况。具体的实现方法可以参考backtrader的文档和示例代码。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

神出鬼没,指的就是我!

必须花钱,数据超好

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值