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

news2024/11/24 16:49:59

接下来我们还是用苹果股票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、绘制回测结果图表

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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

git基本操作二(小白快速上手)

1、前言 接上篇我们接着来继续讲 2、.gitignore忽略文件 创建一个.gitignore文件&#xff0c;并将其置于项目的根目录下&#xff0c;Git将自动识别并根据该规则忽略相应的文件和目录。 # 忽略所有的 .log 文件 *.log# 但跟踪所有的 build.log 文件 !build.log# 忽略所有的 /lo…

Visual Studio 2022 中 Qt 开发环境的搭建

Visual Studio 2022 中 Qt 开发环境的搭建 Visual Studio 2022 中 Qt 开发环境的搭建 Visual Studio 2022 中 Qt 开发环境的搭建 点击扩展&#xff0c;并选择管理扩展。 搜索qt&#xff0c;下载并安装 Qt Visual Studio Tools。 安装完成后&#xff0c;点击扩展&#xff0c;会…

pytorch常用的模块函数汇总(1)

目录 torch&#xff1a;核心库&#xff0c;包含张量操作、数学函数等基本功能 torch.nn&#xff1a;神经网络模块&#xff0c;包括各种层、损失函数和优化器等 torch.optim&#xff1a;优化算法模块&#xff0c;提供了各种优化器&#xff0c;如随机梯度下降 (SGD)、Adam、RMS…

基于SpringBoot的“校园台球厅人员与设备管理系统”的设计与实现(源码+数据库+文档+PPT)

基于SpringBoot的“校园台球厅人员与设备管理系统”的设计与实现&#xff08;源码数据库文档PPT) 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;SpringBoot 工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 系统功能结构图 系统首页界面图…

信息工程大学第五届超越杯程序设计竞赛(同步赛)题解

比赛传送门 博客园传送门 c 模板框架 #pragma GCC optimize(3,"Ofast","inline") #include<bits/stdc.h> #define rep(i,a,b) for (int ia;i<b;i) #define per(i,a,b) for (int ia;i>b;--i) #define se second #define fi first #define e…

YOLOv8算法改进【NO.111】利用shift-wise conv对顶会提出EMO中的iRMB进行二次创新

前 言 YOLO算法改进系列出到这&#xff0c;很多朋友问改进如何选择是最佳的&#xff0c;下面我就根据个人多年的写作发文章以及指导发文章的经验来看&#xff0c;按照优先顺序进行排序讲解YOLO算法改进方法的顺序选择。具体有需求的同学可以私信我沟通&#xff1a; 首推…

Vue + .NetCore前后端分离,不一样的快速发开框架

摘要&#xff1a; 随着前端技术的快速发展&#xff0c;Vue.NetCore框架已成为前后端分离开发中的热门选择。本文将深入探讨Vue.NetCore前后端分离的快速开发框架&#xff0c;以及它如何助力开发人员提高效率、降低开发复杂度。文章将从基础功能、核心优势、适用范围、依赖环境等…

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配置编译器三、参考链接…

wireshark 使用

wireshark介绍 wireshak可以抓取经过主机网卡的所有数据包&#xff08;包括虚拟机使用的虚拟网卡的数据包&#xff09;。 环境安装 安装wireshark: https://blog.csdn.net/Eoning/article/details/132141665 安装网络助手工具&#xff1a;https://soft.3dmgame.com/down/213…

【Java SE】深入理解static关键字

&#x1f970;&#x1f970;&#x1f970;来都来了&#xff0c;不妨点个关注叭&#xff01; &#x1f449;博客主页&#xff1a;欢迎各位大佬!&#x1f448; 文章目录 1.static关键字1.1 static的概念1.2 static的作用1.3 static的用法1.3.1 static修饰成员变量1.3.2 static修饰…

记一次由gzip引起的nginx转发事故

故事背景 书接前几篇文章&#xff0c;仍然是交付甲方遇到的一个特殊诉求&#xff0c;从而引发了本期的事故。甲方的诉求是前端的请求过来&#xff0c;需要加密&#xff0c;但是要经过waf&#xff0c;必须要求是请求明文&#xff0c;那就要在waf和nginx之间做一个解密前置应用处…

【3】3道链表力扣题:删除链表中的节点、反转链表、判断一个链表是否有环

3道链表力扣题 一、删除链表中的节点&#x1f30f; 题目链接&#x1f4d5; 示例&#x1f340; 分析&#x1f4bb; 代码 二、反转链表&#x1f30f; 题目链接&#x1f4d5; 示例&#x1f340; 分析① 递归② 迭代 三、判断一个链表是否有环&#x1f30f; 题目链接&#x1f4d5; …

学习transformer模型-Input Embedding 嵌入层的简明介绍

今天介绍transformer模型的Input Embedding 嵌入层。 背景 嵌入层的目标是使模型能够更多地了解单词、标记或其他输入之间的关系。 从头开始嵌入Embeddings from Scratch 嵌入序列需要分词器tokenizer、词汇表和索引&#xff0c;以及词汇表中每个单词的三维嵌入。Embedding a s…

git clone 后如何 checkout 到 remote branch

what/why 通常情况使用git clone github_repository_address下载下来的仓库使用git branch查看当前所有分支时只能看到master分支&#xff0c;但是想要切换到其他分支进行工作怎么办❓ 其实使用git clone下载的repository没那么简单&#x1f625;&#xff0c;clone得到的是仓库…

一个 hipsolver 特征值示例

1&#xff0c;原理 通过雅可比旋转&#xff0c;对对称矩阵计算特征值和特征向量&#xff1b; 通过初等正交变换&#xff0c;每次把其中一个非主对角元素消成零&#xff0c;最终只剩主对角线非零元素为特征值&#xff0c;同时把初等变换累积下来&#xff0c;构成特征向量。 2&a…

使用1panel部署Ollama WebUI(dcoekr版)浅谈

文章目录 说明配置镜像加速Ollama WebUI容器部署Ollama WebUI使用问题解决&#xff1a;访问页面空白 说明 1Panel简化了docker的部署&#xff0c;提供了可视化的操作&#xff0c;但是我在尝试创建Ollama WebUI容器时&#xff0c;遇到了从github拉取镜像网速很慢的问题&#xf…

Java反射系列(3):从spring反射工具ReflectionUtils说起

传送门 在比较早的时候&#xff0c;就讨论过java反射的一些用法及概念&#xff1a; Java反射系列(1)&#xff1a;入门基础 以及反射的基石Class对象&#xff01; Java反射系列(2)&#xff1a;从Class获取父类方法说起 今天就从工作中实际的例子来看看反射的应用。 兼容…

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

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

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

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

遥感数字图像处理的学习笔记

相关链接&#xff1a; 遥感数字图像处理实验教程&#xff08;韦玉春&#xff09;--部分实验问题回答 目录 1.什么是图像&#xff0c;什么是数字图像&#xff1f; 2.什么是遥感数字图像&#xff1f;模拟图像(照片)与遥感数字图像有什么区别&#xff1f; 3.什么是遥感数字图像…