量化交易实战【1】自己搭建一个的股票交易回测框架,并通过均线择时策略进行回测

前言

本文介绍了一个股票交易的基础回测框架,并且通过均线择时策略,详细的演示了策略搭建的过程,和大家共同学习交流。

  1. 文章示例中使用的‘000001.XSHE.csv’文件,已上传至csdn资源中,可直接下载
  2. 或者关注文末公众号:‘阿旭算法与机器学习’。然后输入:股票数据,即可获取4000多只股票从2015年1月至2022年10月的所有基础数据

1. 导入数据

import pandas as pd
import matplotlib.pyplot as plt
pd.set_option('display.max_columns', None) # 展示所有列
# 000001.XSHE.csv文件包含2015年1月至2022年10月的开盘价、收盘价、最高价、最低价、成交量、成交金额股票的基本数据
df = pd.read_csv('./000001.XSHE.csv')  
df.head()
dateopenclosehighlowvolumemoney
02015-01-059.9810.0010.179.74458099037.04.565388e+09
12015-01-069.909.8510.239.71346952496.03.453446e+09
22015-01-079.729.679.889.55272274401.02.634796e+09
32015-01-089.689.349.729.30225445502.02.128003e+09
42015-01-099.309.429.919.19401736419.03.835378e+09
# 求该股票每日涨跌幅
df['change_pct'] = df['close'].pct_change()
df = df[['date','open','close','high','low','change_pct']]
df['date'] = pd.to_datetime(df['date'])  # 将交易日期字符串变为日期类型
df.head()
dateopenclosehighlowchange_pct
02015-01-059.9810.0010.179.74NaN
12015-01-069.909.8510.239.71-0.015000
22015-01-079.729.679.889.55-0.018274
32015-01-089.689.349.729.30-0.034126
42015-01-099.309.429.919.190.008565

2. 交易框架搭建–以均线策略为例

当短期均线由下向上穿过长期均线的时候,第二天以开盘价全仓买入并在之后一直持有;

当短期均线由上向下穿过长期均线的时候,第二天以开盘价全仓卖出,之后空仓,直到下次买入

2.1 计算均线数值

此处以短期均线MA=5;长期均线MA=20为例进行说明,至于参数的优化,需要依据不同股票的实际情况修改

# 计算均线
ma_short = 5  #短期均线,ma代表:moving_average
ma_long = 20  #长期均线,ma代表:moving_average
df['ma_short'] = df['close'].rolling(ma_short).mean()
df['ma_long'] = df['close'].rolling(ma_long).mean()
df.head(10)
dateopenclosehighlowchange_pctma_shortma_long
02015-01-059.9810.0010.179.74NaNNaNNaN
12015-01-069.909.8510.239.71-0.015000NaNNaN
22015-01-079.729.679.889.55-0.018274NaNNaN
32015-01-089.689.349.729.30-0.034126NaNNaN
42015-01-099.309.429.919.190.0085659.656NaN
52015-01-129.299.229.409.05-0.0212319.500NaN
62015-01-139.159.179.309.12-0.0054239.364NaN
72015-01-149.239.259.499.180.0087249.280NaN
82015-01-159.279.589.589.190.0356769.328NaN
92015-01-169.629.609.759.480.0020889.364NaN
#补全上面均线缺失值:补全方式采用扩展窗口函数expanding,移动计算前面所有值之和的均值
df['ma_short'].fillna(value=df['close'].expanding().mean(),inplace=True)
df['ma_long'].fillna(value=df['close'].expanding().mean(),inplace=True)
df.head(10)
dateopenclosehighlowchange_pctma_shortma_long
02015-01-059.9810.0010.179.74NaN10.00010.000000
12015-01-069.909.8510.239.71-0.0150009.9259.925000
22015-01-079.729.679.889.55-0.0182749.8409.840000
32015-01-089.689.349.729.30-0.0341269.7159.715000
42015-01-099.309.429.919.190.0085659.6569.656000
52015-01-129.299.229.409.05-0.0212319.5009.583333
62015-01-139.159.179.309.12-0.0054239.3649.524286
72015-01-149.239.259.499.180.0087249.2809.490000
82015-01-159.279.589.589.190.0356769.3289.500000
92015-01-169.629.609.759.480.0020889.3649.510000

