诺贝尔奖获得者威廉夏普

简介

策略绩效评价是量化交易很重要的一环,投资不仅需要了解策略的收益率, 也需要了解策略的风险。backtrader提供多种analyzer分析器,可以输出多种绩效指标,用来分析策略回测的效果。

使用方法

cerebro.addanalyzer(bt.analyzers.AnnualReturn, _name='AnnualReturn')
cerebro.addanalyzer(bt.analyzers.SharpeRatio, riskfreerate=0.003, annualize=True, _name='SharpeRatio')
cerebro.addanalyzer(bt.analyzers.DrawDown, _name='DrawDown')strats = cerebro.run()# 第一个策略
strat = strats[0]print("--------------- AnnualReturn -----------------")
print(strat.analyzers.AnnualReturn.get_analysis())
print("--------------- SharpeRatio -----------------")
print(strat.analyzers.SharpeRatio.get_analysis())
print("--------------- DrawDown -----------------")
print(strat.analyzers.DrawDown.get_analysis())

AnnualReturn代表年化收益率分析器 SharpeRatio代表夏普比率分析器, DrawDown代表回测分析器。各个分析器的结果通常以 OrderedDict 字典的形式返回,如下所示,可以通过 keys 取需要的 values, 如strat.analyzers.SharpeRatio.get_analysis()['sharperatio'] 。

--------------- AnnualReturn -----------------
OrderedDict([(2010, 0.0055023006637382466), (2011, 0.0015355496623354892), (2012, 0.004218277850025043), (2013, 0.09220659214070048), (2014, 0.05213380413050861), (2015, 0.012115724348371826), (2016, -0.017901621371985033), (2017, 0.06763449420829182), (2018, 0.01391412496311406), (2019, 0.05472002239157425), (2020, 1.8864865026259587), (2021, 0.36892175167157526), (2022, -0.2711252801992251)])
--------------- SharpeRatio -----------------
OrderedDict([('sharperatio', 0.3360518820606975)])
--------------- DrawDown -----------------
AutoOrderedDict([('len', 145), ('drawdown', 35.66222565509548), ('moneydown', 21054.40185546875), ('max', AutoOrderedDict([('len', 648), ('drawdown', 40.770090001923634), ('moneydown', 24070.00244140625)]))])

可以看出来, Analyzer分析器输出的结果都是一些数值,并没有绘图。以SharpeRatio 为例,夏普比率计算后是一个数值0.3360518820606975。实际上,Analyzers 不包含lines对象,这也意味着analyzer对象对内存非常友好,不会占用很多的内存。

  • 进一步说明:一个analyzer分析器分析的是一个策略的绩效表现而不是整个系统的表现。 通过addanalyzer函数添加的分析器类,在cerebro.run 运行时,每个分析器实例会被链接到各个策略, 如果回测包含3个策略,那么同一个分析器将有3个实例被创建,每个实例会被链接到不同的策略

  • 一些Analyzer 分析器对象会使用其他analyzers分析器来完成任务,比如SharpeRatio 使用了TimeReturn 分析器的输出来进一步计算,但这些依赖用户并不可见

backtrader自带的Analyzers分析器

  • AnnualReturn:年化收益率

  • Calmar: 卡玛比率

  • DrawDown:回撤

  • TimeDrawDown: 指定时间粒度下的回撤

  • GrossLeverage: 总杠杆

  • PositionsValue: 持仓价值

  • PyFolio: 生成数据兼容pyfolio

  • LogReturnsRolling: 滚动log收益率

  • PeriodStats: 给定时段下的基本统计信息

  • Returns: 用对数法计算总、平均、复合、年化收益率

  • SharpeRatio: 夏普比率

  • SharpeRatio_A: 年化夏普比率

  • SQN:系统质量数System Quality Number

  • TimeReturn:不同时段的收益率

  • TradeAnalyzer:交易统计信息,如获胜、失败次数等

  • Transactions: 每笔交易的标的、价格、数量等信息

  • VWR: variability weighted return,波动率加权的收益率

常用的Analyzer分析器有:AnnualReturn, DrawDown, PyFolio, SharpeRatio, TimeReturn。

新建Analyzer分析器

如果backtrader自带的分析器无法满足需求,可以自建Analyzer分析器。以SharpeRatio 为例,参考代码如下。夏普比例是最常用的绩效评价指标之一,其反映的是单位波动贡献的超额收益率,越大越好。夏普比例等于策略相对无风险收益率的超额收益率除以收益率标准差。

