【python量化】大幅提拔预测性能,将NSTransformer用于股价预测

[复制链接]
查看678 | 回复0 | 2023-8-23 11:43:02 | 显示全部楼层 |阅读模式

  
写在前面

  NSTransformer模子来自NIPS 2022的一篇paper《Non-stationary Transformers: Exploring the Stationarity in Time Series Forecasting》。NSTransformer的目的重要是为了办理其他方法出现过安稳化处理的题目。其通过提出序列安稳化以及去安稳化注意力机制可以使得模子面向提拔预测性能的角度进行安稳化处理,相比于Transformer的变体,NSTransformer在预测性能方面实现了大幅度的提拔。下面的这篇文章重要带各人了解一下NSTransformer的根本原理,并利用作者开源的NSTransformer代码,并将其用于股票代价预测当中。
  
1

  NSTransformer模子
  由于Transformer的全局范围的建模本领,使其在时间序列预测中表现出巨大的气力。然而,在联合分布随时间变革的非稳态真实世界数据上,它们的性能可能会退化得很可骇。从前的研究重要采用安稳化技能来削弱原始序列的非安稳特性,以进步预测本领。但是,被剥夺了内在非安稳性的安稳序列对于现实世界中的突发变乱预测的指导意义不大。这个题目,在本文中被称为过安稳化(over-stationarization),导致Transformer对差异序列产生无差异的时序关注,拦阻了深度模子的预测本领。为了办理序列的可预测性和模子本领之间的困境,作者提出了Non-stationary Transformer (NSTransformer)作为一个通用框架,其中有两个相互依靠的模块。序列安稳化(Series Stationarization)和去安稳化注意力(De-stationary Attention)。具体来说,序列安稳化模块统一了每个输入的统计特性,并将输出转换为规复的统计特性,以进步可预测性。为了办理过安稳化题目,去安稳化注意力被计划出来,通过近似于从原始序列中学到的可区分的注意,将内在的非安稳信息规复为时间依靠。作者提出的NSTransformer框架在很大程度上提拔了主流Transformer模子的变体的预测性能,相比于Transformer,MSE低落了49.43%,相比于Informer,低落了47.34%,相比于Reformer,低落了46.89%。
  模子框架
  NSTransformer遵循先前在时间序列预测中利用的Transformer架构,采用标准的编码器-解码器结构,其中编码器从过去的数据中提取信息,而解码器则通过对过去的历史信息进行过汇总来实行预测。典范的NSTransformer是通过对Transformer的输入和输出进行序列安稳化处理,并用提出的非安稳注意机制代替self-attention,这可以进步基础模子的非安稳序列的预测本领。总之,NSTransformer重要包括下面两个模块:序列安稳化模块以及去安稳注意力模块。
  

  
NSTransformer的根本框架

  序列安稳化
  序列安稳化重要模块两个模块,一个是归一化(Normalization)模块,另一个是反归一化(De-Normalization)模块。起首,归一化模块通过一个滑动窗口的形式对每一维时间序列数据进行归一化处理,如许窗口化的方式可以将每一个相邻窗口内的数据都具有相同的均值跟方差,以消除序列之间标准上的差异性,并增长输入数据在时序上的分布稳固性。归一化的过程是:
  

  在模子预测竣事之后,反归一化模块利用归一化时记录的均值跟方差信息,用来将模子的输出映射回原来的标准,以规复归一化时丧失的信息。反归一化的过程是:
  

  通过这两个阶段的变更,模子将吸取到安稳的输入,这些输入遵循稳固的分布,更轻易泛化。如许的计划还使模子对时间序列具有平移稳固性和标准稳固性,从而有利于真实序列的预测。
  

  