2.2 产生交易信号:即找出买入与卖出信号

# 找买入信号
# 当天的短期均线大于等于长期均线
condition1 = (df['ma_short'] >= df['ma_long'])
# 上一个交易日的短期均线小于长期均线
condition2 = (df['ma_short'].shift(1) < df['ma_long'].shift(1))
# 将买入信号当天的signal设为1
df.loc[condition1 & condition2, 'signal'] = 1

# 找卖出信号
# 当天的短期均线小于等于长期均线
condition1 = (df['ma_short'] <= df['ma_long'])
# 上一个交易日的短期均线大于长期均线
condition2 = (df['ma_short'].shift(1) > df['ma_long'].shift(1))
# 将买入信号当天的signal设为0
df.loc[condition1 & condition2,'signal'] = 0

# 浏览产生交易信号的日期
df[df['signal'].notnull()].head(10)
dateopenclosehighlowchange_pctma_shortma_longsignal
322015-02-258.778.628.778.59-0.0137308.6828.68101.0
382015-03-058.438.358.458.30-0.0141688.5628.60750.0
432015-03-128.879.129.348.790.0543358.7208.63301.0
802015-05-0611.9611.8812.3211.71-0.00917412.31412.34450.0
942015-05-2612.5712.5612.6212.310.00079712.25012.11901.0
992015-06-0212.0111.9212.0211.73-0.00831911.94011.98850.0
1022015-06-0512.6212.3212.7012.12-0.00404212.12012.03651.0
1102015-06-1711.9711.8912.0111.680.00592212.13612.24500.0
1762015-09-228.198.288.368.180.0122258.2028.18301.0
1772015-09-238.218.098.258.07-0.0229478.1728.19650.0

2.3 由交易信号计算每天股票仓位

我们用1表示满仓,0表示空仓。当出现买入信号之后,全仓买入,当出现卖出信号之后,全仓卖出。

因此,当交易信号signal为1时,第二天仓位position也会变为1,之后的仓位一直为1,直到出现卖出信号,仓位变为0。

注意:此处因为signal的计算运用了收盘价,是每天收盘之后产生的信号,所以仓位position在第二天才会发生改变。
例如2015-05-01产生买入信号,2015-05-02仓位才会变成1,满仓用1表示,空仓用0表示
将signal信号下移一格,表示第二天买入,用1表示满仓,0表示空仓

# 新建列Pos表示仓位,将交易信号下移一格,表示第二天买入,1表示满仓,0表示空仓
df['pos'] = df['signal'].shift()
# 向上填充,将买入之后的pos全部设置为1
df['pos'].fillna(method='ffill', inplace=True)
# 没有买入的pos全部设置为0
df['pos'].fillna(value=0,inplace=True)
# 预览仓位数据
df[(df['date']>='2015-02-24')&(df['date']<'2015-05-06')][['date','signal','pos']].head(20)
datesignalpos
322015-02-251.00.0
332015-02-26NaN1.0
342015-02-27NaN1.0
352015-03-02NaN1.0
362015-03-03NaN1.0
372015-03-04NaN1.0
382015-03-050.01.0
392015-03-06NaN0.0
402015-03-09NaN0.0
412015-03-10NaN0.0
422015-03-11NaN0.0
432015-03-121.00.0
442015-03-13NaN1.0
452015-03-16NaN1.0
462015-03-17NaN1.0
472015-03-18NaN1.0
482015-03-19NaN1.0
492015-03-20NaN1.0
502015-03-23NaN1.0
512015-03-24NaN1.0

