量化策略——准备3 数据、Backtrader回测框架与quantstats评价指标

news2025/1/11 5:02:42

我们一般使用AKShare这个库来获取股票数据或策略中用得到的数据:
AKShare github主页:https://github.com/akfamily/akshare

使用Backtrader框架作为回测的框架:
Backtrader github主页:https://github.com/mementum/backtrader

使用quantstats库作为回测结果评价的库:
quantstats github主页:https://github.com/ranaroussi/quantstats

这一部分准备好之后,后续我们将关注点主要放在【策略】上,对于数据、评价指标这些如无特殊处理,将不再赘述。整个量化的框架构造起来不太容易,如果以前有所了解,可以用自己习惯的方式;如果觉得困难较大,也可以先跳过,等后面能力够了之后,再上手构建。

ps:大家要慎重的使用网上的量化平台,因为偷策略这种事太正常了,大家还是最好自己本地搭一个测试的平台~

文章目录

  • 1. 数据准备
  • 2. Backtrader回测框架准备
  • 3. 评价指标

下面的数据准备与Backtrader回测准备,只是博主提供的一个参考,在开始正式介绍量化策略的时候是不会涉及到每个数据的采集,手把手的代码,这些都是不会提及的,只会提供一个backtrader的策略类,作为对策略的编程实现。

1. 数据准备

比如股票数据:

  1. 首先创建一个data文件夹,然后在文件夹里创建一个stock_data的文件夹

  2. 创建一个code文件夹用来存放程序文件

  3. 然后新建一个python文件,使用如下代码:

    import time
    import akshare as ak
    from tqdm import tqdm
    from loguru import logger
    
    
    def extract_data():
        start_date = "20150101"
        end_date = "20221101"
        stock_list = ak.stock_zh_a_spot_em()  # 东方财富网-沪深京 A 股-实时行情数据
    
        for stock_code in tqdm(stock_list['代码']):
            time.sleep(1)
            stock_df = ak.stock_zh_a_hist(
                symbol=stock_code, period="daily", start_date=start_date, end_date=end_date, adjust="hfq")  # 后复权
            stock_df.to_pickle("../data/stock_data/{}.pkl".format(stock_code))
            logger.debug("ADD DATA {}", stock_code)
    
    
    if __name__ == '__main__':
        extract_data()
    

这里博主把股票数据保存到data/stock_data/文件夹下,以股票代码.pkl的格式保存:

在这里插入图片描述

2. Backtrader回测框架准备

Backtrader足够简单,同时也非常接近实盘(国外可以一键切换实盘,国内没有接口)

Backtrader的使用请参考:Backtrader量化&回测1——基本的交易策略与挂单买卖

从策略到最终影响金额,都会经历四个步骤:

  1. 策略信号
  2. 委托
  3. 订单
  4. 金额与标的的置换

因此盯紧订单的变化就可以了解策略对金额变动的影响,为了将更多精力用于策略本身的编写上,我们写一个策略模版,然后以后的策略都可以通过继承这个模版,把与策略无关的变量、操作都写在模版里:

from loguru import logger
import backtrader as bt