序列安稳化过程(图片来自:https://zhuanlan.zhihu.com/p/587665491)

  去安稳化注意力
  去安稳化注意力机制去安稳化注意力机制的过程如下图所示:
  

  
去安稳化注意力机制(图片来自:https://zhuanlan.zhihu.com/p/587665491)

  如前面所提到的,过安稳化题目是由内在的非安稳信息的消散引起的,这将使模子无法捕获到用于预测的变乱性时间依靠。因此,作者试图近似从原始的非安稳序列中去学习注意力。下面是Transformer计算注意力的原始公式:
  

  对于输入时间序列x,计算它的均值跟方差得到:
  

  除此之外,为了简化分析这里假设了用于嵌入的前馈层在时间维度是线性的,基于线性假设可以推导出  ,  与对安稳化后时间序列计算得到的  跟  的关系是:
  

  其中,  与  是  跟  在时间维度的均值。如许从原始时间序列中计算注意力机制的公式,可以被从安稳化后的时间序列中计算的  跟  替换为下面的公式,具体的推导可以参考原文的附录。
  

  如许得到的注意力公式不仅包含了原始时间序列的信息,也包含了颠末安稳化后的序列信息。之后为了规复对非安稳序列的原始注意力,这里作者将消散的非安稳信息重新引入到非安稳序列的计算中。为了得到所必要的计算值,作者引入了去安稳因子   跟  ,它们都是通过两个多层感知器从非安稳时间序列中计算得到的。进而,去安稳化注意力的计算方式可以得到如下形式:
  

  如许,它既能利用安稳序列的可预测性,又能保持原始序列固有的时间依靠性。
  
2

  情况配置
  本地情况:
  1. Python 3.7
  2. IDE:Pycharm
复制代码
库版本:
  1. numpy 1.18.1
  2. pandas 1.0.3
  3. sklearn 0.22.2
  4. matplotlib 3.2.1
  5. torch 1.10.1
复制代码
NSTransformer源码Github链接:
  https://github.com/thuml/Nonstationary_Transformers
  
3

  代码实现
  NSTransformer的官方代码实现借鉴了Informer的代码,因此,与之前的推文【python量化】将Informer用于股价预测雷同,起首将NSTransformer的源码下载到本地,然后将我们的数据集放到某个路径下,这里仍然用到了上证指数14到17年四年的开高低收获交量数据。在将NSTransformer的代码用于我们的股票数据预测任务时,同样必要明白的是,我们的任务是基于高低收获交量来预测收盘价,这是一个多变量输入,单变量输出的预测任务。所以重要必要修改下面几个参数的设置。当必要进行差异预测任务时,如增长某些特性,预测多个目的变量等则可通过修改features,target,enc_in,dec_in以及c_out参数进行实现。NSTransformer的重要参数与Informer雷同,重要包括:
  1. model: ns_transformer,可以选择其他模型包括Informer,Transformer以及Autoformer。
  2. data: 设置为custom,用于调用Dataset_Custom类,从而可以自定义数据集。
  3. root_path:指定数据集存放的文件夹。
  4. data_path:指定csv数据集的名称。
  5. features:设置为MS,这是因为我们是用开高低收成交量来预测收盘价,所以是多变量输出来预测单变量。
  6. target:表示预测变量,设置为Close,对应我们csv文件预测变量的列名。
  7. freq:表示预测频率,设置为d,因为我们用到的是日线级别的数据。
  8. seq_len:表示输入encoder的序列长度,这里设置为20。
  9. label_len:表示输入decoder中的token的长度,这里设置为10,即通过前10个真实值来辅助decoder进行预测。
  10. pred_len:表示预测序列的长度,这里设置为5,即表示预测后5个时刻的序列值。
  11. enc_in:表示encoder的输入维度,这里设置为5,因为我们用到了开高低收以及成交量5个特征。
  12. dec_in:表示decoder的输入维度,同enc_in。
  13. c_out:表示输出序列的维度,这里设置为1,因为我们的目标变量只有收盘价。
  14. moving_avg: 移动平均的窗口大小。
  15. p_hidden_dims:去平稳性映射器的隐层维度。
  16. p_hidden_layers:映射器的层数。
复制代码
其他参数像模子层数,维度之类的可以根据自己的电脑配置进行修改。下面是进行上证指数预测实验的参数配置:
  1. args.is_training = 1
  2. args.model_id = 'test'
  3. args.model='ns_Transformer'  # model name, options: [ns_Transformer, Transformer]
  4. # data loader
  5. args.data = 'custom' # dataset type
  6. args.root_path ='./data/stock/' # root path of the data file
  7. args.data_path ='SH000001.csv'  # data file
  8. args.features='MS'   # forecasting task, options:[M, S, MS]; M:multivariate predict multivariate, S:univariate predict univariate, MS:multivariate predict univariate'
  9. args.target='Close'    # 'target feature in S or MS task'
  10. args.freq = 'd'     # freq for time features encoding, options:[s:secondly, t:minutely, h:hourly, d:daily, b:business days, w:weekly, m:monthly], you can also use more detailed freq like 15min or 3h'
  11. args.checkpoints ='./checkpoints/'  # 'location of model checkpoints'
  12. # forecasting task
  13. args.seq_len = 20     # input sequence length
  14. args.label_len = 10   # start token length
  15. args.pred_len = 5    # prediction sequence lengthargs.
  16. # model define
  17. args.enc_in = 5    # encoder input size
  18. args.dec_in = 5    # decoder input size
  19. args.c_out = 1     # output size
  20. args.d_model = 256 # dimension of model
  21. args.n_heads = 4   # num of heads
  22. args.e_layers = 2  # num of encoder layers
  23. args.d_layers = 1  # num of decoder layers
  24. args.d_ff = 256   # dimension of fcn
  25. args.moving_avg = 5  # window size of moving average
  26. args.factor = 1    # attn factor
  27. args.distil= True  # whether to use distilling in encoder, using this argument means not using distilling
  28. args.dropout = 0.05    # dropout
  29. args.embed = 'timeF'   # time features encoding, options:[timeF, fixed, learned]
  30. args.activation = 'gelu'  # 'activation'
  31. args.output_attention = True    #  help='whether to output attention in encoder
  32. args.do_predict = True # whether to predict unseen future data
  33. # optimization
  34. args.num_workers = 0     # data loader num workers
  35. args.itr = 1              # experiments times
  36. args.train_epochs = 20    # train epochs
  37. args.batch_size = 32      # batch size of train input data
  38. args.patience = 3         # early stopping patience
  39. args.learning_rate = 0.0001  # optimizer learning rate
  40. args.des = 'test'         # exp description
  41. args.loss = 'mse'         # loss function
  42. args.lradj = 'type1'      # adjust learning rate
  43. args.use_amp = False       # use automatic mixed precision training
  44. # GPU
  45. args.use_gpu = False       # use gpu
  46. args.gpu = 0              # gpu
  47. args.use_multi_gpu = False
  48. args.devices = '0,1,2,3'  # device ids of multile gpus
  49. args.seed = 2021          # random seed
  50. # de-stationary projector params
  51. args.p_hidden_dims = [128, 128]  # hidden layer dimensions of projector (List)
  52. args.p_hidden_layers = 2         # number of hidden layers in projector
复制代码
按照设置的参数,然后对模子进行练习跟测试:
  1. Exp = Exp_Main
  2. if args.is_training:
  3.     for ii in range(args.itr):
  4.         # setting record of experiments
  5.         setting = '{}_{}_{}_ft{}_sl{}_ll{}_pl{}_dm{}_nh{}_el{}_dl{}_df{}_fc{}_eb{}_dt{}_{}_{}'.format(
  6.             args.model_id,
  7.             args.model,
  8.             args.data,
  9.             args.features,
  10.             args.seq_len,
  11.             args.label_len,
  12.             args.pred_len,
  13.             args.d_model,
  14.             args.n_heads,
  15.             args.e_layers,
  16.             args.d_layers,
  17.             args.d_ff,
  18.             args.factor,
  19.             args.embed,
  20.             args.distil,
  21.             args.des, ii)
  22.         exp = Exp(args)  # set experiments
  23.         print('>>>>>>>start training : {}>>>>>>>>>>>>>>>>>>>>>>>>>>'.format(setting))
  24.         exp.train(setting)
  25.         print('>>>>>>>testing : {}<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<'.format(setting))
  26.         exp.test(setting)
  27.         if args.do_predict:
  28.             print('>>>>>>>predicting : {}<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<'.format(setting))
  29.             exp.predict(setting, True)
  30.         torch.cuda.empty_cache()
  31. else:
  32.     ii = 0
  33.     setting = '{}_{}_{}_ft{}_sl{}_ll{}_pl{}_dm{}_nh{}_el{}_dl{}_df{}_fc{}_eb{}_dt{}_{}_{}'.format(
  34.     args.model_id,
  35.     args.model,
  36.     args.data,
  37.     args.features,
  38.     args.seq_len,
  39.     args.label_len,
  40.     args.pred_len,
  41.     args.d_model,
  42.     args.n_heads,
  43.     args.e_layers,
  44.     args.d_layers,
  45.     args.d_ff,
  46.     args.factor,
  47.     args.embed,
  48.     args.distil,
  49.     args.des, ii)
  50.     exp = Exp(args)  # set experiments
  51.     print('>>>>>>>testing : {}<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<'.format(setting))
  52.     exp.test(setting, test=1)
  53.     torch.cuda.empty_cache()
复制代码
下面是在CPU上的练习过程。颠末练习,可以看出模子的练习集上的loss不停下降,由于参加了early stop机制,所以颠末6个epoch模子就制止练习了。颠末在测试集上的测试,NSTransformer实现了0.0027的mse跟0.0415的mae(归一化后的效果)。必要注意的是,模子颠末练习跟测试之后,会在当前路径的./checkpoints中生存模子参数,在./results/{settings}/下天生pred.npy以及true.npy文件用来分别存放测试集上的预测效果跟ground truth。
  1. Use CPU
  2. >>>>>>>start training : test_ns_Transformer_custom_ftMS_sl20_ll10_pl5_dm256_nh4_el2_dl1_df256_fc1_ebtimeF_dtTrue_test_0>>>>>>>>>>>>>>>>>>>>>>>>>>
  3. train 659
  4. val 95
  5. test 191
  6. Epoch: 1 cost time: 1.7922062873840332
  7. Epoch: 1, Steps: 20 | Train Loss: 0.0605105 Vali Loss: 0.0028209 Test Loss: 0.0028931
  8. Validation loss decreased (inf --> 0.002821).  Saving model ...
  9. Updating learning rate to 0.0001
  10. Epoch: 2 cost time: 1.569800615310669
  11. Epoch: 2, Steps: 20 | Train Loss: 0.0374645 Vali Loss: 0.0030634 Test Loss: 0.0028165
  12. EarlyStopping counter: 1 out of 3
  13. Updating learning rate to 5e-05
  14. Epoch: 3 cost time: 1.5588312149047852
  15. Epoch: 3, Steps: 20 | Train Loss: 0.0328267 Vali Loss: 0.0024335 Test Loss: 0.0027673
  16. Validation loss decreased (0.002821 --> 0.002434).  Saving model ...
  17. Updating learning rate to 2.5e-05
  18. Epoch: 4 cost time: 1.5059726238250732
  19. Epoch: 4, Steps: 20 | Train Loss: 0.0296629 Vali Loss: 0.0031771 Test Loss: 0.0028892
  20. EarlyStopping counter: 1 out of 3
  21. Updating learning rate to 1.25e-05
  22. Epoch: 5 cost time: 1.6665441989898682
  23. Epoch: 5, Steps: 20 | Train Loss: 0.0289504 Vali Loss: 0.0029475 Test Loss: 0.0027112
  24. EarlyStopping counter: 2 out of 3
  25. Updating learning rate to 6.25e-06
  26. Epoch: 6 cost time: 1.5528459548950195
  27. Epoch: 6, Steps: 20 | Train Loss: 0.0285680 Vali Loss: 0.0029404 Test Loss: 0.0027616
  28. EarlyStopping counter: 3 out of 3
  29. Early stopping
  30. >>>>>>>testing : test_ns_Transformer_custom_ftMS_sl20_ll10_pl5_dm256_nh4_el2_dl1_df256_fc1_ebtimeF_dtTrue_test_0<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
  31. test 191
  32. test shape: (5, 32, 5, 1) (5, 32, 5, 1)
  33. test shape: (160, 5, 1) (160, 5, 1)
  34. mse:0.0027673370204865932, mae:0.04154161363840103
复制代码
调用模子的predict()方法可以直接预测所有数据背面的未知数据,由于这里预测长度为5,所以直接调用就相称于预测后5天的收盘价走势了。预测的效果会生存在./results/{settings}/下面的real_prediction.npy文件中。
  1. exp = Exp(args)
  2. exp.predict(setting, True)
  3. # the prediction will be saved in ./results/{setting}/real_prediction.npy
  4. prediction = np.load('./results/'+setting+'/real_prediction.npy')
  5. plt.figure()
  6. plt.plot(prediction[0,:,-1])
  7. plt.show()
复制代码

  最后将完备的测试集上的预测效果进行可视化。
  1. plt.figure()
  2. plt.plot(trues[:,0,-1].reshape(-1), label='GroundTruth')
  3. plt.plot(preds[:,0,-1].reshape(-1), label='Prediction')
  4. plt.legend()
  5. plt.show()
复制代码

  除此之外,NSTransformer的源码中也提供了Transformer、Informer以及Autoformer的实现。为了直观展示这几种模子之间的预测体现,我们采用相同的超参数对另外几种模子进行实验,并将其可视化的效果进行展示。必要注意是这里为了方便实验,每种模子的超参没有进行寻优而是统一采用了NSTransformer模子的实验配置。
  

  
Transformer

  

  
Informer

  

  
Autoformer

  从可视化效果中可以看出相比于其他Transformer模子,NSTransformer实现了最好的拟合效果。最后统计这几种模子的MSE跟MAE(未反归一化),从偏差上也可以看出NSTransformer实现了最低的预测偏差,其次是Autoformer,Informer以及Transformer。
  1. NSTransformer:
  2. mse:0.0027673370204865932, mae:0.04154161363840103
  3. Transformer:
  4. mse:0.0147677231580019, mae:0.10331685841083527
  5. Informer:
  6. mse:0.01395915262401104, mae:0.09441611915826797
  7. Autoformer:
  8. mse:0.008317378349602222, mae:0.07286103069782257
复制代码
4

  总结
  时间序列的非安稳性是现及时间序列数据中存在的重要特性,尤其是对金融时间序列来说。而现有大多数方法都会对数据做安稳化处理来提拔预测性能,然而,安稳化处理的过程也会导致一部分信息的丢失。而NSTransformer通过计划的序列安稳化以及去安稳化注意力机制,使得它既能利用安稳序列的可预测性,又能保持原始序列固有的时间依靠性。文中通过将NSTransformer与其他模子包括Transformer,Informer以及Autoformer的实验对比,进一步证明了NSTransformer的预测性能。因此,在面对非安稳,高噪声的股票数据预测中,NSTransformer大概可以取得较好的预测体现。
  本文内容仅仅是技能探究和学习,并不构成任何投资发起。
  参考文献:
  Liu, Y., Wu, H., Wang, J., & Long, M. (2022). Non-stationary Transformers: Exploring the Stationarity in Time Series Forecasting. In Advances in Neural Information Processing Systems.
  NeurIPS2022 | NSTransformers: 非安稳时间序列的通用预测框架 - 游凯超的文章 - 知乎 https://zhuanlan.zhihu.com/p/587665491
  获取完备代码与数据以及其他历史文章完备源码与数据可参加《人工智能量化实验室》知识星球。
  
  
《人工智能量化实验室》知识星球

  
  参加人工智能量化实验室知识星球,您可以获得:(1)定期推送最新人工智能量化应用相干的研究效果,包括高程度期刊论文以及券商优质金融工程研究陈诉,便于您随时随地了解最新前沿知识;(2)公众号历史文章Python项目完备源码;(3)优质Python、呆板学习、量化买卖业务相干电子书PDF;(4)优质量化买卖业务资料、项目代码分享;(5)跟星友一起交流,结交守望相助朋侪。(6)向博主发起提问,答疑解惑。
  

来源:https://blog.csdn.net/FrankieHello/article/details/129222504
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则