2.4 排除当天开盘就涨跌停导致无法交易的情况,此时仓位不能改变

注意:开盘涨跌停,是基本无法买入或者卖出股票的。

# 找出开盘涨停的日期:即今天的开盘价相对于昨天的收盘价上涨了9.7%以上,此处不用10%是因为由于4舍5入,涨停不一定就是10%
cond_cannot_buy = df['open'] > df['close'].shift(1) * 1.097

# 将开盘涨停且当前position为1时的'pos'设为空值
df.loc[cond_cannot_buy & (df['pos'] == 1),'pos'] = None

# 找出开盘跌停的日期,即今天的开盘价相对于昨天的收盘价跌了9.7%(1-0.097=0.903)
cond_cannot_sell = df['open'] < df['close'].shift(1) *0.903

# 将开盘跌停且当前position为0时的'pos'设为空值
df.loc[cond_cannot_sell & (df['pos'] == 0),'pos'] = None

# position为空的日期表示不能买卖。position仓位只能和前一个交易日保持一致
df['pos'].fillna(method='ffill', inplace=True)

2.5 计算资金曲线并绘图

# 资金曲线:假设起始资金为100万元的每天资金变化情况
# 首先计算资金曲线每天的涨跌幅,‘equity_change’表示资金每天的涨跌幅
# 当天空仓时,pos为0,资产涨幅为0
# 当天满仓时,pos为1,资产涨幅为股票本身的涨跌幅
df['equity_change'] = df['change_pct'] * df['pos']
# 根据每天的涨跌幅计算资金曲线
df['equity_curve'] = 1000000 * (df['equity_change'] + 1).cumprod()
df = df[['date' , 'change_pct','pos','equity_change','equity_curve']]
df.reset_index(inplace=True, drop=True)  # 重置索引,让他从0开始
df.head(10)
datechange_pctposequity_changeequity_curve
02015-01-05NaN0.0NaNNaN
12015-01-06-0.0150000.0-0.01000000.0
22015-01-07-0.0182740.0-0.01000000.0
32015-01-08-0.0341260.0-0.01000000.0
42015-01-090.0085650.00.01000000.0
52015-01-12-0.0212310.0-0.01000000.0
62015-01-13-0.0054230.0-0.01000000.0
72015-01-140.0087240.00.01000000.0
82015-01-150.0356760.00.01000000.0
92015-01-160.0020880.00.01000000.0
# 绘制资金曲线
plt.plot(df['equity_curve'])
plt.show()

请添加图片描述

通过资金曲线我们可以看到,该只股票从2015年1月开始至今,通过该策略最高是1.8倍收益,后面又跌回去了…

当然这个策略只是在此处搭建框架做演示用的,实际策略参数等都需要根据具体情况进行调整。

3. 总结

本文详细的介绍了通过股票数据及均线策略构建股票回测框架的过程:

  1. 导入数据,计算涨跌幅
  2. 通过均线策略产生交易信号
  3. 通过交易信号计算每天仓位
  4. 排除涨跌停无法进行交易的情况
  5. 通过仓位计算每天资金曲线,回测

对于其他择时交易策略,我们只需更改上述框架中的交易信号产生部分,就可以实现回测啦。

后续待改进的地方,本文在计算资金曲线的过程中,没有考虑交易手续费,买入时的价格与当天涨幅不匹配(当天开盘价买入,资金涨幅与当天股票涨幅不是同等的)等问题,下期会写一个实际操作过程中的资金曲线,将这些问题包含进去。如果有不对的地方,欢迎提出来共同学习交流,谢谢!

如果内容对你有帮助,记得点赞、关注哦!也欢迎小伙伴们和我共同学习交流。

更多干货内容持续更新中…

-------欢迎关注下方我的公众号,共同学习交流,获取更多学习资源。------
请添加图片描述

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

阿_旭

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值