class TemplateStrategy(bt.Strategy):

    def __init__(self):
        # 记录用
        self.buy_bond_record = {}  # 记录购买的订单
        self.sell_bond_record = {} # 记录卖出的订单

    def next(self):
        """最核心的触发策略"""
        raise

    def notify_order(self, order):
        """通知订单状态,当订单状态变化时触发"""
        today_time_string = self.datetime.datetime().strftime('%Y-%m-%d')
        if order.status in [order.Submitted, order.Accepted]:  # 接受订单交易,正常情况
            return
        if order.status in [order.Completed]:
            if order.isbuy():
                self.buy_bond_record.setdefault(today_time_string, {})
                self.buy_bond_record[today_time_string].setdefault(order.data._name.replace(".", "_"), [])
                self.buy_bond_record[today_time_string][order.data._name.replace(".", "_")].append({
                    "order_ref": order.ref,
                    "bond_name": order.data._name,
                    "size": order.size,
                    "price": order.executed.price,
                    "value": order.executed.value,
                    "trade_date": self.datetime.datetime(0),
                })
                logger.debug('{} 订单{} 已购入 {} , 购入单价 {:.2f}, 数量 {}, 费用 {:.2f}, 手续费 {:.2f}'.
                             format(self.datetime.date(), order.ref, order.data._name, order.executed.price, order.size,
                                    order.executed.value, order.executed.comm))
            elif order.issell():
                self.sell_bond_record.setdefault(today_time_string, {})
                self.sell_bond_record[today_time_string].setdefault(order.data._name.replace(".", "_"), [])
                self.sell_bond_record[today_time_string][order.data._name.replace(".", "_")].append({
                    "order_ref": order.ref,
                    "bond_name": order.data._name,
                    "size": order.size,
                    "price": order.executed.price,
                    "value": - order.executed.price * order.size,
                    "sell_type": order.info.sell_type,
                    "trade_date": self.datetime.datetime(0),
                })
                logger.debug('{} 订单{} 已卖出 {}, 卖出金额 {:.2f}, 数量 {}, 费用 {:.2f}, 手续费 {:.2f}'.
                             format(self.datetime.date(), order.ref, order.data._name, order.executed.price, order.size,
                                    -order.executed.price * order.size, order.executed.comm))
        elif order.status in [order.Margin, order.Rejected]:
            logger.warning('{} 订单{} 现金不足、金额不足拒绝交易', self.datetime.date(), order.ref)
        elif order.status in [order.Canceled]:
            logger.debug("{} 订单{} 已取消", self.datetime.date(), order.ref)
        elif order.status in [order.Expired]:
            logger.warning('{} 订单{} 超过有效期已取消, 订单开价 {}, 当天最高价{}, 最低价{}', self.datetime.date(), order.ref, order.price,
                           order.data.high[0], order.data.low[0])

之后的策略均继承此TemplateStrategy策略类,并覆写def next(self)函数即可,这一部分会将所有的订单在日志中打印下来

在程序中,购买的订单可以使用如下代码:

self.buy(
	data=self.getdatabyname(stock_name), # 针对哪一个股票代码
	size=100, # 数量
	price=self.getdatabyname(stock_name).close[0], # 以当天的收盘价购买
	exectype=bt.Order.Limit, # 限价单
	valid=self.getdatabyname(stock_name).datetime.date(1),  # 有效期1天
	)

3. 评价指标

我们使用quantstats这个库来对回测结果进行评价,这个库里的计算方法简单粗暴,通过对已有的计算方法的封装,我们得到可以方便的进行评价的方法:

import pandas as pd
import quantstats as qs


def cal_daily_return(fund_values: pd.Series):
    """根据资金变动,计算日资产的变化率
    :param fund_values: 每日的总资产
    """
    fund_values = fund_values.sort_index()
    daily_re: pd.Series = (fund_values / fund_values.shift(1)) - 1
    daily_re.iloc[0] = 0
    return daily_re


def cal_rolling_feature(daily_return_series: pd.Series, rf=0.02, record_dict: dict = None):
    """计算各种指标
    :param daily_return_series: 日收益的变化率
    :param rf: 无风险收益,这里定为0.02
    :param record_dict: 指标的结果会追加到这个字典中
    """
    if record_dict is None:
        record_dict = {}
    daily_return_series.index = pd.to_datetime(daily_return_series.index.values)
    feature_df = pd.DataFrame(index=daily_return_series.index)
    feature_df['累积收益率'] = qs.stats.compsum(daily_return_series).values
    feature_df['回撤'] = qs.stats.to_drawdown_series(daily_return_series)
    record_dict.update({"累积收益率": feature_df['累积收益率'].iloc[-1]})
    feature_dict = {
        "复合年增长": qs.stats.cagr(daily_return_series, rf=rf),
        "夏普比率": qs.stats.sharpe(daily_return_series, rf=rf),
        "索蒂诺": qs.stats.sortino(daily_return_series, rf=rf),
        "omega": qs.stats.omega(pd.DataFrame(daily_return_series), rf=rf),
        "最大回撤": qs.stats.max_drawdown(daily_return_series),
        "最大回撤期(天)": int(qs.stats.drawdown_details(feature_df['回撤'])['days'].max()),
        "年波动率": qs.stats.volatility(daily_return_series),
    }
    record_dict.update(feature_dict)
    # 决定保留的小数
    for key, value in record_dict.items():
        if isinstance(value, float):
            record_dict[key] = value.round(3)
    return feature_df, record_dict

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

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

