量化交易入门(三十三)BIAS指标实现和回测

接下来我们还是用苹果股票2020年1月1日到2023年12月30日的历史数据进行回测,看看这个指标的效果如何,具体回测结果如下:

策略运行结果及解读

执行的结果:
Starting Portfolio Value: 100000.00 
Final Portfolio Value: 186723.04
Annualized Return: 16.93%
Sharpe Ratio: 0.94
Max Drawdown: 24.55%
Max Drawdown Period: 142

 根据回测结果,我们可以得出以下结论:

1、资金增长情况:

  • 初始资金为100,000美元
  • 最终资金为186,723.04美元
  • 资金增长了86,723.04美元,增幅为86.72%

2、年化收益率:

  • 年化收益率为16.93%
  • 这意味着平均每年的收益率为16.93%
  • 年化收益率是衡量策略盈利能力的重要指标,16.93%的年化收益率表现相对较好

3、夏普比率:

  • 夏普比率为0.94
  • 夏普比率衡量投资组合的风险调整后收益,即每承担一单位风险,可以获得多少超额收益
  • 夏普比率越高,表示策略的风险调整后收益越好
  • 0.94的夏普比率表明该策略的风险调整后收益处于较好水平

4、最大回撤:

  • 最大回撤为24.55%
  • 最大回撤衡量投资组合从高点到低点的最大跌幅
  • 最大回撤反映了策略可能面临的最大潜在损失
  • 24.55%的最大回撤说明该策略在某个时期经历了较大的回撤,需要注意风险管理

5、最大回撤期:

  • 最大回撤期为142个交易日
  • 最大回撤期表示从高点到低点的持续时间
  • 142个交易日的最大回撤期相对较长,说明该策略可能面临较长时间的回撤风险

综合来看,该策略在2020年1月1日到2023年12月30日的苹果股票历史数据上表现相对较好,取得了16.93%的年化收益率和0.94的夏普比率。但是,该策略也经历了24.55%的最大回撤,回撤持续时间较长,需要注意风险管理。

需要注意的是,这只是基于历史数据的回测结果,不能完全代表未来的实际表现。在实际应用中,还需要考虑市场环境变化、交易成本、滑点等因素的影响。此外,该策略仅使用了单一股票(苹果)的数据进行回测,可能存在集中风险。在实际投资中,建议采用多样化的投资组合来分散风险。

代码实现

下面是使用Backtrader框架对苹果股票这次回测的代码:

import backtrader as bt
import yfinance as yf

# 定义BIAS指标
class BIAS(bt.Indicator):
    lines = ('bias',)
    params = (('period', 20),)

    def __init__(self):
        self.lines.bias = (self.data.close - bt.indicators.SMA(period=self.params.period)) / bt.indicators.SMA(period=self.params.period) * 100

# 定义交易策略
class BIASStrategy(bt.Strategy):
    params = (('bias_upper', 10), ('bias_lower', -10),)

    def __init__(self):
        self.bias = BIAS(self.data)
        self.order = None


    def next(self):
        if not self.position:
            if self.bias[0] > self.params.bias_upper:
                pass  # 不进行操作
            elif self.bias[0] < self.params.bias_lower:
                commission_info = self.broker.getcommissioninfo(self.data)
                cash = self.broker.get_cash()
                size = int(cash / (self.data.close[0] * (1 + commission_info.p.commission)))
                self.order = self.buy(size=size)
                print(f'BUY: {size} shares')
        else:
            if self.order and self.order.isbuy() and self.bias[0] > self.params.bias_upper:
                self.close()
                self.order = None  # 重置self.order为None
                print(f'SELL: {self.position.size} shares')
            elif self.order and self.order.issell() and self.bias[0] < self.params.bias_lower:
                self.close()
                self.order = None  # 重置self.order为None
                print(f'BUY: {self.position.size} shares')

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

        if order.status in [order.Completed]:
            if order.isbuy():
                print(f'BUY executed at {self.data.num2date(order.executed.dt).date()}, Price: {order.executed.price:.2f}, Cost: {order.executed.value:.2f}, Comm: {order.executed.comm:.2f}')
            elif order.issell():
                cost = order.executed.value
                profit = order.executed.value - order.created.size * order.created.price
                profit_percent = (profit / cost) * 100
                print(f'SELL executed at {self.data.num2date(order.executed.dt).date()}, Price: {order.executed.price:.2f}, Cost: {cost:.2f}, Profit: {profit:.2f}, Profit %: {profit_percent:.2f}%')

        elif order.status in [order.Canceled, order.Margin, order.Rejected]:
            print('Order Canceled/Margin/Rejected')    



