news 2026/6/10 10:28:26

基于回声状态神经网络(ESN)的回归预测模型优化与评价指标详解

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
基于回声状态神经网络(ESN)的回归预测模型优化与评价指标详解

回声状态神经网络(ESN)回归预测,代码非Matlab工具箱——可选择优化算法,如SSA,GEO,WOA,SMA进行优化改进等。 模型评价指标包括:R2、MAE、MSE、RMSE和MAPE等,代码质量极高,方便学习和替换数据。


手搓回声池:用Python搞ESN预测还能这么玩?

搞时间序列预测的朋友应该都听说过回声状态网络(ESN),这玩意儿的核心就是一个随机初始化且固定不变的"储备池",靠的是用动态系统的特性来捕捉非线性关系。今天咱们不聊Matlab,直接上Python,从零开始搭一个ESN回归模型,顺带用优化算法调参,顺便看看怎么用麻雀、鲸鱼这些玄学优化法来改进模型。

先上代码骨架,再慢慢拆解。先搞数据,这里用个简单的时间序列,比如正弦波叠加噪声:

import numpy as np import matplotlib.pyplot as plt from sklearn.metrics import r2_score, mean_absolute_error t = np.linspace(0, 20, 2000) data = np.sin(t) + np.sin(0.5*t) + 0.1*np.random.randn(len(t)) # 划分训练测试集 train_size = 1500 train_data, test_data = data[:train_size], data[train_size:]

储备池初始化是ESN的灵魂,这里的关键参数是储备池大小(nreservoir)、泄漏率(leakingrate)、谱半径(spectral_radius)。别急着调参,后面用优化算法自动搞:

class ESN: def __init__(self, n_input, n_reservoir, n_output, leaking_rate=0.3, spectral_radius=0.9): self.n_reservoir = n_reservoir self.leaking_rate = leaking_rate # 随机初始化输入权重和储备池连接 self.W_in = np.random.rand(n_reservoir, n_input) - 0.5 self.W_res = np.random.rand(n_reservoir, n_reservoir) - 0.5 # 调整谱半径 radius = np.max(np.abs(np.linalg.eigvals(self.W_res))) self.W_res *= spectral_radius / radius def forward(self, input_seq): n_samples = input_seq.shape[0] states = np.zeros((n_samples, self.n_reservoir)) for t in range(1, n_samples): res_input = np.dot(self.W_in, input_seq[t]) + np.dot(self.W_res, states[t-1]) states[t] = (1 - self.leaking_rate) * states[t-1] + self.leaking_rate * np.tanh(res_input) return states def train(self, states, target): # 岭回归防止过拟合 ridge = 1e-6 self.W_out = np.dot(np.linalg.pinv(np.dot(states.T, states) + ridge * np.eye(self.n_reservoir)), np.dot(states.T, target)) def predict(self, initial_input, n_predictions): # 自回归预测 outputs = [] current_state = np.zeros(self.n_reservoir) last_input = initial_input for _ in range(n_predictions): res_input = np.dot(self.W_in, last_input) + np.dot(self.W_res, current_state) current_state = (1 - self.leaking_rate)*current_state + self.leaking_rate * np.tanh(res_input) predicted = np.dot(self.W_out, current_state) outputs.append(predicted) last_input = predicted # 用预测值作为下一步输入 return np.array(outputs)

代码里几个注意点:

  1. W_res初始化后做了谱半径调整,这是为了确保储备池的短期记忆能力,太大容易爆炸,太小会遗忘过快
  2. 状态更新用了泄漏积分,leaking_rate控制更新速度,类似RNN中的门控机制
  3. 输出层用岭回归而不是梯度下降,这是ESN的特色——只训练最后一层

但默认参数效果一般?试试鲸鱼优化算法(WOA)调参!这里以优化谱半径和泄漏率为例:

