0

0

深入理解LSTM时间序列预测:数据准备与模型构建最佳实践

碧海醫心

碧海醫心

发布时间:2025-12-08 17:40:17

|

759人浏览过

|

来源于php中文网

原创

深入理解lstm时间序列预测:数据准备与模型构建最佳实践

本教程旨在解决使用LSTM进行时间序列预测时常见的“数据基数模糊”错误,并提供一套完整的解决方案。文章详细阐述了如何正确准备序列数据,将其转换为LSTM模型所需的输入格式,并指导读者构建一个适用于回归任务的LSTM模型,包括选择合适的激活函数和损失函数,最终实现准确的时间序列预测。

1. 理解LSTM与时间序列数据

长短期记忆网络(LSTM)是循环神经网络(RNN)的一种变体,特别擅长处理和预测时间序列数据。在进行时间序列预测时,我们通常希望模型能够根据历史序列数据预测未来的一个或多个值。一个常见的场景是,给定前 N 个时间步的数据,预测第 N+1 个时间步的值。

然而,初学者在使用Keras构建LSTM模型时,常会遇到 ValueError: Data cardinality is ambiguous 这样的错误。这通常是由于输入 X 和输出 Y 的样本数量不匹配,或者输入数据的形状不符合LSTM层的要求所致。此外,输出层的激活函数选择不当也是一个常见问题

2. 时间序列数据准备:构建监督学习样本

要训练一个LSTM模型,我们需要将原始时间序列数据转换为监督学习问题所需的输入-输出对。假设我们有一个一维时间序列 [1, 2, 3, 4, 5, 6, 7],并且我们知道每个样本与其前两个样本之间存在关系(例如,1, 2 预测 3;2, 3 预测 4)。这意味着我们的“序列长度”或“回溯步长”是 2。

我们需要创建一个数据生成器,将原始序列转换为以下形式:

  • 输入 (X): 包含 sequences_length 个连续时间步的序列。
  • 输出 (Y): 紧随输入序列的下一个时间步的值。

以下是一个实现此逻辑的Python函数:

import numpy as np
import tensorflow as tf
from tensorflow import keras
from tensorflow.keras import layers

# 原始时间序列数据
data = np.array([1, 2, 3, 4, 5, 6, 7])
sequences_length = 2 # 定义回溯步长

def create_sequences(data, sequences_length):
    """
    将一维时间序列数据转换为LSTM模型所需的输入-输出对。

    参数:
    data (np.array): 原始一维时间序列数据。
    sequences_length (int): 每个输入序列的长度(回溯步长)。

    返回:
    tuple: (X, Y),其中X是输入序列数组,Y是对应的目标值数组。
    """
    X, Y = [], []
    for i in range(len(data) - sequences_length):
        # 提取输入序列
        X.append(data[i : i + sequences_length])
        # 提取目标值
        Y.append(data[i + sequences_length])
    return np.array(X), np.array(Y)

# 生成训练数据
X_raw, Y = create_sequences(data, sequences_length)

# 打印生成的样本以进行检查
print("原始X样本:")
for i in range(X_raw.shape[0]):
    print(f"X[{i}]: {X_raw[i]}, Y[{i}]: {Y[i]}")

# 原始X样本:
# X[0]: [1 2], Y[0]: 3
# X[1]: [2 3], Y[1]: 4
# X[2]: [3 4], Y[2]: 5
# X[3]: [4 5], Y[3]: 6
# X[4]: [5 6], Y[4]: 7

从上述输出可以看出,我们成功从7个原始数据点中生成了5个监督学习样本。现在,X_raw 的形状是 (5, 2),Y 的形状是 (5,)。这解决了 Data cardinality is ambiguous 的问题,因为 X 和 Y 现在都有相同的样本数量(5个)。

3. LSTM输入形状的重塑

LSTM层期望的输入形状是 (samples, timesteps, features):

  • samples: 样本数量,即训练数据中的输入-输出对的数量。
  • timesteps: 每个序列的长度,即 sequences_length。
  • features: 每个时间步的特征数量。对于一维时间序列,特征数量通常是 1。

因此,我们需要将 X_raw 从 (5, 2) 重塑为 (5, 2, 1)。

WeShop唯象
WeShop唯象

WeShop唯象是国内首款AI商拍工具,专注电商产品图片的智能生成。

下载
# 重塑X以符合LSTM输入要求
X = np.reshape(X_raw, (X_raw.shape[0], sequences_length, 1))

print(f"\n重塑后的X形状: {X.shape}")
print(f"Y的形状: {Y.shape}")

现在,X 的形状是 (5, 2, 1),Y 的形状是 (5,)。

4. 构建LSTM模型

在构建LSTM模型时,除了输入形状,还需要注意输出层的选择。由于我们正在进行数值预测(回归任务),输出层应该是一个 Dense 层,且不应使用 softmax 激活函数。softmax 适用于多类别分类问题,而对于回归问题,我们通常使用线性激活(即不指定激活函数,Keras默认是线性)。

