-
Notifications
You must be signed in to change notification settings - Fork 0
/
time-series-regression.qmd
622 lines (502 loc) · 22.1 KB
/
time-series-regression.qmd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
# 时间序列回归 {#sec-time-series-regression}
```{r}
#| echo: false
source("_common.R")
```
```{r}
#| message: false
library(cmdstanr)
library(zoo)
library(xts) # xts 依赖 zoo
library(fGarch)
library(INLA)
library(mgcv)
library(tensorflow)
library(ggplot2)
library(bayesplot)
```
## 随机波动率模型
随机波动率模型主要用于股票时间序列数据建模。本节以美团股价数据为例介绍随机波动率模型,并分别以 Stan 框架和 **fGarch** 包拟合模型。
```{r}
#| label: fig-meituan-stack
#| message: false
#| fig-cap: 美团股价走势
#| fig-showtext: true
#| fig-width: 5
#| fig-height: 4
# 美团上市至 2023-07-15
meituan <- readRDS(file = "data/meituan.rds")
library(zoo)
library(xts)
library(ggplot2)
autoplot(meituan[, "3690.HK.Adjusted"]) +
theme_classic() +
labs(x = "日期", y = "股价")
```
对数收益率的计算公式如下:
$$
\text{对数收益率} = \ln(\text{今日收盘价} / \text{昨日收盘价} ) = \ln (1 + \text{普通收益率})
$$
下图给出股价对数收益率变化和股价对数收益率的分布,可以看出在不同时间段,收益率波动幅度是不同的,美团股价对数收益率的分布可以看作正态分布。
```{r}
#| label: fig-meituan-log-return
#| fig-cap: 美团股价对数收益率的情况
#| fig-subcap:
#| - 对数收益率的变动
#| - 对数收益率的分布
#| fig-showtext: true
#| fig-width: 5
#| fig-height: 4
#| layout-ncol: 2
meituan_log_return <- diff(log(meituan[, "3690.HK.Adjusted"]))[-1]
autoplot(meituan_log_return) +
theme_classic() +
labs(x = "日期", y = "对数收益率")
ggplot(data = meituan_log_return, aes(x = `3690.HK.Adjusted`)) +
geom_histogram(color = "black", fill = "gray", bins = 30) +
theme_classic() +
labs(x = "对数收益率", y = "频数(天数)")
```
检查对数收益率序列的自相关图
```{r}
#| label: fig-log-return
#| fig-cap: 对数收益率的自相关图
#| fig-showtext: true
#| fig-width: 5
#| fig-height: 4
#| par: true
acf(meituan_log_return, main = "")
```
发现,滞后 2、3、6、26 阶都有出界,滞后 17 阶略微出界,其它的自相关都在零水平线的界限内。
```{r}
Box.test(meituan_log_return, lag = 12, type = "Ljung")
```
在 0.05 水平下拒绝了白噪声检验,说明对数收益率序列存在相关性。同理,也注意到对数收益率的绝对值和平方序列都不是独立的,存在相关性。
```{r}
# ARCH 效应的检验
Box.test((meituan_log_return - mean(meituan_log_return))^2,
lag = 12, type = "Ljung")
```
结果高度显著,说明有 ARCH 效应。
### Stan 框架
随机波动率模型如下
$$
\begin{aligned}
y_t &= \epsilon_t \exp(h_t / 2) \\
h_{t+1} &= \mu + \phi (h_t - \mu) + \delta_t \sigma \\
h_1 &\sim \textsf{normal}\left( \mu, \frac{\sigma}{\sqrt{1 - \phi^2}} \right) \\
\epsilon_t &\sim \textsf{normal}(0,1) \\
\delta_t &\sim \textsf{normal}(0,1)
\end{aligned}
$$
其中, $y_t$ 表示在时间 $t$ 时股价的回报(对数收益率),$\epsilon_t$ 表示股价回报在时间 $t$ 时的白噪声扰/波动,$\delta_t$ 表示波动率在时间$t$ 时的波动。$h_t$ 表示对数波动率,带有参数 $\mu$ (对数波动率的均值),$\phi$ (对数波动率的趋势)。代表波动率的序列 $\{h_t\}$ 假定是平稳 $(|\phi| < 1)$ 的随机过程,$h_1$ 来自平稳的分布(此处为正态分布),$\epsilon_t$ 和 $\delta_t$ 是服从不相关的标准正态分布。
Stan 代码如下
```{verbatim, file="code/stochastic_volatility_models.stan", lang="stan"}
```
编译和拟合模型
```{r}
#| message: false
library(cmdstanr)
# 编译模型
mod_volatility_normal <- cmdstan_model(
stan_file = "code/stochastic_volatility_models.stan",
compile = TRUE, cpp_options = list(stan_threads = TRUE)
)
# 准备数据
mdata = list(T = 1274, y = as.vector(meituan_log_return))
# 拟合模型
fit_volatility_normal <- mod_volatility_normal$sample(
data = mdata,
chains = 2,
parallel_chains = 2,
iter_warmup = 1000,
iter_sampling = 1000,
threads_per_chain = 2,
seed = 20232023,
show_messages = FALSE,
refresh = 0
)
# 输出结果
fit_volatility_normal$summary(c("mu", "phi", "sigma", "lp__"))
```
### fGarch 包
[《金融时间序列分析讲义》](https://www.math.pku.edu.cn/teachers/lidf/course/fts/ftsnotes/html/_ftsnotes/index.html)两个波动率建模方法
- 自回归条件异方差模型(Autoregressive Conditional Heteroskedasticity,简称 ARCH)。
- 广义自回归条件异方差模型 (Generalized Autoregressive Conditional Heteroskedasticity,简称 GARCH )
确定 ARCH 模型的阶,观察残差的平方的 ACF 和 PACF 。
```{r}
#| label: fig-log-return-resid
#| fig-cap: 对数收益率的残差平方
#| fig-subcap:
#| - 自相关图
#| - 偏自相关图
#| fig-showtext: true
#| fig-width: 5
#| fig-height: 4
#| layout-ncol: 2
#| par: true
acf((meituan_log_return - mean(meituan_log_return))^2, main = "")
pacf((meituan_log_return - mean(meituan_log_return))^2, main = "")
```
发现 ACF 在滞后 1、2、3 阶比较突出,PACF 在滞后 1、2、16、18、29 阶比较突出。所以下面先来考虑低阶的 ARCH(2) 模型,设 $r_t$ 为对数收益率。
$$
\begin{aligned}
r_t &= \mu + a_t, \quad a_t = \sigma_t \epsilon_t, \quad \epsilon_t \sim \mathcal{N}(0,1) \\
\sigma_t^2 &= \alpha_0 + \alpha_1 a_{t-1}^2
+ \alpha_2 a_{t-2}^2.
\end{aligned}
$$
拟合 ARCH 模型,比较模型估计结果,根据系数显著性的结果,采纳 ARCH(2) 模型。
```{r}
#| message: false
library(fGarch)
meituan_garch1 <- garchFit(
formula = ~ 1 + garch(2, 0),
data = meituan_log_return, trace = FALSE, cond.dist = "std"
)
summary(meituan_garch1)
```
函数 `garchFit()` 的参数 `cond.dist` 默认值为 `"norm"` 表示标准正态分布,`cond.dist = "std"` 表示标准 t 分布。模型均值的估计值接近 0 是符合预期的,且显著性没通过,对数收益率在 0 上下波动。将估计结果代入模型,得到
$$
\begin{aligned}
r_t &= -5.665 \times 10^{-5} + a_t, \quad a_t = \sigma_t \epsilon_t, \quad \epsilon_t \sim \mathcal{N}(0,1) \\
\sigma_t^2 &= 1.070 \times 10^{-3} + 0.1156 a_{t-1}^2 + 0.1438a_{t-2}^2.
\end{aligned}
$$
下面考虑 GARCH(1,1) 模型
$$
\begin{aligned}
r_t &= \mu + a_t, \quad a_t = \sigma_t \epsilon_t, \quad \epsilon_t \sim \mathcal{N}(0,1) \\
\sigma_t^2 &= \alpha_0 + \alpha_1 a_{t-1}^2
+ \beta_1 \sigma_{t-1}^2.
\end{aligned}
$$
```{r}
meituan_garch2 <- garchFit(
formula = ~ 1 + garch(1, 1),
data = meituan_log_return, trace = FALSE, cond.dist = "std"
)
summary(meituan_garch2)
```
波动率的贡献主要来自 $\sigma_{t-1}^2$ ,其系数 $\beta_1$ 为 0.918。通过对数似然的比较,可以发现 GARCH(1,1) 模型比 ARCH(2) 模型更好。
## 贝叶斯可加模型
大规模时间序列回归,观察值是比较多的,可达数十万、数百万,乃至更多。粗粒度时时间跨度往往很长,比如数十年的天粒度数据,细粒度时时间跨度可短可长,比如数年的半小时级数据,总之,需要包含多个季节的数据,各种季节性重复出现。通过时序图可以观察到明显的季节性,而且往往是多种周期不同的季节性混合在一起,有时还包含一定的趋势性。举例来说,比如 2018-2023 年美国旧金山犯罪[事件报告数据](https://data.sfgov.org/Public-Safety/Police-Department-Incident-Reports-2018-to-Present/wg3w-h783),事件数量的变化趋势,除了上述季节性因素,特殊事件疫情肯定会影响,数据规模约 200 M 。再比如 2018-2023 年美国境内和跨境旅游业中的[航班数据](https://www.transtats.bts.gov/),原始数据非常大,R 包 [nycflights13](https://github.com/tidyverse/nycflights13) 提供纽约机场的部分航班数据。
为简单起见,下面以 R 内置的数据集 AirPassengers 为例,介绍 Stan 框架和 INLA 框架建模的过程。数据集 AirPassengers 包含周期性(季节性)和趋势性。作为对比的基础,下面建立非线性回归模型,趋势项和周期项是可加的形式:
$$
y = at + b + c \sin(\frac{t}{12} \times 2\pi) + d \cos(\frac{t}{12} \times 2\pi) + \epsilon
$$
根据数据变化的周期规律,设置周期为 12,还可以在模型中添加周期为 3 或 4 的小周期。其中,$y$ 代表观察值, $a,b,c,d$ 为待定的参数,$\epsilon$ 代表服从标准正态分布的随机误差。
```{r}
#| label: fig-lm
#| fig-cap: 非线性回归
#| fig-showtext: true
#| fig-width: 5
#| fig-height: 4
#| par: true
air_passengers_df <- data.frame(y = as.vector(AirPassengers), t = 1:144)
fit_lm1 <- lm(y ~ t + sin(t / 12 * 2 * pi) + cos(t / 12 * 2 * pi), data = air_passengers_df)
fit_lm2 <- update(fit_lm1, . ~ . +
sin(t / 12 * 2 * 2 * pi) + cos(t / 12 * 2 * 2 * pi), data = air_passengers_df
)
fit_lm3 <- update(fit_lm2, . ~ . +
sin(t / 12 * 3 * 2 * pi) + cos(t / 12 * 3 * 2 * pi), data = air_passengers_df
)
plot(y ~ t, air_passengers_df, type = "l")
lines(x = air_passengers_df$t, y = fit_lm1$fitted.values, col = "red")
lines(x = air_passengers_df$t, y = fit_lm2$fitted.values, col = "green")
lines(x = air_passengers_df$t, y = fit_lm3$fitted.values, col = "orange")
```
模型 1 已经很好地捕捉到趋势和周期信息,当添加小周期后,略有改善,继续添加更多的小周期,不再有明显改善。实际上,小周期对应的回归系数也将不再显著。所以,这类模型的优化空间见顶了,需要进一步观察和利用残差的规律,使用更加复杂的模型。
### Stan 框架
非线性趋势、多季节性(多个周期混合)、特殊节假日、突发热点事件、残差成分(平稳),能同时应对这五种情况的建模方法是贝叶斯可加模型和神经网络模型,比如基于 Stan 实现的 prophet 包和 tensorflow 框架。
::: callout-tip
prophet 包是如何同时处理这些情况,是否可以在 cmdstanr 包中实现,是否可以在 mgcv 和 INLA 中实现?
:::
```{r}
library(cmdstanr)
```
### INLA 框架 {#sec-kaust-inla}
阿卜杜拉国王科技大学(King Abdullah University of Science and Technology 简称 KAUST)的 Håvard Rue 等开发了 INLA 框架 [@Rue2009]。《贝叶斯推断与 INLA 》的第3章混合效应模型中随机游走部分 [@Virgilio2020],一个随机过程(如随机游走、AR(p) 过程)作为随机效应。AirPassengers 的方差在变大,取对数尺度后,方差基本保持不变,一阶差分后基本保持平稳。
```{r}
#| label: fig-log-airpassengers
#| fig-cap: AirPassengers 的时序图
#| fig-subcap:
#| - 对数尺度
#| - 一阶差分
#| layout-ncol: 2
#| fig-width: 5
#| fig-height: 4
#| fig-showtext: true
library(ggfortify)
autoplot(log(AirPassengers)) +
theme_classic() +
labs(x = "年月", y = "对数值")
autoplot(diff(log(AirPassengers))) +
theme_classic() +
labs(x = "年月", y = "差分对数值")
```
因此,下面基于对数尺度建模。首先考虑 RW1 随机游走模型,而后考虑季节性。RW1 模型意味着取对数、一阶差分后序列平稳高斯过程,序列值服从高斯分布。下面设置似然函数的高斯先验 $\mathcal{N}(1,0.2)$ ,目的是防止过拟合。
```{r}
#| message: false
library(INLA)
inla.setOption(short.summary = TRUE)
air_passengers_df <- data.frame(
y = as.vector(AirPassengers),
year = as.factor(rep(1949:1960, each = 12)),
month = as.factor(rep(1:12, times = 12)),
ID = 1:length(AirPassengers)
)
mod_inla_rw1 <- inla(
formula = log(y) ~ year + f(ID, model = "rw1"),
family = "gaussian", data = air_passengers_df,
control.family = list(hyper = list(prec = list(param = c(1, 0.2)))),
control.predictor = list(compute = TRUE)
)
summary(mod_inla_rw1)
```
这里,将年份作为因子型变量,从输出结果可以看出,以1949年作为参照,回归系数的后验均值在逐年变大,这符合 AirPassengers 时序图呈现的趋势。
存在周期性的波动规律,考虑季节性
```{r}
mod_inla_sea <- inla(
formula = log(y) ~ year + f(ID, model = "seasonal", season.length = 12),
family = "gaussian", data = air_passengers_df,
control.family = list(hyper = list(prec = list(param = c(1, 0.2)))),
control.predictor = list(compute = TRUE)
)
summary(mod_inla_sea)
```
最后,将两个模型的拟合结果展示出来,见下图,黑线表示原对数值,红线表示拟合值,灰色区域表示在置信水平 95% 下的区间。区间更短说明季节性模型更好。
```{r}
#| label: fig-fitted-airpassengers
#| fig-cap: AirPassengers 的拟合图
#| fig-subcap:
#| - 随机游走模型
#| - 季节效应模型
#| layout-ncol: 2
#| fig-width: 5
#| fig-height: 4
#| fig-showtext: true
mod_inla_rw1_fitted <- data.frame(
ID = 1:length(AirPassengers),
y = as.vector(log(AirPassengers)),
mean = mod_inla_rw1$summary.fitted.values$mean,
`0.025quant` = mod_inla_rw1$summary.fitted.values$`0.025quant`,
`0.975quant` = mod_inla_rw1$summary.fitted.values$`0.975quant`,
check.names = FALSE
)
mod_inla_sea_fitted <- data.frame(
ID = 1:length(AirPassengers),
y = as.vector(log(AirPassengers)),
mean = mod_inla_sea$summary.fitted.values$mean,
`0.025quant` = mod_inla_sea$summary.fitted.values$`0.025quant`,
`0.975quant` = mod_inla_sea$summary.fitted.values$`0.975quant`,
check.names = FALSE
)
ggplot(data = mod_inla_rw1_fitted, aes(ID)) +
geom_ribbon(aes(ymin = `0.025quant`, ymax = `0.975quant`), fill = "gray") +
geom_line(aes(y = y)) +
geom_line(aes(y = mean), color = "red") +
theme_classic() +
labs(x = "序号", y = "对数值")
ggplot(data = mod_inla_sea_fitted, aes(ID)) +
geom_ribbon(aes(ymin = `0.025quant`, ymax = `0.975quant`), fill = "gray") +
geom_line(aes(y = y)) +
geom_line(aes(y = mean), color = "red") +
theme_classic() +
labs(x = "序号", y = "对数值")
```
## 一些非参数模型
### mgcv 包 {#sec-gnu-mgcv}
**mgcv** 包 [@Wood2017] 是 R 软件内置的推荐组件,由 Simon Wood 开发和维护,历经多年,成熟稳定。函数 `bam()` 相比于函数 `gam()` 的优势是可以处理大规模的时间序列数据。对于时间序列数据预测,数万和百万级观测值都可以 [@wood2015]。
```{r}
air_passengers_tbl <- data.frame(
y = as.vector(AirPassengers),
year = rep(1949:1960, each = 12),
month = rep(1:12, times = 12)
)
mod1 <- gam(y ~ s(year) + s(month, bs = "cr", k = 12),
data = air_passengers_tbl, family = gaussian
)
summary(mod1)
```
观察年和月的趋势变化,逐年增长趋势基本是线性的,略有波动,逐月变化趋势比较复杂,不过,可以明显看出在 7-9 月是高峰期,11 月和1-3月是低谷期。
```{r}
#| label: fig-mgcv-trend
#| fig-cap: 年和月的趋势变化
#| fig-showtext: true
#| fig-width: 7
#| fig-height: 4
#| par: true
layout(matrix(1:2, nrow = 1))
plot(mod1, shade = TRUE)
```
将拟合效果绘制出来,见下图,整体上,捕捉到了趋势和周期,不过,存在欠拟合,年周期内波动幅度随时间有变化趋势,趋势和周期存在交互作用。
```{r}
#| label: fig-mgcv-1
#| fig-cap: 趋势拟合效果
#| fig-showtext: true
#| fig-width: 5
#| fig-height: 4
#| par: true
air_passengers_ts <- ts(mod1$fitted.values, start = c(1949, 1), frequency = 12)
plot(AirPassengers)
lines(air_passengers_ts, col = "red")
```
整体上,乘客数逐年呈线性增长,每年不同月份呈现波动,淡季和旺季出行的流量有很大差异,近年来,这种差异的波动在扩大。为了刻画这种情况,考虑年度趋势和月度波动的交互作用。
```{r}
mod2 <- gam(y ~ s(year, month), data = air_passengers_tbl, family = gaussian)
summary(mod2)
```
可以看到,调整的 $R^2$ 明显增加,拟合效果更好,各年各月份的乘客数变化,见下图。
```{r}
#| label: fig-mgcv-interaction
#| fig-cap: 交互作用
#| fig-showtext: true
#| fig-width: 5
#| fig-height: 4
op <- par(mar = c(4, 4, 2, 0))
plot(mod2)
on.exit(par(op), add = TRUE)
```
上图是轮廓图,下面用透视图展示趋势拟合的效果。
```{r}
#| label: fig-mgcv-persp
#| fig-cap: 趋势拟合效果
#| fig-showtext: true
#| fig-width: 5
#| fig-height: 4
op <- par(mar = c(0, 1.5, 0, 0))
vis.gam(mod2, theta = -35, phi = 20, ticktype = "detailed", expand = .65, zlab = "")
on.exit(par(op), add = TRUE)
```
最后,在原始数据的基础上,添加拟合数据,得到如下拟合趋势图,与前面的拟合图比较,可以看出效果提升很明显。
```{r}
#| label: fig-mgcv-2
#| fig-cap: 趋势拟合效果
#| fig-showtext: true
#| fig-width: 5
#| fig-height: 4
#| par: true
air_passengers_ts <- ts(mod2$fitted.values, start = c(1949, 1), frequency = 12)
plot(AirPassengers)
lines(air_passengers_ts, col = "red")
```
### nnet 包 {#sec-gnu-nnet}
前面介绍的模型都具有非常强的可解释性,比如各个参数对模型的作用。对于复杂的时间序列数据,比较适合用复杂的模型来拟合,看重模型的泛化能力,而不那么关注模型的机理。
多层感知机是一种全连接层的前馈神经网络。**nnet** 包的函数 `nnet()` 实现了单隐藏层的简单前馈神经网络,可用于时间序列预测,也可用于分类数据的预测。作为对比的基础,下面先用 nnet 包训练和预测数据。
```{r}
# 准备数据
air_passengers <- as.matrix(embed(AirPassengers, 4))
colnames(air_passengers) <- c("y", "x3", "x2", "x1")
data_size <- nrow(air_passengers)
# 拆分数据集
train_size <- floor(data_size * 0.67)
train_data <- air_passengers[1:train_size, ]
test_data <- air_passengers[train_size:data_size, ]
# 随机数种子对结果的影响非常大 试试 set.seed(20232023)
set.seed(20222022)
# 单隐藏层 8 个神经元
mod_nnet <- nnet::nnet(
y ~ x1 + x2 + x3,
data = air_passengers, # 数据集
subset = 1:train_size, # 训练数据的指标向量
linout = TRUE, size = 4, rang = 0.1,
decay = 5e-4, maxit = 400, trace = FALSE
)
# 预测
train_pred <- predict(mod_nnet, newdata = air_passengers[1:train_size,], type = "raw")
# 训练集 RMSE
sqrt(mean((air_passengers[1:train_size, "y"] - train_pred )^2))
# 预测
test_pred <- predict(mod_nnet, newdata = air_passengers[-(1:train_size),], type = "raw")
# 测试集 RMSE
sqrt(mean((air_passengers[-(1:train_size), "y"] - test_pred)^2))
```
下面将原观测序列,训练集和测试集上的预测序列放在一张图上展示。图中,红色曲线表示训练集上的预测结果,绿色曲线为测试集上预测结果。
```{r}
#| label: fig-nnet
#| fig-cap: 单层感知机预测
#| fig-showtext: true
#| fig-width: 5
#| fig-height: 4
#| par: true
train_pred_ts <- ts(data = train_pred, start = c(1949, 3), frequency = 12)
test_pred_ts <- ts(data = test_pred, start = c(1957, 1), frequency = 12)
plot(AirPassengers)
lines(train_pred_ts, col = "red")
lines(test_pred_ts, col = "green")
```
由图可知,在测试集上,随着时间拉长,预测越来越不准。
### keras3 包 {#sec-google-keras3}
[**keras3**](https://github.com/rstudio/keras) 包通过 **reticulate** 包引入 [Keras 3](https://github.com/keras-team/keras) 框架,这个框架支持 [TensorFlow](https://github.com/tensorflow/tensorflow) 和 [PyTorch](https://github.com/pytorch/pytorch) 等多个后端,目前,keras3 包通过 [tensorflow 包](https://github.com/rstudio/tensorflow)仅支持 TensorFlow 后端。
下面使用 keras3 包构造多层感知机训练数据和预测。
```{r}
#| message: false
library(keras3)
set_random_seed(20222022)
# 模型结构
mod_mlp <- keras_model_sequential(shape = c(3)) |>
layer_dense(units = 12, activation = "relu") |>
layer_dense(units = 8, activation = "relu") |>
layer_dense(units = 1)
# 训练目标
compile(mod_mlp,
loss = "mse", # 损失函数
optimizer = "adam", # 优化器
metrics = "mae" # 监控度量
)
# 模型概览
summary(mod_mlp)
```
输入层为 3 个节点,中间两个隐藏层,第一层为 12 个节点,第二层为 8 个节点,全连接网络,最后输出为一层单节点,意味着单个输出。每一层都有节点和权重,参数总数为 161。
```{r}
# 拟合模型
fit(mod_mlp,
x = train_data[, c("x1", "x2", "x3")],
y = train_data[, "y"],
epochs = 200,
batch_size = 10, # 每次更新梯度所用的样本量
validation_split = 0.2, # 从训练数据中拆分一部分用作验证集
verbose = 0 # 不显示训练进度
)
# 将测试数据代入模型,计算损失函数和监控度量
evaluate(mod_mlp, test_data[, c("x1", "x2", "x3")], test_data[, "y"])
# 测试集上的预测
mlp_test_pred <- predict(mod_mlp, test_data[, c("x1", "x2", "x3")])
mlp_train_pred <- predict(mod_mlp, train_data[, c("x1", "x2", "x3")])
sqrt(mean((test_data[, "y"] - mlp_test_pred)^2)) # 计算均方根误差
```
从 RMSE 来看,MLP(多层感知机)预测效果比单层感知机稍好些,可网络复杂度是增加很多的。
```{r}
#| label: fig-tensorflow-mlp
#| fig-cap: 多层感知机预测
#| fig-showtext: true
#| fig-width: 5
#| fig-height: 4
#| par: true
mlp_train_pred_ts <- ts(data = mlp_train_pred, start = c(1949, 3), frequency = 12)
mlp_test_pred_ts <- ts(data = mlp_test_pred, start = c(1957, 1), frequency = 12)
plot(AirPassengers)
lines(mlp_train_pred_ts, col = "red")
lines(mlp_test_pred_ts, col = "green")
```
下面用 LSTM (长短期记忆)神经网络来训练时间序列数据,预测未来一周的趋势。输出不再是一天(单点输出),而是 7 天的预测值(多点输出)。参考 **tensorflow** 包的[官网](https://tensorflow.rstudio.com/guides/keras/working_with_rnns#introduction)中 RNN 递归神经网络的介绍。
## 习题
1. 基于 R 软件内置的数据集 `sunspots` 和 `sunspot.month` 比较 INLA 和 **mgcv** 框架的预测效果。
```{r}
#| label: fig-sunspots
#| fig-cap: 预测月粒度太阳黑子数量
#| fig-width: 7
#| fig-height: 4
#| fig-showtext: true
#| code-fold: true
#| echo: !expr knitr::is_html_output()
sunspots_tbl <- broom::tidy(sunspots)
sunspots_month_tbl <- broom::tidy(sunspot.month)
ggplot() +
geom_line(data = sunspots_month_tbl, aes(x = index, y = value), color = "red") +
geom_line(data = sunspots_tbl, aes(x = index, y = value)) +
theme_bw() +
labs(x = "年月", y = "数量")
```
图中黑线和红线分别表示 1749-1983 年、1984-2014 年每月太阳黑子数量。