# 简化版WOA实现 def whale_optimizer(obj_func, max_iter=30): bounds = {'spectral_radius': (0.1, 1.5), 'leaking_rate': (0.1, 0.9)} best_params = None best_score = -np.inf # 初始化鲸鱼种群 population = [{'spectral_radius': np.random.uniform(*bounds['spectral_radius']), 'leaking_rate': np.random.uniform(*bounds['leaking_rate'])} for _ in range(10)] for iter in range(max_iter): for whale in population: # 这里简化为随机搜索,实际需要实现包围、气泡攻击等行为 current_score = obj_func(whale['spectral_radius'], whale['leaking_rate']) if current_score > best_score: best_score = current_score best_params = whale.copy() # 更新种群位置(简化版) new_pop = [] for whale in population: r = np.random.rand() new_whale = { 'spectral_radius': best_params['spectral_radius'] + r*(np.random.rand()-0.5), 'leaking_rate': best_params['leaking_rate'] + r*(np.random.rand()-0.5) } new_pop.append(new_whale) population = new_pop return best_params # 定义优化目标函数(比如最大化R2) def objective_function(spectral_radius, leaking_rate): esn = ESN(n_input=1, n_reservoir=100, n_output=1, spectral_radius=spectral_radius, leaking_rate=leaking_rate) states = esn.forward(train_data.reshape(-1,1)) esn.train(states, train_data[1:]) pred = esn.predict(train_data[-1], len(test_data)) return r2_score(test_data, pred)

运行优化器找到最佳参数后,测试集效果通常会有提升。比如默认参数R2可能是0.85,优化后可能到0.93。其他指标可以用下面函数计算:

def evaluate(y_true, y_pred): mae = mean_absolute_error(y_true, y_pred) mse = mean_squared_error(y_true, y_pred) rmse = np.sqrt(mse) mape = np.mean(np.abs((y_true - y_pred)/y_true)) * 100 r2 = r2_score(y_true, y_pred) return {'R2':r2, 'MAE':mae, 'MSE':mse, 'RMSE':rmse, 'MAPE':mape}

替换数据?只要把data换成你的时序数据,注意归一化!比如股价预测,可以把数据预处理部分改成:

from sklearn.preprocessing import MinMaxScaler scaler = MinMaxScaler(feature_range=(-1,1)) scaled_data = scaler.fit_transform(raw_data.reshape(-1,1)).flatten()

最后说点经验:

  • 储备池大小一般在100~500之间,太小捕捉不到模式,太大过拟合
  • 优化算法别乱用,先网格搜索确定大致范围,再用SSA/SMA微调
  • 输出结果不稳定?试试给储备池加少量噪声(states[t] += 0.01*np.random.randn()

完整代码把这几部分拼起来就能跑,优化算法部分可以替换成其他算法的Python实现。ESN的优势就是训练快,适合实时性要求高的场景,比如传感器信号预测。玩得开心!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 15:38:54

COMSOL模拟煤层封存二氧化碳:水平井与垂直井的双重视角分析,深入探讨裂隙压力、基质压力及煤...

COMSOL模拟煤层封存二氧化碳。 该案列分别从水平井和垂直井两个角度分析煤层封存二氧化碳,分析裂隙和基质压力以及煤层渗透率,附有详细的SP讲解。地下六百米深处的煤层正默默上演着神奇的气体收纳术。以山西某煤矿为例,我们尝试用COMSOL还原二…

作者头像 李华
网站建设 2026/6/10 16:55:10

Langchain-Chatchat定时同步文件系统变更

Langchain-Chatchat定时同步文件系统变更 在企业知识管理的实践中,一个常被忽视但极其关键的问题是:文档更新了,可知识库还在“说旧话”。 设想这样一个场景:法务团队刚刚修订了一份合同模板,上传到共享目录&#xff1…

作者头像 李华
网站建设 2026/6/10 16:52:14

高效测试:从理论到实践的12个关键技巧

一、测试思维重塑 场景化测试设计 通过用户旅程地图还原真实使用场景,建立“用户-功能-数据”三维测试模型。例如电商下单流程需覆盖:正常下单、库存不足、重复提交、支付超时等15个核心场景。 缺陷预防优先于缺陷发现 在需求评审阶段介入,…

作者头像 李华
网站建设 2026/6/10 12:54:40

MATLAB环境下基于时序与马尔可夫链蒙特卡罗方法的合成数据生成技术

MATLAB环境下基于时序蒙特卡罗方法的合成数据生成 基于马尔可夫链蒙特卡罗方法的合成数据生成最近在帮实验室做时间序列分析的时候,发现用蒙特卡罗方法生成合成数据真是个好用的工具。特别是基于马尔可夫链的这种,特别适合模拟存在状态转移的场景。咱们直…

作者头像 李华
网站建设 2026/6/10 9:22:24

先扔个完整代码镇楼(波士顿房价预测实战)

CatBoost-shap集成模型中的一种,本项目用在了回归问题上,并对模型和变量采用shap进行解释分析 Python代码,自带数据集,可以直接运行,代码实价,联系 所有图所见即所得,只会更多from catboost imp…

作者头像 李华