相关文章

【CTF】git源码泄露和代码审计

目录 源码获取 函数绕过 解法一: 解法二: 参考文章: 源码获取 这里做的web题目是buuctf中的:[GXYCTF2019]禁止套娃 打开页面,查看源码是没有可以利用的点的。开始进行目录扫描 这里使用的工具是dirsearch。直接…

前端js实现多次添加累加文件上传和选择删除(django+js)- 编辑回显文件并上传 (二)

前言 后端返回的是文件地址,并不是文件流或base64编码字符串,而修改数据的接口又只接受文件。 本篇文章主要是基于累加文件上传介绍的。 添加上传文件文章链接:https://blog.csdn.net/qq_43030934/article/details/128726549?spm1001.2014.…

jmeter 压测java代码

一、背景 直接压测、调用java工程中的方法。(没有http等的入口) 二、java项目改造 一个java项目,想要压测其中的几个方法。我们需要在该工程中,添加一个压测入口的类, 这个类必须继承或者实现jmeter提供的接口/类。…

C语言萌新如何使用scanf函数?

🐎作者的话 如果你搜索输入输出函数,那么你会看到输入输出流、Turbo标准库、标准输出端、stdout什么什么乱七八糟的,作为一个萌新,哪懂这些? 本文介绍萌新在前期的学习中,常用的输入输出函数及其功能~ 跳…

HTML5+CSS3小实例:炫彩的发光字特效

前言: 今天我们向大家精选了一款HTML5CSS3文字特效,文字特效有超酷的动画类型,不多说,一起来看看。 描述: 这款文字特效既有倒影的效果,又有随机的颜色,看起来非常的炫酷。全文基于 HTML5CSS3 完…

log4j.properties自定义日志配置

一、通用的写法log4j.properties# 设置root logger等级为ALL,且appender有A1和FILE log4j.rootLoggerALL, A1,A3#设置com.example.test logger log4j.logger.com.example.testDEBUG,A1,A3 # 取消继承父类 log4j.additivity.com.example.testfalse# 设置个控制台输出…

即时通讯开发之TCP 连接的建立与中止

TCP 是一个面向连接的协议,所以在连接双方发送数据之前,都需要首先建立一条连接。这和前面讲到的协议完全不同。前面讲的所有协议都只是发送数据 而已,大多数都不关心发送的数据是不是送到,UDP 尤其明显,从编程的角度来说,UDP 编程也要简单 的多----UDP 都不用考虑数据分片。 书…

Ubuntu下源码编译VirtualBox一 —— 源码下载

