这是我正在与 quantstrat 合作的多时间框架策略的示例。这是执行多时间框架策略的正确方法还是我做错了?在 quantstrat 演示或谷歌搜索中,我没有遇到任何其他执行多时间框架的示例。
为了保持策略部分简单(这不是某人会交易的策略)并保持对多时间框架方面的关注,我将演示一个使用分时数据和5 分钟 OHLC 数据的简单策略。策略逻辑是当分时数据上穿 5 分钟数据的 30 周期 SMA 时买入,当分时数据下穿同一 SMA 时平仓。
例如:如果策略是平的,时间是 13:02,之前观察到的 5 分钟数据的 30 周期 SMA 是 90.55(对于 12:55-结束 13:00 的周期),并且分时数据从低于 90.55 到高于它 (90.56) 是买入,当分时数据再次收盘低于它时,它退出头寸。
为了实现这一点,我需要将分时数据和 5 分钟、30 周期 SMA 放入同一个对象中,以便 quantstrat 进行处理。我得到了 5 分钟的 OHLC xts 并计算了它的 30 周期 SMA。然后我将它合并到刻度数据 xts 对象中,这将为我提供一个包含所有刻度数据的对象,然后每 5 分钟我将获得最后一个观察到的 5 分钟、30 周期 SMA 的一行。
如果在 13:00 有一个 30 周期的 SMA 值,这是针对 12:55-13:00 的 5 分钟。由于 SMA 的下一次更新是 5 分钟后,我需要填写行,直到观察到下一个值(在 13:05)等等。
这是head
刻度数据(我拥有的刻度数据不包括毫秒,但我使用以下方法使行独一无二make.index.unique(clemtick)
:
head(clemtick)
Price Volume
2013-01-15 09:00:00 93.90 1
2013-01-15 09:00:00 93.89 1
2013-01-15 09:00:00 93.89 1
2013-01-15 09:00:00 93.88 2
2013-01-15 09:00:00 93.89 1
2013-01-15 09:00:00 93.89 2
这是head
1 分钟的数据(每分钟代表前一分钟的数据,例如时间戳 09:01:00 == 09:00:00 - 09:01:00 的数据):
head(clemin)
Open High Low Close Volume
2013-01-15 09:01:00 93.90 94.04 93.87 93.97 1631
2013-01-15 09:02:00 93.97 93.98 93.90 93.91 522
2013-01-15 09:03:00 93.91 93.97 93.90 93.96 248
2013-01-15 09:04:00 93.95 93.98 93.93 93.95 138
2013-01-15 09:05:00 93.95 93.96 93.91 93.92 143
2013-01-15 09:06:00 93.93 93.97 93.91 93.91 729
将 1 分钟数据转换为 5 分钟数据:
cle5min <- to.minutes5(clemin)
clemin.Open clemin.High clemin.Low clemin.Close clemin.Volume
2013-01-15 09:04:00 93.90 94.04 93.87 93.95 2539
2013-01-15 09:09:00 93.95 93.97 93.81 93.89 2356
2013-01-15 09:14:00 93.90 94.05 93.86 93.89 4050
2013-01-15 09:19:00 93.90 94.03 93.84 94.00 2351
2013-01-15 09:24:00 93.99 94.21 93.97 94.18 3261
2013-01-15 09:29:00 94.18 94.26 94.18 94.19 1361
您会注意到第一个 OHLC 是 09:04:00,这是由于该to.minutes5
函数的工作方式,这在此线程中进行了讨论。本质上是第一个时间戳 09:04:00 == OHLC 4 分钟数据,从 09:00:00 到 09:04:00。09:09:00 时间戳是从 09:04:00 到 09:09:00 的下一个完整 5 分钟。理想情况下,我希望每个时间戳为 5、10、15 等,但我还没有弄清楚如何做到这一点。
将 5min 数据的 30 SMA 转换为分时数据
clemtick$sma30 <- SMA(cle5min$clemin.Close, 30)
这将使用 SMA 创建一个新列。SMA 需要 30 个周期来计算第一个值,并且SMA 只会出现在每 5 分钟的时间戳(11:29:00、11:34:00、11:39,...)。看起来像:
clemtick["2013-01-15 11:28:59::2013-01-15 11:29:00"]
Price Volume SMA30
2013-01-15 11:28:59 93.87 1 NA
2013-01-15 11:28:59 93.87 1 NA
2013-01-15 11:28:59 93.88 1 NA
2013-01-15 11:29:00 93.87 1 93.92633
2013-01-15 11:29:00 93.87 1 NA
2013-01-15 11:29:00 93.88 1 NA
2013-01-15 11:29:00 93.88 1 NA
现在我需要SMA30
用重复值填充列。11:29:00的值适用SMA30
于 11:24:00 - 11:29:00 的 OHLC。该值的下一次更新将在 11:34:00 之前进行,因此我需要填写行直到下一个值,因为这是策略在逐行处理时将引用的内容。
clemtick <- na.locf(clemtick)
现在,如果我再次查询该对象,
clemtick["2013-01-15 11:33:58::2013-01-15 11:34:01"]
Price Volume SMA30
2013-01-15 11:33:58 93.84 1 93.92633
2013-01-15 11:34:00 93.84 1 93.92267
2013-01-15 11:34:00 93.85 1 93.92267
2013-01-15 11:34:01 93.84 1 93.92267
现在我们有了最终的对象,这里正在运行策略:
require(quantstrat)
options("getSymbols.warning4.0"=FALSE)
rm(list=ls(.blotter), envir=.blotter)
Sys.setenv(TZ="UTC")
symbols <- "clemtick"
currency('USD')
stock(symbols, currency="USD", multiplier=1)
account.st <- 0
strategy.st <- portfolio.st <- account.st <- "multi"
rm.strat(portfolio.st)
rm.strat(strategy.st)
initDate <- "1980-01-01"
tradeSize <- 1000
initEq <- tradeSize*length(symbols)
initPortf(portfolio.st, symbols=symbols, initDate=initDate, currency='USD')
initAcct(account.st, portfolios=portfolio.st,
initDate=initDate, currency='USD', initEq=initEq)
initOrders(portfolio.st, initDate=initDate)
strategy(strategy.st, store=TRUE)
add.signal(strategy.st, name="sigCrossover",
arguments=list(columns=c("Price", "sma30"), relationship="gt"),
label="golong")
add.signal(strategy.st, name="sigCrossover",
arguments=list(columns=c("Price", "sma30"), relationship="lt"),
label="exitlong")
#enter rule
add.rule(strategy.st, name="ruleSignal",
arguments=list(sigcol="golong",
sigval=TRUE,
ordertype="market",
orderside="long",
replace=TRUE,
prefer="Price",
orderqty=1),
type="enter", path.dep=TRUE, label="long")
#exit rule
add.rule(strategy.st, name = "ruleSignal",
arguments=list(sigcol="exitlong",
sigval=TRUE,
ordertype="market",
orderside="long",
replace=TRUE,
prefer="Price",
orderqty=-1),
type="exit", path.dep=TRUE, label="exitlong")
#apply strategy
t1 <- Sys.time()
out2 <- applyStrategy(strategy=strategy.st, portfolios=portfolio.st, debug=TRUE)
t2 <- Sys.time()
print(t2-t1)
head(mktdata)
nrow(mktdata)
总而言之,这是执行多时间框架策略的最佳方法吗?