首页 > 其他分享 >ARIMA时间序列模型水质预测应用

ARIMA时间序列模型水质预测应用

时间:2024-05-25 15:33:07浏览次数:14  
标签:模型 ARIMA 差分 df2 水质 序列 import root

ARIMA 时间序列模型简介

时间序列是研究数据随时间变化而变化的一种算法,是一种预测性分析算法。它的基本出发点就是事物发展都有连续性,按照它本身固有的规律进行。ARIMA(p,d,q)模型全称为差分自回归移动平均模型 (Autoregressive Integrated Moving Average Model,简记 ARIMA). 是比较成熟且简单的时间预测模型之一。其中 AR 为自回归, I 为差分, MA 为移动平均。
趋势参数:

  • p:趋势自回归阶数。
  • d:趋势差分阶数。
  • q:趋势移动平均阶数。

差分

差分(difference)又名差分函数或差分运算,差分的结果反映了离散量之间的一种变化,是研究离散数学的一种工具。它将原函数f(x) 映射到f(x+a)-f(x+b) 。差分运算,相应于微分运算,是微积分中重要的一个概念。总而言之,差分对应离散,微分对应连续。差分又分为前向差分、向后差分及中心差分三种。
通常情况下我们用到的是前向差分公式如下:
xk=x0+kh,(k=0,1,...,n)
△f(xk)=f(xk+1)−f(xk)
差分的阶
称为阶的差分,即前向阶差分 ,如果数学运用根据数学归纳法,有其中,为二项式系数。特别的,有前向差分有时候也称作数列的二项式变换

在高锰酸盐指数序列预测可行性的说明

通过观察水质变化趋势,高锰酸盐指数波动不剧烈,存在明显的中心波动规律。

python实现

环境准备

import numpy as np
import pandas as pd
import matplotlib.pyplot as plt
import seaborn as sns
import warnings
import pymysql

warnings.filterwarnings("ignore")  
# 忽略警告,不然一大堆警告,多是python及对应包升高导致,不影响使用
from IPython.core.interactiveshell import InteractiveShell
InteractiveShell.ast_node_interactivity = "all"
from matplotlib.pylab import style  # 自定义图表风格
style.use('ggplot')

# 解决中文乱码问题
plt.rcParams['font.sans-serif'] = ['Simhei']
# 解决坐标轴刻度负号乱码
plt.rcParams['axes.unicode_minus'] = False

# pip install statsmodels

from statsmodels.graphics.tsaplots import plot_acf, plot_pacf  # 自相关图、偏自相关图
from statsmodels.tsa.stattools import adfuller as ADF  # 平稳性检验
from statsmodels.stats.diagnostic import acorr_ljungbox  # 白噪声检验
import statsmodels.api as sm  # D-W检验,一阶自相关检验
from statsmodels.graphics.api import qqplot  # 画QQ图,检验一组数据是否服从正态分布
from statsmodels.tsa.arima.model import ARIMA

连接数据

通过数据库,excel 都可以,列名为监测时间、设备名称、设备因子、监测值。

def conn_sql():
    conn = pymysql.connect(host=" ",
                   port= ,
                   user=" ",
                   password=" ",
                   db=" ",
                   charset="utf8")
    sql = ""
    read_sql = pd.read_sql(sql, conn)
    return read_sql
read_sql=conn_sql()

数据处理