import backtrader as btfrom backtrader.utils.py3 import map
from backtrader.mathsupport import average, standarddevclass SharpeRatio(bt.Analyzer):params = (('timeframe', bt.TimeFrame.Years), ('riskfreerate', 0.01),)def __init__(self):super(SharpeRatio, self).__init__()self.anret = bt.analyzers.AnnualReturn()# analyzer与策略一样,都是从第0根bar开始运行def start(self):passdef prenext(self):passdef nextstart(self):passdef next(self):pass# 一般对策略整体的评价指标是在策略结束后开始计算的def stop(self):retfree = [self.p.riskfreerate] * len(self.anret.rets)retavg = average(list(map(operator.sub, self.anret.rets, retfree)))retdev = standarddev(self.anret.rets)self.ratio = retavg / retdev# Analyzer类特有方法,返回包含分析结果的类dict对象def get_analysis(self):return dict(sharperatio=self.ratio)# 支持与策略一样的信息打印函数def notify_order(self, order):passdef notify_trade(self, trade):passdef notify_cashvalue(self, cash, value):passdef notify_fund(self, cash, value, fundvalue, shares):pass

从上述代码可以看出,尽管Analyzer 对象没有Lines对象且并不需要在lines上迭代,但它们跟策略一样遵守相同的运行模式,当然也有get_analysis 这种特有的方法用来返回分析结果。

在backtrader中使用PyFolio

cerebro.addanalyzer(bt.analyzers.PyFolio)
strats = cerebro.run()
# retrieve the 1st strategy
strat0 = strats[0]
pyfolio = strats.analyzers.getbyname('pyfolio')
returns, positions, transactions, gross_lev = pyfoliozer.get_pf_items()import pyfolio as pf
pf.create_full_tear_sheet(returns)

pyfolio 在Jupyter Notebook外也可以运行,但在Jupyter里面运行支持的最好。

跟基线benchmark对比

import backtrader as btcerebro = bt.Cerebro()spypd = bt.feeds.PandasData(dataname=spydata, name="spy")
cerebro.adddata(spypd)cerebro.addanalyzer(bt.analyzers.TimeReturn, timeframe=bt.TimeFrame.Years, _name='TimeReturn')
cerebro.addanalyzer(bt.analyzers.TimeReturn, timeframe=bt.TimeFrame.Years, data=spypd, _name='BenchTimeReturn')strats = cerebro.run()time_return = strat.analyzers.getbyname('TimeReturn').get_analysis()
bench_time_return = strat.analyzers.getbyname('BenchTimeReturn').get_analysis()print("--------------- TimeReturn -----------------")
print(time_return)
print("--------------- BenchTimeReturn -----------------")
print(bench_time_return)

输出日志:

--------------- TimeReturn -----------------
OrderedDict([(datetime.date(2009, 12, 31), 0.0), (datetime.date(2010, 12, 31), 0.0055023006637382466), (datetime.date(2011, 12, 31), 0.0015355496623354892), (datetime.date(2012, 12, 31), 0.004218277850025043), (datetime.date(2013, 12, 31), 0.09220659214070048), (datetime.date(2014, 12, 31), 0.05213380413050861), (datetime.date(2015, 12, 31), 0.012115724348371826), (datetime.date(2016, 12, 31), -0.017901621371985033), (datetime.date(2017, 12, 31), 0.06763449420829182), (datetime.date(2018, 12, 31), 0.01391412496311406), (datetime.date(2019, 12, 31), 0.05472002239157425), (datetime.date(2020, 12, 31), 1.8864865026259587), (datetime.date(2021, 12, 31), 0.36892175167157526), (datetime.date(2022, 12, 31), -0.2711252801992251)])
--------------- SpyTimeReturn -----------------
OrderedDict([(datetime.date(2009, 12, 31), -0.011793865755532318), (datetime.date(2010, 12, 31), 0.12840988195524994), (datetime.date(2011, 12, 31), -0.001988071570576566), (datetime.date(2012, 12, 31), 0.134741065036728), (datetime.date(2013, 12, 31), 0.2968892471880906), (datetime.date(2014, 12, 31), 0.11289182180470925), (datetime.date(2015, 12, 31), -0.008124930541476227), (datetime.date(2016, 12, 31), 0.09643402233275422), (datetime.date(2017, 12, 31), 0.19384416771302204), (datetime.date(2018, 12, 31), -0.06347893319524989), (datetime.date(2019, 12, 31), 0.28785206349908), (datetime.date(2020, 12, 31), 0.16162313396749006), (datetime.date(2021, 12, 31), 0.2703540848726258), (datetime.date(2022, 12, 31), -0.13563244077211734)])

注意不是所有Analyzers分析器都支持Benchmark对比。

结论 & 交流

关注微信公众号:诸葛说talk,获取更多内容。同时还能获取邀请加入投资交流群、量化投资研讨群, 与众多投资爱好者、量化从业者、技术大牛一起交流、切磋,快速提升自己的投资水平。

写文章不易,觉得本文对你有帮助的话,帮忙点个赞吧。

