分享

使用Backtrader对股票历史数据进行指标回测

 禁忌石 2022-11-25 发布于浙江

本次使用tushare接口,股票为600018,时间是20200101——20211231。

使用的指标是:收盘价大于简单移动平均价。

佣金为0.001。

英文代码的英文原文:Quickstart Guide - Backtrader 

注意:代码最后plot画图,在spyder中不能正常显示图片,只能显示提示信息:

<IPython.core.display.Javascript object>
<IPython.core.display.HTML object>

在Jupyter Notebook中则可以正常显示图片。

期初资金: 100000.00
2020-02-26, Close, 11.20
2020-02-26, 买入单, 11.20
2020-02-27, 已买入, 价格: 11.20, 费用: 112.00, 佣金 0.11
2020-02-27, Close, 11.21
2020-02-28, Close, 10.85
2020-02-28, 卖出单, 10.85
2020-03-02, 已卖出, 价格: 10.95, 费用: 112.00, 佣金 0.11
2020-03-02, 交易利润, 毛利润 -2.50, 净利润 -2.72
2020-03-02, Close, 11.04
2020-03-03, Close, 11.06
......
2022-01-21, Close, 8.76
2022-01-24, Close, 8.69
2022-01-25, Close, 8.56
2022-01-25, 卖出单, 8.56
2022-01-26, 已卖出, 价格: 8.57, 费用: 86.30, 佣金 0.09
2022-01-26, 交易利润, 毛利润 -0.60, 净利润 -0.77
2022-01-26, Close, 8.57
2022-01-27, Close, 8.45
2022-01-28, Close, 8.41
期末资金: 99968.07

  1. from __future__ import (absolute_import, division, print_function,
  2. unicode_literals)
  3. from datetime import datetime # For datetime objects
  4. # Import the backtrader platform
  5. import backtrader as bt
  6. import pandas as pd
  7. import tushare as ts
  8. # 创建策略继承bt.Strategy
  9. class TestStrategy(bt.Strategy):
  10. params = (
  11. # 均线参数设置15天,15日均线
  12. ('maperiod', 15),
  13. )
  14. def log(self, txt, dt=None):
  15. # 记录策略的执行日志
  16. dt = dt or self.datas[0].datetime.date(0)
  17. print('%s, %s' % (dt.isoformat(), txt))
  18. def __init__(self):
  19. # 保存收盘价的引用
  20. self.dataclose = self.datas[0].close
  21. # 跟踪挂单
  22. self.order = None
  23. # 买入价格和手续费
  24. self.buyprice = None
  25. self.buycomm = None
  26. # 加入指标
  27. # Add a MovingAverageSimple indicator
  28. self.sma = bt.indicators.SimpleMovingAverage(
  29. self.datas[0], period=self.params.maperiod)
  30. # Indicators for the plotting show
  31. bt.indicators.ExponentialMovingAverage(self.datas[0], period=25)
  32. bt.indicators.WeightedMovingAverage(self.datas[0], period=25,
  33. subplot=True)
  34. bt.indicators.StochasticSlow(self.datas[0])
  35. bt.indicators.MACDHisto(self.datas[0])
  36. rsi = bt.indicators.RSI(self.datas[0])
  37. bt.indicators.SmoothedMovingAverage(rsi, period=10)
  38. bt.indicators.ATR(self.datas[0], plot=False)
  39. # 订单状态通知,买入卖出都是下单
  40. def notify_order(self, order):
  41. if order.status in [order.Submitted, order.Accepted]:
  42. # broker 提交/接受了,买/卖订单则什么都不做
  43. return
  44. # 检查一个订单是否完成
  45. # 注意: 当资金不足时,broker会拒绝订单
  46. if order.status in [order.Completed]:
  47. if order.isbuy():
  48. self.log(
  49. '已买入, 价格: %.2f, 费用: %.2f, 佣金 %.2f' %
  50. (order.executed.price,
  51. order.executed.value,
  52. order.executed.comm))
  53. self.buyprice = order.executed.price
  54. self.buycomm = order.executed.comm
  55. elif order.issell():
  56. self.log('已卖出, 价格: %.2f, 费用: %.2f, 佣金 %.2f' %
  57. (order.executed.price,
  58. order.executed.value,
  59. order.executed.comm))
  60. # 记录当前交易数量
  61. self.bar_executed = len(self)
  62. elif order.status in [order.Canceled, order.Margin, order.Rejected]:
  63. self.log('订单取消/保证金不足/拒绝')
  64. # 其他状态记录为:无挂起订单
  65. self.order = None
  66. # 交易状态通知,一买一卖算交易
  67. def notify_trade(self, trade):
  68. if not trade.isclosed:
  69. return
  70. self.log('交易利润, 毛利润 %.2f, 净利润 %.2f' %
  71. (trade.pnl, trade.pnlcomm))
  72. def next(self):
  73. # 记录收盘价
  74. self.log('Close, %.2f' % self.dataclose[0])
  75. # 如果有订单正在挂起,不操作
  76. if self.order:
  77. return
  78. # 如果没有持仓则买入
  79. if not self.position:
  80. # 今天的收盘价在均线价格之上
  81. if self.dataclose[0] > self.sma[0]:
  82. # 买入
  83. self.log('买入单, %.2f' % self.dataclose[0])
  84. # 跟踪订单避免重复
  85. self.order = self.buy()
  86. else:
  87. # 如果已经持仓,收盘价在均线价格之下
  88. if self.dataclose[0] < self.sma[0]:
  89. # 全部卖出
  90. self.log('卖出单, %.2f' % self.dataclose[0])
  91. # 跟踪订单避免重复
  92. self.order = self.sell()
  93. def get_data(code,start='2020-01-01',end='2022-01-31'):
  94. df=ts.get_k_data(code,autype='qfq',start=start,end=end)
  95. df.index=pd.to_datetime(df.date)
  96. df['openinterest']=0
  97. df=df[['open','high','low','close','volume','openinterest']]
  98. return df
  99. dataframe=get_data('600018')
  100. start=datetime(2020, 1, 1)
  101. end=datetime(2021, 12, 31)
  102. if __name__ == '__main__':
  103. # 初始化cerebro回测系统设置
  104. cerebro = bt.Cerebro()
  105. # 取得股票历史数据
  106. data = bt.feeds.PandasData(dataname=dataframe, fromdate=start, todate=end)
  107. # 为Cerebro引擎添加策略
  108. cerebro.addstrategy(TestStrategy)
  109. # 加载交易数据
  110. cerebro.adddata(data)
  111. # 设置投资金额
  112. cerebro.broker.setcash(100000.0)
  113. # 每笔交易使用固定交易量
  114. cerebro.addsizer(bt.sizers.FixedSize, stake=10)
  115. # 设置佣金为0.001,除以100去掉%号
  116. cerebro.broker.setcommission(commission=0.001)
  117. #获取回测开始时的总资金
  118. print('期初资金: %.2f' % cerebro.broker.getvalue())
  119. #运行回测系统
  120. cerebro.run()
  121. #获取回测结束后的总资金
  122. print('期末资金: %.2f' % cerebro.broker.getvalue())
  123. # Plot the result
  124. cerebro.plot()

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多