def nseri(s,y ):
    aidunqiao = read_sql.loc[read_sql['设备名称'] == s, :]
    ai_cod = aidunqiao.loc[read_sql['监测因子'] == y, :]
    ai_cod_mn = ai_cod.loc[:, ["监测时间", '监测值']]
    baseline = ai_cod.loc[:, ["监测时间", '监测值']]

    ai_cod_mn.set_index('监测时间', inplace=True)
    interp_cod_mn = ai_cod_mn["监测值"].interpolate()
    ai_cod_mn["cod"] = interp_cod_mn
    starttime = baseline.iloc[0, 0]
    rows = baseline.shape[0]
    endtime = baseline.iloc[rows - 1, 0]

    year_month_day = pd.date_range(starttime, endtime, freq="h").strftime("%Y%m%d%h%m%s")
    a_ser = pd.DataFrame({'监测时间': year_month_day})
    a_ser.set_index('监测时间', inplace=True)
    df = pd.concat([a_ser, ai_cod_mn], axis=0, join="outer")
    df = df.reset_index(drop=False)
    df['监测时间'] = pd.to_datetime(df['监测时间'])
    df1 = df.drop_duplicates(subset="监测时间", keep="last", ignore_index=True)
    df2 = df1.sort_values(by="监测时间", ignore_index=True)
    df2["cod"] = df2["监测值"].interpolate()
    df2.drop(columns="监测值", inplace=True)
    df2.set_index('监测时间', inplace=True)
    return df2

主要是 将数据生成无空连续的逐小时 时间序列数据 插值方法为线性插值

数据解读

查看acf

df2 = df2.dropna()
# 解决有nan的问题
plot_acf(df2,lags=50).show()

解读 拖尾为p 。基本大于0.5 现在和未来有很强的相关性

单位根检验

print('原始序列的ADF检验结果为:',ADF(df2.cod))

原始序列的ADF检验结果为: (-7.19465930048855, 2.452407467867345e-10, 37, 9199, {'1%': -3.431061069214289, '5%': -2.8618542472812902, '10%': -2.5669372687639176}, 11281.50483165621)

解读:P值小于显著性水平α(0.05),不接受原假设(非平稳序列),说明原始序列是平稳序列。

白噪声检验

print('一阶差分序列的白噪声检验结果为:',acorr_ljungbox(df2,lags=1,return_df =bool))

一阶差分序列的白噪声检验结果为: lb_stat lb_pvalue 1 7467.631465 0.0

p值为0小于0.05,不是白噪声

综上可以采用 arima 模型

定阶 人工识图

#一阶差分,我们不需要这么做,看下代码怎么写的。
df2_mn=df2.diff(periods=1, axis=0).dropna()
#自相关图
plot_acf(df2,lags=20).show()
#解读:拖尾 有长期相关性 p 取1 
#偏自相关图 
plot_pacf(df2,lags=20).show()
#偏自相关图
plot_pacf(df2,lags=50).show()
#解读:自相关图,0阶拖尾;偏自相关图,截尾。则ARIMA(p,d,q)=ARIMA(1,0,n)

参数调优

AIC调优

from statsmodels.tsa.arima.model import ARIMA
aic_matrix=[]
for p in range(5):
    tmp=[]
    for q in range(5):
        try:
            tmp.append(ARIMA(df2,order=(p,0,q)).fit().aic)
        except:
            tmp.append(None)
    aic_matrix.append(tmp)
aic_matrix

# p,q=aic_matrix.stack().idxmin() #最小值的索引
# 手动查找最小值 同样为1,0,4

也可以用BIC调优 不再赘述

模型建立

model = ARIMA(df2, order=(1, 0, 4))
result_arima = model.fit()

模型预测

定义画图函数

def pic1(result_arima,df2):
    t1 = "2022/7/6 00:00:00"
    t2 = "2022/7/8 00:00:00"
    predict_more=result_arima.predict(t1 ,t2 )
    t = pd.date_range(t1, t2 , freq="h").strftime("%y%m%d%h%m%s")
    new_ticks = pd.date_range(t1, t2 , freq="d").strftime("%y%m%d%h%m%s")
    axc.clear()
    axc.set_title("局部历史值与真实值对比")
    axc.plot(t,df2[t1 :t2],linestyle = "--",alpha=0.5)
    axc.plot(t,predict_more,linestyle = ":")
    axc.legend(['真实值','预测值'])
    axc.set_xticks(new_ticks)   

    # 创建画布控件
    canvas = FigureCanvasTkAgg(fig1, master=root)  # A tk.DrawingArea.
    canvas.draw()
    canvas.get_tk_widget().place(x=63,y=200)