参考

  • Analyzers - Backtrader

  • Analyzers Reference - Backtrader

  • Analyzers - PyFolio - Integration - Backtrader

  • backtrader/pyfoliotest.py at master · mementum/backtrader · GitHub

  • Observers - Benchmarking - Backtrader

量化框架backtrader之一文读懂Analyzer分析器相关推荐

  1. 量化框架backtrader之一文读懂observer观测器

    简介 Backtrader observer观测器主要用于观察策略运行过程中的各个状态指标,如资金.买卖点等,在调用cerebro.plot()后可以方便地可视化状态指标的变化情况,如下图展示的Bro ...

  2. 任务调度框架 Quartz 一文读懂

    1.Quartz 简介 Quartz是OpenSymphony开源组织在Job scheduling领域又一个开源项目,完全由Java开发,可以用来执行定时任务,类似于java.util.Timer. ...

  3. hdfs文档存储服务器,一文读懂HDFS分布式存储框架分析

    一文读懂HDFS分布式存储框架分析 HDFS是一套基于区块链技术的个人的数据存储系统,利用无处不在的私人PC存储空间及便捷的网络为个人提供数据加密存储服务,将闲置的存储空间利用起来,服务于正处于爆发期 ...

  4. hdfs读写流程_一文读懂HDFS分布式存储框架分析

    一文读懂HDFS分布式存储框架分析 HDFS是一套基于区块链技术的个人的数据存储系统,利用无处不在的私人PC存储空间及便捷的网络为个人提供数据加密存储服务,将闲置的存储空间利用起来,服务于正处于爆发期 ...

  5. 一文读懂程序化交易算法交易量化投资高频交易统计利

    转 一文读懂程序化交易.算法交易.量化投资.高频交易. 统计套利 在央行发布的<中国金融稳定报告(2016)>中,对于高频交易的解释为程序化交易的频率超过一定程度,就成为高频交易.而对程序 ...

  6. 一文读懂PQuant与QQuant量化易金工

    转 一文读懂P Quant与 Q Quant ,量化交易与金融工程 原标题:P Quant 和 Q Quant 到底哪个是未来?  来源:李老师与何老师的CFA学习课堂  作者:何璇 P-Quant ...

  7. ​一文读懂EfficientDet

    一文读懂EfficientDet. 今年年初Google Brain团队在 CVPR 2020 上发布了 EfficientDet目标检测模型, EfficientDet是一系列可扩展的高效的目标检测 ...

  8. 独家 | 一文读懂语音识别(附学习资源)

    原标题:独家 | 一文读懂语音识别(附学习资源) 一.前言 6月27日,美国权威科技杂志<MIT科技评论>公布2017全球最聪明50家公司榜单.科大讯飞名列中国第一.全球第六.全世界排在科 ...

  9. 【原理篇】一文读懂Mask RCNN

    Mask RCNN 何凯明大神的经典论文之一,是一个实例分割算法,正如文中所说,Mask RCNN是一个简单.灵活.通用的框架,该框架主要作用是实例分割,目标检测,以及人的关键点检测.Mask RCN ...

最新文章

  1. 人工智能 有信息搜索 (启发式)
  2. Android笔记三十三.BroadcastReceiver使用
  3. 极客时间VIP年卡,任意专栏免费看,谁要?
  4. python结巴_python结巴(jieba)分词
  5. cas server php下载,关于用CAS Server与Php、Jetty配置实现SSO#4
  6. BBIAF的完整形式是什么?
  7. 数据结构之串:KMP算法
  8. Visual Studio配置64/32位汇编程序开发环境
  9. php企业应用,PHP企业级应用缓存技术详解
  10. maven安装 maven上传jar包到库里面
  11. 高效分页存储过程代码
  12. ExtAspNet v2.0.6发布 - AJAX性能提升
  13. hdu1113 Word Amalgamation(详解--map和string的运用)
  14. imac 蓝牙机械键盘_最好的蓝牙机械键盘
  15. linux网卡配置没生效,linux上网络配置不生效的解决办法
  16. 《构建之法》读书笔记——第1章 概论
  17. 浩辰3D软件新手入门攻略:草图平面
  18. i++和++i的区别
  19. VMware中kali2022通过物理机代理上网(桥接模式)
  20. Java日期时间API

热门文章

  1. 第78句 2020年地球日:霍金留给世界的遗言比以往任何时候都更有意义
  2. 支付宝:账号不存在,或对方关闭了“通过手机号或邮箱找到我“隐私开关!怎么解决?
  3. java 面向对象1
  4. python数组拆分_Python将数组拆分为多个数组
  5. Nasm汇编GDB调试
  6. Apache Kylin | 麒麟出没,必有祥瑞
  7. SunTorque12.27分享|拧紧工艺中常见的装配失效及排查
  8. 【考公人的福利】Python爬取中公官网资料
  9. hive 知识点总结
  10. lio-sam中雅克比推导