# 创建Cerebro引擎
cerebro = bt.Cerebro()

# 设置初始资金
cerebro.broker.setcash(100000.0)

# 下载苹果股票数据
data = yf.download('AAPL', '2020-01-01', '2023-12-30')
data = data.dropna()

# 将数据添加到Cerebro引擎中
data = bt.feeds.PandasData(dataname=data)
cerebro.adddata(data)

# 添加MACD策略
cerebro.addstrategy(BIASStrategy)

# 设置佣金为0.1%
cerebro.broker.setcommission(commission=0.001)

# 添加分析指标
cerebro.addanalyzer(bt.analyzers.Returns, _name='returns')
cerebro.addanalyzer(bt.analyzers.SharpeRatio, _name='sharpe')
cerebro.addanalyzer(bt.analyzers.DrawDown, _name='drawdown')

# 运行回测
print('Starting Portfolio Value: %.2f' % cerebro.broker.getvalue())
results = cerebro.run()
print('Final Portfolio Value: %.2f' % cerebro.broker.getvalue())

# 获取回测结果
strat = results[0]
returns = strat.analyzers.returns.get_analysis()
sharpe = strat.analyzers.sharpe.get_analysis()
drawdown = strat.analyzers.drawdown.get_analysis()

# 打印回测指标
print('Annualized Return: %.2f%%' % (returns['rnorm100']))
print('Sharpe Ratio: %.2f' % (sharpe['sharperatio']))
print('Max Drawdown: %.2f%%' % (drawdown['max']['drawdown']))
print('Max Drawdown Period: %s' % (drawdown['max']['len']))


# 绘制回测结果
cerebro.plot()

代码解析

将详细分析这段代码:

1、导入必要的库:

  • backtrader: Backtrader回测框架
  • yfinance: 用于下载股票数据

2、定义BIAS指标:

  • 创建一个名为BIAS的类,继承自bt.Indicator
  • 定义指标的参数period,默认值为20
  • __init__方法中计算BIAS指标的值,公式为:(当前收盘价 - 简单移动平均价) / 简单移动平均价 * 100

3、定义交易策略:

  • 创建一个名为BIASStrategy的类,继承自bt.Strategy
  • 定义策略的参数bias_upperbias_lower,默认值分别为10和-10
  • __init__方法中创建BIAS指标和订单变量
  • next方法中实现交易逻辑:
    • 如果当前没有持仓:
      • 如果BIAS指标大于上限值,不进行操作
      • 如果BIAS指标小于下限值,计算可用资金和可买入的股票数量,执行买入操作
    • 如果当前有持仓:
      • 如果订单为买入订单且BIAS指标大于上限值,执行卖出操作并重置订单为None
      • 如果订单为卖出订单且BIAS指标小于下限值,执行买入操作并重置订单为None
  • notify_order方法中处理订单状态变化:
    • 如果订单状态为SubmittedAccepted,不进行操作
    • 如果订单状态为Completed:
      • 如果是买入订单,打印买入信息(执行日期、价格、成本、手续费)
      • 如果是卖出订单,打印卖出信息(执行日期、价格、成本、利润、利润百分比)
    • 如果订单状态为CanceledMarginRejected,打印相应信息

4、创建Cerebro引擎,设置初始资金为100,000美元

5、下载苹果股票数据,时间范围为2020年1月1日至2023年12月30日,并删除缺失值

6、将股票数据添加到Cerebro引擎中

7、添加BIAS交易策略到Cerebro引擎中

8、设置佣金为0.1%

9、添加分析指标:

  • 收益率(bt.analyzers.Returns)
  • 夏普比率(bt.analyzers.SharpeRatio)
  • 最大回撤(bt.analyzers.DrawDown)