def fore_picture(result_arima,df2):
    df3 = df2.reset_index(drop=False)
    rows = df3.shape[0]
    endtime = df3.iloc[rows - 1, 0]
    forecast = pd.Series(result_arima.forecast(48), index=pd.date_range(endtime, periods=48, freq='H'))
    df_last = df2.iloc[-48:]    
    data = pd.concat((df_last, forecast), axis=0)
    data.columns = ['监测值浓度', '未来48小时']
    axc2.clear()
    axc2.set_title("未来48小时预测")
    axc2.plot(data) 
    
        # 创建画布控件
    canvas = FigureCanvasTkAgg(fig2, master=root)  # A tk.DrawingArea.
    canvas.draw()
    canvas.get_tk_widget().place(x=600,y=200)

def compare2(result_arima,df2):
    df3 = df2.reset_index(drop=False)
    rows = df3.shape[0]
    endtime = df3.iloc[rows - 1, 0]
    starttime = df3.iloc[0, 0]
    predict=result_arima.predict(starttime , endtime)
    axc3.clear()
    axc3.set_title("全部预测值真实值对比")
    axc3.plot(df2.index,df2['cod'],linestyle = "--",alpha=0.5)
    axc3.plot(df2.index,predict,linestyle = ":")
    axc3.legend(['真实值','预测值'])
    canvas = FigureCanvasTkAgg(fig3, master=root)  # A tk.DrawingArea.
    canvas.draw()
    canvas.get_tk_widget().place(x=1200,y=200)

模型可视化及GUI初探

用Tkinter 实现自动选择站点及因子

# 副本
from tkinter.ttk import *
from  tkinter import *
import matplotlib
matplotlib.use('TkAgg')
from matplotlib.backends.backend_tkagg import FigureCanvasTkAgg
from matplotlib.figure import Figure

root = Tk()
root.title("ARIMA预测模型")
root.geometry("1800x900+50+50")  # 长x宽+x*y
           
lb1 = Label(root,text='站点选择',fg='black', font=('微软雅黑',15),  height=2,  relief=FLAT)
lb2 = Label(root,text='因子选择',fg='black', font=('微软雅黑',15),  height=2,  relief=FLAT)
lb3 = Label(root,text='预测结果(48h)',fg='black', font=('微软雅黑',15),  height=2,  relief=FLAT)
# lb4 = Label(root,text='历史预测对比',fg='black', font=('微软雅黑',15),  height=2,  relief=FLAT)

# lb5 = Label(root,text=comb1.get(),fg='black', font=('微软雅黑',15),  height=2,  relief=FLAT)
lb1.place(x=63,y=20)
lb2.place(x=300,y=20)
lb3.place(x=63,y=110)
# lb4.place(x=510,y=110)
# lb5.place(x=820,y=110)
var1 = StringVar()
comb1= Combobox(root,textvariable=var1,values = site)
comb1.place(x=63,y=80)
var2 = StringVar()
comb2= Combobox(root,textvariable=var2,values=factor)
comb2.place(x=300,y=80)

def select_device(event):
    s = comb1.get()
    print(s)
    return s

def select_factor (event):
    y = comb2.get()
    print(y)
    return y

comb1.bind("<<ComboboxSelected>>", select_device)
comb2.bind("<<ComboboxSelected>>", select_factor)

def click(event):
    s = comb1.get()
    y = comb2.get()
    df2 = nseri(s,y )
    model = ARIMA(df2, order=(1, 0, 4))
    result_arima = model.fit()
    
    fig1 = Figure(figsize=(4, 3), dpi=120)
    axc = fig1.add_subplot(111)
    axc.clear()
    pic1(result_arima,df2)

    fig2 = Figure(figsize=(4, 3), dpi=120)
    axc2 = fig2.add_subplot(111)
    axc2.clear()
    fore_picture(result_arima,df2)

    fig3 = Figure(figsize=(4, 3), dpi=120)
    axc3 = fig3.add_subplot(111)
    axc3.clear()
    compare2(result_arima,df2)    
    