VirtualBox想必大家都不陌生,做Linux开发的尤其是嵌入式Linux开发的人应该基本都知道或玩过VMware和VirtualBox。但通常都是为了在Windows电脑上能够使用Linux环境、即在Windows环境下通过下载可执行文件安装的VirtualBox。本文介绍在Linux环境(Ubuntu 2…

系分 - 论文 - 总览知识点

个人总结,仅供参考,欢迎加好友一起讨论 文章目录系分 - 论文 - 总览往年论文一览论文考点考试时间考试过程论文技巧论文写作论文扣分与加分系分 - 论文 - 总览 往年论文一览 一般情况下,往下数5、6年的题目出题形式,具有参考意义…

痞子衡嵌入式:盘点国内Cortex-M内核MCU厂商高主频产品(2023)

大家好,我是痞子衡,是正经搞技术的痞子。今天痞子衡给大家介绍的是国内Cortex-M内核MCU厂商高主频产品。 在 2021 年初痞子衡写了篇 《盘点国内Cortex-M内核MCU厂商高性能产品》,搜罗了当时市面上主频不低于 96MHz 的 CM 核国产 MCU。如今过去…

LSTM MultiheadAttention 输入维度

最近遇到点问题,对于模块的输入矩阵的维度搞不清楚,这里在学习一下,记录下来,方便以后查阅。 LSTM & Attention 输入维度LSTM记忆单元门控机制LSTM结构LSTM的计算过程遗忘门输入门更新记忆单元输出门LSTM单元的pytorch实现Pyt…

Spring Security in Action 第七章 配置授权:限制访问

本专栏将从基础开始,循序渐进,以实战为线索,逐步深入SpringSecurity相关知识相关知识,打造完整的SpringSecurity学习步骤,提升工程化编码能力和思维能力,写出高质量代码。希望大家都能够从中有所收获&#…

[leetcode 72] 编辑距离

题目 题目:https://leetcode.cn/problems/edit-distance/description/ 类似题目:[leetcode 583] 两个字符串的删除操作 解法 动态规划 这题应该是字符串dp的终极形态了吧🤣,不看答案完全不会…看了答案发现原来还是dp… 以例题…

未来的竞争是认知和执行力的竞争,只有认知高,强执行才能赚钱

之前很火的一句话是:你永远赚不到认知范围之外的钱所以只有持续不断地提升认知才能持续成长,持续提升,持续赚钱。未来的竞争从另一方面来说也是认知的竞争。不同的认知对待同一事物、信息有不同的理解;不同的认知对待同一事物、信…

固高科技在创业板提交注册:业绩开始下滑,实控人均为“学院派”

近日,固高科技股份有限公司(下称“固高科技”)在深圳证券交易所创业板递交注册。据贝多财经了解,固高科技于2021年12月在创业板递交上市申请,2022年8月17日获得上市委会议通过。 本次冲刺创业板上市,固高科…

【一道面试题】说一下Synchronized?

说一下Synchronized? Synchronized锁是Java中为了解决线程安全问题的一种方式,是一种悲观锁Synchronized可以用来修饰方法或者以代码块,用来保证线程执行方法或代码块时的原子性Java中任何一个类的对象都可以用来作为锁对象,但是…

docker-15-镜像Ubuntu20.04中安装python3.9

1 拉取并运行镜像 从docker hub 拉取镜像,以ubuntu20.04为例: docker pull ubuntu:20.04 docker run -it ubuntu:20.04 /bin/bash发现命令行变为root1234abcd5678:,这样就是进入docker容器里了。以下是docker常用的命令: # 以…

8086到80386汇编数据传送指令的扩展

80386及以上汇编的数据传送指令如下; MOV 传送字或字节. MOVSX 先符号扩展,再传送. MOVZX 先零扩展,再传送. PUSH 把字压入堆栈. POP 把字弹出堆栈. PUSHA 把AX,CX,DX,BX,SP,BP,SI,DI依次压入堆栈. POPA 把DI,SI,BP,SP,BX,DX,CX,A…

人大金仓数据库KSQL常用命令

第三章KSQL常用命令 登陆前显示ksql的帮助命令 Ksql --help 列出所有的SQL命令清单 test# \h 列出某个SQL命令语法大纲 \h <sql命令> 如&#xff1a;\h delect 查看ksql元命令的帮助 ..... 查看数据库列表 显示当前连接的数据库和登录用户 \c 显示当前test数据库的…

数学和统计方法

平均数&#xff0c;加权平均数&#xff0c;中位数&#xff0c;众数 1、平均数&#xff1a;所有数加在一起求平均 2、中位数&#xff1a;对于有限的数集&#xff0c;可以通过把所有观察值高低排序后找出正中间的一个作为中位数。如果观察值有偶数个&#xff0c;通常取最中间的 …