# 构建LSTM模型
model = keras.Sequential([
    # LSTM层,input_shape=(timesteps, features)
    layers.LSTM(64, input_shape=(sequences_length, 1)),
    # 全连接输出层,用于回归任务,不使用激活函数(默认为线性)
    layers.Dense(1)
])

# 编译模型
# 优化器选择 'adam'
# 损失函数选择 'mse' (均方误差) 适用于回归任务
model.compile(optimizer="adam", loss="mse")

model.summary()

注意事项:

  • layers.LSTM(64, ...) 中的 64 是LSTM单元的数量,可以根据模型复杂度和数据规模调整。
  • input_shape=(sequences_length, 1) 严格匹配我们准备好的 X 的 (timesteps, features) 部分。
  • layers.Dense(1) 表示输出一个单一的连续值。
  • loss="mse" (Mean Squared Error) 是回归任务的常用损失函数。
  • metrics=["accuracy"] 不适用于回归任务,应避免使用。

5. 模型训练

使用准备好的 X 和 Y 数据对模型进行训练。为了更好地学习数据中的模式,通常需要多个 epochs。

# 训练模型
print("\n开始训练模型...")
model.fit(X, Y, epochs=1000, batch_size=1, verbose=0) # verbose=0 关闭训练进度输出
print("模型训练完成。")

6. 模型预测

训练完成后,我们可以使用模型对新的序列数据进行预测。重要的是,用于预测的输入数据也必须按照与训练数据相同的格式进行准备和重塑。

假设我们想预测 [8, 9] 之后的下一个值。

# 准备用于预测的新数据
inference_data_raw = np.array([[8, 9]]) # 注意这里是二维数组
inference_data = np.reshape(inference_data_raw, (1, sequences_length, 1))

# 进行预测
print(f"\n预测 {inference_data_raw.reshape(-1)} 之后的下一个值...")
prediction = model.predict(inference_data)

print(f"预测结果: {prediction[0][0]:.4f}")
# 期望结果接近 10

7. 总结与最佳实践

本教程展示了如何正确地为LSTM时间序列预测任务准备数据和构建模型。核心要点包括:

  1. 数据基数匹配: 确保输入 X 和输出 Y 的样本数量(第一维度)完全一致,这是避免 Data cardinality is ambiguous 错误的关键。
  2. LSTM输入形状: LSTM层期望的输入形状是 (samples, timesteps, features)。对于一维时间序列,features 通常为 1。
  3. 数据生成器: 编写一个函数来将原始时间序列转换为 (input_sequence, target_value) 对,这是构建监督学习样本的有效方法。
  4. 输出层与激活函数: 对于回归任务,输出层应使用 Dense(1) 且通常不指定激活函数(默认为线性),避免使用 softmax。
  5. 损失函数: 针对回归任务,选择 mse (均方误差) 或 mae (平均绝对误差) 等合适的损失函数。
  6. 预测数据格式: 进行预测时,新的输入数据必须与训练数据具有相同的 (timesteps, features) 形状,并包裹在 (1, timesteps, features) 的批次维度中。

遵循这些最佳实践,可以有效避免常见的错误,并成功构建和训练用于时间序列预测的LSTM模型。

相关专题

更多
python开发工具
python开发工具

php中文网为大家提供各种python开发工具,好的开发工具,可帮助开发者攻克编程学习中的基础障碍,理解每一行源代码在程序执行时在计算机中的过程。php中文网还为大家带来python相关课程以及相关文章等内容,供大家免费下载使用。

773

2023.06.15

python打包成可执行文件
python打包成可执行文件

本专题为大家带来python打包成可执行文件相关的文章,大家可以免费的下载体验。

684

2023.07.20

python能做什么
python能做什么

python能做的有:可用于开发基于控制台的应用程序、多媒体部分开发、用于开发基于Web的应用程序、使用python处理数据、系统编程等等。本专题为大家提供python相关的各种文章、以及下载和课程。

765

2023.07.25

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

719

2023.07.31

python教程
python教程

Python已成为一门网红语言,即使是在非编程开发者当中,也掀起了一股学习的热潮。本专题为大家带来python教程的相关文章,大家可以免费体验学习。

1425

2023.08.03

python环境变量的配置
python环境变量的配置

Python是一种流行的编程语言,被广泛用于软件开发、数据分析和科学计算等领域。在安装Python之后,我们需要配置环境变量,以便在任何位置都能够访问Python的可执行文件。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

570

2023.08.04

python eval
python eval

eval函数是Python中一个非常强大的函数,它可以将字符串作为Python代码进行执行,实现动态编程的效果。然而,由于其潜在的安全风险和性能问题,需要谨慎使用。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

579

2023.08.04

scratch和python区别
scratch和python区别

scratch和python的区别:1、scratch是一种专为初学者设计的图形化编程语言,python是一种文本编程语言;2、scratch使用的是基于积木的编程语法,python采用更加传统的文本编程语法等等。本专题为大家提供scratch和python相关的文章、下载、课程内容,供大家免费下载体验。

751

2023.08.11

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

25

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 18.3万人学习

Django 教程
Django 教程

共28课时 | 3.4万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号