but1 = Button(root, text='计算',font=('微软雅黑',15),  height=1)
but1.place(x=300,y=110)  

but1.bind("<Button-1>",click)

root.mainloop()

结果预览
111.png

模型评价

模型评价方法: 浓度准确率, 等级准确率

浓度准确率

等级准确率:实测的类别与预测的类别相同时,则视为预测正确,预测正确的个数占预测的总个数的百分比,即为模型预测准确率。指标预测准确率的详细计算方法如下式:image.png

Pi为类别相对误差,T 为验证期内实测值的时间点数,t为实测值与预测值对应的时刻,pit为实测的类别与模拟的类别相比值,如果类别相同则为1,否则为0。

结果提取

def format1(df2):
    df7=pd.Series()
    for i in range(180) :
        df3= df2[:-4*(1+i)]        
        model = ARIMA(df3, order=(1, 0, 4))
        result_arima1 = model.fit()
        df4 = df3.reset_index(drop=False)
        rows = df4.shape[0]
        endtime = df4.iloc[rows - 1, 0]
        forecast = pd.Series(result_arima1.forecast(5), index=pd.date_range(endtime, periods=5, freq='H'))
        df8 = forecast.tail(1)    
        df7 = pd.concat((df7,df8),axis=0,join='inner')
    return df7 
f2 =format1(df2)
f2.to_excel("forceful.xls")

结果分析

时间原因用的excel 分析

对比了6月21日~2022/7/15 高指真实值与预测值的结果,浓度预测准确率为84.61%,等级准确率40.74%,等级准确率偏低的原因为实际监测结果在6附近波动,为Ⅲ类水质标准。
预测对比时间窗口存在降雨,实际结果有一定波动,浓度预测准确率能到达84.6%,有一定的推广价值。

ARIMA .summary() 解读

  1. 左上 为模型基本信息,Dep. Variable(需要预测的变量)、Model(模型及其参数)、Date、Time、Sample(样本数据)、No. Observations(观测数据的数量)
  2. 右上 Log Likelihood(对数似然函数)标识最适合采样数据的分布。虽然它很有用,但AIC和BIC会惩罚模型的复杂性,这有助于使我们的ARIMA模型变得简洁。赤池的信息准则(AIC)有助于确定线性回归模型的强度。AIC 会惩罚添加参数的模型,因为添加更多参数将始终增加最大似然值。贝叶斯信息准则(BIC)与 AIC 一样,BIC 也会惩罚模型的复杂性,但它也包含数据中的行数。Hannan-Quinn信息标准(HQIC),与AIC和BIC一样是模型选择的另一个标准;但是它在实践中并不常用。AIC 、BIC 越小越好
  3. 中部 确保模型中的每个项在统计意义上是否显著。若p值大于0.05,则项不显著。
  4. 下部:Ljung-Box(modified Box-Pierce test)测试错误是白噪音 Ljung-Box (L1) (Q) 为Lag1的LBQ检验统计量,其Prob(Q)为 0.01,p值为0.94。由于p值高于0.05,因此我们不能拒绝零假设(误差是白噪音)

讨论与总结

  1. ARIMA 模型在高锰酸盐指数上的预测效果超过80%,经过初步研究,适用于水质在线站点。
  2. 模型可用于单站点单因子预测,不需要其他参数,约束小,预测精度高。
  3. 模型对波动剧烈的因子,预测效果不好,不适用于所有因子,所有站点。
  4. 对于新的数据集需要做平稳性检验,白噪声检验。
  5. 需要采用数据人工识图+自动的方式实现定阶,选择最优的 p,d,q。
  6. 可以继续在 ARIMAX(多元时间序列模型)等方面深入研究。

相关文章

{% post_link 'prophet时间序列模型水质预测应用' %}

标签:模型,ARIMA,差分,df2,水质,序列,import,root
From: https://www.cnblogs.com/chenwenmao/p/18212472

