用R语言实现简单的交易回测

一、R代码实现。

用google在网上搜索到的回测语句都大同小异,下面给出一个示例,改编自:量化策略回测

# 以S&P500为例,其雅虎代码为^GSPC;我们要计算5日均线指标的函数就是SMA(),

# 先载入需要的扩展包
> library(quantmod)
> library(PerformanceAnalytics)

# 先获取S&P500的交易数据,然后根据其收盘价(由函数Cl()抽取)计算其5日均线值:
> getSymbols('^GSPC') #S&P500 OHLC data
> close <- Cl(GSPC)
> mv5 <- SMA(close, 5)

# 策略是当收盘价大于5日均线,代表可以入市,取1,否则代表清仓,取0。(-1是代表卖空,不适用。)
> sig <- ifelse(close < mv5, 1, 0)

# 使用Lag()将信号序列向“过去”推迟一天,代表将昨天的信号,应用到今天。
> sig <- Lag(sig) #将该序列向“过去”延迟一天

# 计算收益序列
# discrete代表用离散方式计算当天收益率,即(Close-Close(-1))/Close(-1)
# continous代表用连续方式计算当天收益率,即ln(Close/Close(-1))
> roc <- ROC(type='discrete',close) 
> ret <- roc * sig

# 画出策略收益图
> charts.PerformanceSummary(ret)

# 最上面的板块是积累收益,相当于对cumprod(1+ret)的绘图;
# 第二个是日收益,相当于对ret原始收益数据的绘图;
# 最下面的是下跌图(又称“水下图”),将下跌成分独立绘出,有助于我们分析亏损状况和研究弥补措施。
回测示意图

以上需要注意的地方是制定策略的语句

> sig <- ifelse(close < mv5, 1, 0)
# 当收盘价大于5日均线时,sig取1,否则取0。

在网上公布的代码当中,有的是取-1,这里-1代表卖空,不是清仓,得到的结果是不对的。

二、excel模拟。

当初得到这个结果的时候我很怀疑,用几条R语句就把量化策略及结果模拟出来了?看上去这么神奇,结果对不对呢?我在excel里对该策略进行模拟:

以上面的策略为基础,细化得到的交易方案是这样的:

  1. 当股票收盘价格高于五日均线,且无持仓,那么第二天满仓入市。
  2. 当股票收盘价格高于五日均线,且满仓,那么第二天继续持仓。
  3. 当股票收盘价格低于五日均线,且无持仓,那么第二天继续空仓。
  4. 当股票收盘价格低于五日均线,且满仓,那么第二天全部平仓。

为了简化计算,假设以前一天的收盘价作为成交价:即以前一天的收盘价满仓入市,或以前一天的收盘价全部平仓。当然这可以在进一步的研究中细化,但这里我们只是为了演示,就不过于复杂了。

excel表格里得到的结果如下:

CLOSE SMA sig ROC ROC*sig 股票 现金 总收益
1994-1-3 5.6 0 1 1
1994-1-4 5.55 0 1 1
1994-1-5 5.65 0 1 1
1994-1-6 6.1 0 1 1
1994-1-7 6.25 5.83 0 1 1
1994-1-10 6.45 6 1 0.032 0.032 1.032 0 1.032
1994-1-11 6.15 6.12 1 -0.047 -0.047 0.984 0 0.984
1994-1-12 6.15 6.22 1 0 0 0.984 0 0.984
1994-1-13 6.25 6.25 0 0.016 0 0 0.984 0.984
1994-1-14 6.1 6.22 0 -0.024 0 0 0.984 0.984
1994-1-17 6 6.13 0 -0.016 0 0 0.984 0.984
1994-1-18 6.2 6.14 0 0.033 0 0 0.984 0.984
1994-1-19 6 6.11 1 -0.032 -0.032 0.952 0 0.952
1994-1-20 6 6.06 0 0 0 0 0.952 0.952

由于是取5日平均,所以5日平均值SMA从第5个交易日开始出现。这几天的资产都是现金,总收益为1。