10、运行回测,打印初始和最终资金情况

11、获取回测结果,包括年化收益率、夏普比率、最大回撤等指标

12、打印回测指标

13、绘制回测结果图表

这个指标回策的结果也是相当的不错的,但是记住这个只是一个简单的例子,你们可以使用其他的股票进行回测,比如特斯拉,谷歌,微软等等,欢迎大家在留言区进行分享你们的回测结果。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/503784.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

pytest--python的一种测试框架--request请求加入headers

一、request headers中的cookie和session机制的作用与区别 Cookie 和 Session 是两种在客户端和服务器之间保持状态的技术。HTTP 协议本身是无状态的&#xff0c;这意味着服务器无法从上一次的请求中保留任何信息到下一次请求。Cookie 和 Session 机制就是为了解决这个问题。 …

JeeSite Vue3:前端如何实现权限管理

随着技术的飞速发展&#xff0c;前端开发技术日新月异。在这个背景下&#xff0c;JeeSite Vue3 作为一个基于 Vue3、Vite、Ant-Design-Vue、TypeScript 和 Vue Vben Admin 的前端框架&#xff0c;引起了广泛关注。它凭借其先进的技术栈和丰富的功能模块&#xff0c;为初学者和团…

对象数组的添加,删除和遍历.Java

创建一个Student的类&#xff0c;属性包含学号&#xff0c;姓名&#xff0c;年龄 &#xff0c;在此基础上进行对象的添加&#xff0c;删除&#xff0c;遍历 null调用方法必定报错&#xff0c;所以要判断数组里的元素&#xff08;本题数组里的每个元素都是一个对象&#xff09;…

P2802 回家

P2802 回家 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 虽然是普及-难度的题&#xff0c;但是感觉细节有很多。 细节&#xff1a; bfs第一次到 ( i , j ) (i, j) (i,j)&#xff0c;但是距离不一定是最小的 鼠标是一次性物品 血量到达 ( x x , y y ) (xx, yy) (xx,yy)为…

罗永浩和阿里云服务器ECS经济型e实例性能如何?

罗永浩直播卖阿里云&#xff0c;带货爆款云服务器ECS经济型e实例是什么&#xff1f;阿里云服务器ECS经济型e实例的使命是什么&#xff1f;一般来讲&#xff0c;学生、开发者和小微企业主要使用云服务器主要开展网站建设、开发测试和业务灾备、搭建小程序后端服务、Web应用、云上…

Vite为什么比Webpack快得多?

Vite为什么比Webpack快得多&#xff1f; 在前端开发中&#xff0c;构建工具扮演着至关重要的角色&#xff0c;而Vite和Webpack无疑是两个备受关注的工具。然而&#xff0c;众多开发者纷纷赞誉Vite的速度之快&#xff0c;本文将深入探讨Vite相较于Webpack为何更快的原因&#xf…

六大排序总结

前面分别分享了六大排序的详细内容&#xff0c;本博客是数据结构中六大排序的总结&#xff0c;下期分享C的学习干货&#xff0c;我们一起进步。 排序算法复杂度及稳定性分析 稳定性&#xff1a; 这个博客如果对你有帮助&#xff0c;给博主一个免费的点赞就是最大的帮助❤ 欢迎…

【QT+QGIS跨平台编译】045:【netcdf4+Qt跨平台编译】(一套代码、一套框架,跨平台编译)

点击查看专栏目录 文章目录 一、NetCDF4介绍二、文件下载三、文件分析四、pro文件五、编译实践一、NetCDF4介绍 NetCDF4 是 NetCDF(Network Common Data Form)的更新版本,相比于 NetCDF3,NetCDF4 提供了更多的高级功能和性能优化。以下是 NetCDF4 的一些特点和介绍: HDF5 …

阿里云服务器4核8G配置最新活动价格

阿里云服务器4核8g配置云服务器u1价格是955.58元一年&#xff0c;4核8G配置还可以选择ECS计算型c7实例、计算型c8i实例、计算平衡增强型c6e、ECS经济型e实例、AMD计算型c8a等机型等ECS实例规格&#xff0c;规格不同性能不同&#xff0c;价格也不同&#xff0c;阿里云服务器网al…