相关文章

  • 算法学习笔记——动态规划.最长上升/下降子序列 2024.5.24
    LanqiaoOJ 773这道题是一道动态规划的题目,主要考察最长上升/下降子序列,难度中等,但是对思维的考察比较重要,特别是如何解决其第二问题目描述某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统。但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以......
  • 数据子序列
    Description输入一个长度为n的整数序列(A1,A2,……An),从中找出一段连续的长度不超过m的子序列,使得这个子序列的和最大。Input有多组测试数据。对于每组测试数据的第一行,包含二个整数n和m,表示本组有n个测试数据,子序列长度为m,下一行为n个测试数据。Output对于每组测试数据,输出最......
  • P8765 [蓝桥杯 2021 国 AB] 翻转括号序列
    本文参考博客[蓝桥杯2021国AB]翻转括号序列(线段树上二分)一、问题简析线段树+二分初步分析令(的值为1,)的值为-1,则对于序列\(a_La_{L+1}a_{L+2}...a_R\),其为合法序列的条件为\[\begin{cases}\sum_{n=L}^R{a_n}=0\\\forall~k\in[L,R],\sum_{n=L}^k{a_n}\ge......
  • InfoTS: 具有信息感知增强的时间序列对比学习《Time Series Contrastive Learning wit
    现在是2024年5月23日,14:30,开始看论文。论文:TimeSeriesContrastiveLearningwithInformation-AwareAugmentations或者是:Timeseriescontrastivelearningwithinformation-awareaugmentationsGitHub:https://github.com/chengw07/InfoTSAAAI2023的论文。 摘要近年......
  • SCAU 19523 最长上升子序列长度
    19523 最长上升子序列长度时间限制:1000MS 代码长度限制:10KB题型:编程题   语言:不限定Description当元素ai1<ai2<……<aiK。就说这个序列是有序上升的。给定序列(a1,a2,……,aN),存在许多这样的子序列(ai1,ai2,……,aiK),其中1<=i1<i2<……<iK......
  • 优美子序列 题解
    有n个整数从左往右排成一行,构成一个序列a。如果通过删除原序列的若干个数(可以是删除0个),其他数保持位置不动,那么得到的序列就称为“子序列”。记sum表示序列a的所有数的总和,即sum=a[1]+a[2]+a[3]+...+a[n]。如果一个“子序列”的各个数加起来的和等于sum-1,那么这个“子序列”......
  • 类转词典 转json 序列化 魔术方法
      classTest:def__init__(self,totals,uniques,pre):self.totals=totalsself.uniques=uniquesself.cumulative_sum=totalsifpre:self.cumulative_sum+=pre.cumulative_sumpre=Test(......
  • NFLS NOI模拟 序列
    涉及知识点:分治、贪心前言没错……又是一道叫序列的题……题意有一个长为\(n\(\leq10^5)\)的序列\(a\),你可以花费\(x^2\)的代价将\(a_i\)变成\(a_i+x\),使得代价加上\(a\)两两数之差的绝对值乘以一个给定常数\(c\)的总和最小。思路拿到手觉得是一个贪心,但是直接......
  • DeepMTS深度学习神经网络多元时间序列预测宏观经济数据可视化|附数据代码
    原文链接:https://tecdat.cn/?p=36237原文出处:拓端数据部落公众号在数据科学领域,时间序列分析一直是一个至关重要的研究方向,尤其在金融、气象、医学以及许多其他科学和工业领域中,准确的时间序列预测对于制定策略、政策规划以及资源管理都具有极其重要的意义。随着技术的不断进步,......
  • 序列化和反序列化
    1.什么是序列化和反序列化?序列化:把对象转化为可传输的字节序列过程称为序列化。反序列化:把字节序列还原为对象的过程称为反序列化。2.序列化的目的? 序列化最终的目的是为了对象可以跨平台存储,和进行网络传输。而我们进行跨平台存储和网络传输的方式就是IO,而我们的......