从第6个交易日开始,通过判断收盘价与5日均线之间的大小,得到交易信号sig,注意,这里的sig是滞后一日的,代表前一天的收盘价与5日均线之间的相对大小。

  1. 1994-1-10 close>SMA,sig=1,以前一天的收盘价6.25满仓入市,当天收盘价为6.45,日内收益率为0.032,股票价值1.032,现金为0,总资产价值为1.032。

  2. 1994-1-11 close>SMA,sig=1,鉴于前一天已入市,继续持仓,当天收盘价为6.15,日内收益率为-0.047,股票价值0.984,现金为0,总资产价值为0.984。

  3. 1994-1-12 close>SMA,sig=1,继续持仓,当天收盘价为6.15,日内收益率为0,股票价值0.984,现金为0,总资产价值为0.984。

  4. 1994-1-13 close<SMA,sig=0,以前一天的收盘价6.15全部清仓,股票价值为0,现金为前一天的股票价值0.984,总资产价值为0.984。

此时,经历了一次持仓以及清仓之后,投资者手里只拥有现金资产,总资产价值为0.984,亏损0.016。

通过对比excel计算的ret(ROC*sig)与R代码计算的ret,两者的结果完全一致。

通过用excel对以上回测策略的模拟,可以发现以上回测策略的隐含前提:

(1) 入市价格和清仓价格均为前一天的收盘价。

(2) 入市则全部满仓,离市则全部清仓。

三、神秘的charts.PerformanceSummary()函数。

在R代码中,一条charts.PerformanceSummary()语句,画出在该策略下,对应股票的总资产收益,日收益和下跌图。它就像一个神奇的黑箱,给我们以策略是否有效以最直观的图形表示。但光有图形还不够,这三条曲线分别对应哪些数据,能不能分别导出方便进一步研究呢?

  1. 总资产收益。

    对应的是excle表格当中的总收益,直观表示了应用该策略,投资者总资产在回测时间内的增减情况。虽然在excel模拟中用股票、现金、总资产三个条目来代表,但其实用一条R代码就可以实现了:

     cumprod(1+ret)
    

    也就是对(1+ret)的连乘。很容易理解:假设到了第n天,那么这一天的总收益Rn应该等于前一天的总收益R(n-1)乘以当天的收益率,不是ret,而应当是(1+ret)。

    R代码得到的结果与excel是一致的。

  2. 日收益。

    就是R代码当中的ret和excel表格中的ROC*sig。

  3. 下跌图(这部分参考了R包计算回撤)。

    这里只是将下跌成分独立绘出,对应的excel数据也是ROC*sig,但这里的下跌图经过了整理。对应的R函数有:

    a)chart.Drawdown:下跌图,也就是charts.PerformanceSummary()的第三张图。

     data(edhec)
     chart.Drawdown(edhec[,c(1,2)], main="Drawdown from Peak Equity Attained", legend.loc="bottomleft")
    

    b)findDrawdowns:返回回撤的起始时间,时间间隔,回撤数值,常与sortDrawdowns连用找最大回撤。

     data(edhec)     
     findDrawdowns(edhec[,"Funds of Funds", drop=FALSE])     
     sortDrawdowns(findDrawdowns(edhec[,"Funds of Funds", drop=FALSE]))
    

    c)maxDrawdown:返回收益时间序列的最大回撤。

     data(edhec)
     t(round(maxDrawdown(edhec[,"Funds of Funds"]),4))
    

    d)table.Drawdowns:返回最差回撤的统计量表格。

     data(edhec)
     table.Drawdowns(edhec[,1,drop=FALSE])
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容

  • 技术分析:股指是最好的参照基本分析:把握宏观经济投资组合:根据行情调整成功者的启示:具备与众不同的思维方式超越专家...
    你在学校阅读 4,538评论 0 13
  • 美豹金融的美股大学--https://www.usmeibao.com/knowledge.html 一、基础篇 ...
    格式化_001阅读 2,504评论 1 5
  • 你根本就不知道每个夜深人静的时候,我的心有多痛,有多么的思念你!可是,这一切只有我自己明白,没有谁会理解那...
    凯丽818阅读 180评论 0 1
  • 在iOS开发当中,我们有很多选择可以做半透明模糊效果,下面就是一些常见的方式或者说工具。1. Core Image...
    sky_kYU阅读 1,035评论 2 2
  • 当我们老了 吃过午饭,一起去楼下的河边晒太阳 你拿着鱼竿钓鱼 我拿着相机拍照 波光粼粼的小河 在你装鱼的小桶里,也...
    李公子Alice34阅读 134评论 0 0