第3章.引导ChatGPT精准角色扮演:高效输出专业内容

角色提示技术 角色提示技术&#xff08;role prompting technique&#xff09;&#xff0c;是通过模型扮演特定角色来产出文本的一种方法。用户为模型设定一个明确的角色&#xff0c;它就能更精准地生成符合特定上下文或听众需求的内容。 比如&#xff0c;想生成客户服务的回复…

文章解读与仿真程序复现思路——电网技术EI\CSCD\北大核心《强沙尘暴下新能源基地的弹性评估及其提升方法 》

本专栏栏目提供文章与程序复现思路&#xff0c;具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…

Qt for WebAssembly 环境搭建 - Windows新手入门

Qt for WebAssembly 环境搭建 - Windows新手入门 一、所需工具软件1、安装Python2、安装Git2.1 注册Github账号2.2 下载安装Git2.2.1配置Git&#xff1a;2.2.2 配置Git环境2.2.3解决gitgithub.com: Permission denied (publickey) 3 安装em编译器 二、Qt配置编译器三、参考链接…

跨越时空,启迪智慧:奇趣相机重塑儿童摄影与教育体验

【科技观察】近期&#xff0c;奇趣未来公司以其创新之作——“奇趣相机”微信小程序&#xff0c;强势进军儿童AI摄影市场。这款专为亚洲儿童量身定制的应用&#xff0c;凭借精准贴合亚洲儿童面部特征的AIGC大模型&#xff0c;以及丰富的摄影模板与场景设定&#xff0c;正在重新…

实时数仓建设实践——滴滴实时数据链路组件的选型

目录 前言 一、实时数据开发在公司内的主要业务场景 二、实时数据开发在公司内的通用方案 三、特定场景下的实时数据开发组件选型 3.1 实时指标监控场景 3.2 实时BI分析场景 3.3 实时数据在线服务场景 3.4 实时特征和标签系统 四、各组件资源使用原则 五、总结和展望…

机器学习——降维算法-奇异值分解(SVD)

机器学习——降维算法-奇异值分解&#xff08;SVD&#xff09; 在机器学习中&#xff0c;降维是一种常见的数据预处理技术&#xff0c;用于减少数据集中特征的数量&#xff0c;同时保留数据集的主要信息。奇异值分解&#xff08;Singular Value Decomposition&#xff0c;简称…

240330-大模型资源-使用教程-部署方式-部分笔记

A. 大模型资源 Models - Hugging FaceHF-Mirror - Huggingface 镜像站模型库首页 魔搭社区 B. 使用教程 HuggingFace HuggingFace 10分钟快速入门&#xff08;一&#xff09;&#xff0c;利用Transformers&#xff0c;Pipeline探索AI。_哔哩哔哩_bilibiliHuggingFace快速入…

代码学习第32天---动态规划

随想录日记part32 t i m e &#xff1a; time&#xff1a; time&#xff1a; 2024.03.30 主要内容&#xff1a;今天开始要学习动态规划的相关知识了&#xff0c;今天的内容主要涉及两个方面&#xff1a; 不同路径 &#xff1b; 不同路径 II。 62.不同路径 63. 不同路径 II 动态…

Linux内核之Binder驱动container_of进阶用法(三十四)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a;多媒…

LeetCode 双指针专题

11.盛最多水的容器 给定一个长度为 n 的整数数组 height 。有 n 条垂线&#xff0c;第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。 找出其中的两条线&#xff0c;使得它们与 x 轴共同构成的容器可以容纳最多的水。 返回容器可以储存的最大水量。 说明&#xff1a;你不…

java数据结构与算法刷题-----LeetCode1091. 二进制矩阵中的最短路径

java数据结构与算法刷题目录&#xff08;剑指Offer、LeetCode、ACM&#xff09;-----主目录-----持续更新(进不去说明我没写完)&#xff1a;https://blog.csdn.net/grd_java/article/details/123063846 文章目录 广度优先双分裂蛇 广度优先双分裂蛇 双分裂蛇&#xff1a;是求二…