ConvLSTM模型输出解析:预测未来温度数据
本文将探讨如何使用ConvLSTM模型预测未来5天的温度数据,并解释模型输出outputs的含义。/n/n在您的代码中,outputs是由layer_output_list和last_state_list组成的元组,其中layer_output_list包含了每一层LSTM的输出,而last_state_list包含了每一层LSTM最后一个时间步的隐藏状态和细胞状态。/n/n如果您只需要最后一个时间步的预测结果,可以使用outputs[0][:,-1,:,:,:],这样可以获取最后一层LSTM在最后一个时间步的输出。如果您需要所有时间步的预测结果,可以使用outputs[0],这样可以获取所有层LSTM的输出。不过需要注意的是,此时的输出维度为(batch_size, num_time_steps-6, hidden_dim, num_lon, num_dep),需要进行reshape才能和标签数据进行比较计算损失函数。/n/n以下是对修改后的代码的详细解释:/n/npython/nimport pandas as pd/nimport numpy as np/nimport torch/nimport torch.nn as nn/nimport torch.optim as optim/nfrom datetime import datetime/nfrom torch.utils.data import DataLoader/nfrom torch.utils.data import TensorDataset/n/ndevice = torch.device('cuda:0' if torch.cuda.is_available() else 'cpu')/n/nclass ConvLSTMCell(nn.Module):/n def __init__(self, input_dim, hidden_dim, kernel_size, bias):/n super(ConvLSTMCell, self).__init__()/n self.input_dim = input_dim/n self.hidden_dim = hidden_dim/n self.kernel_size = kernel_size/n self.padding = kernel_size[0] // 2, kernel_size[1] // 2/n self.bias = True/n self.conv = nn.Conv2d(in_channels=self.input_dim + self.hidden_dim,/n out_channels=4 * self.hidden_dim,/n kernel_size=self.kernel_size,/n padding=self.padding,/n bias=self.bias).cuda()/n/n def forward(self, input_tensor, cur_state):/n h_cur, c_cur = cur_state/n h_cur= h_cur.cuda()/n combined = torch.cat([input_tensor, h_cur], dim=1)/n combined_conv = self.conv(combined.float())/n cc_i, cc_f, cc_o, cc_g = torch.split(combined_conv, self.hidden_dim, dim=1)/n i = torch.sigmoid(cc_i)/n f = torch.sigmoid(cc_f)/n o = torch.sigmoid(cc_o)/n g = torch.tanh(cc_g)/n c_next = f * c_cur + i * g/n h_next = o * torch.tanh(c_next)/n return h_next, c_next/n/n def init_hidden(self, batch_size, image_size):/n height, width = image_size/n return (torch.zeros(batch_size, self.hidden_dim, height, width, device=self.conv.weight.device),/n torch.zeros(batch_size, self.hidden_dim, height, width, device=self.conv.weight.device))/n/nclass ConvLSTM(nn.Module):/n def __init__(self, input_dim,hidden_dim, kernel_size, num_layers,/n batch_first=False, bias=True, return_all_layers=False):/n super(ConvLSTM, self).__init__()/n self._check_kernel_size_consistency(kernel_size)/n/n kernel_size = self._extend_for_multilayer(kernel_size, num_layers)/n hidden_dim = self._extend_for_multilayer(hidden_dim, num_layers)/n/n if not len(kernel_size) == len(hidden_dim) == num_layers:/n raise ValueError('Inconsistent list length.')/n/n self.input_dim = input_dim/n self.hidden_dim = hidden_dim/n self.kernel_size = kernel_size/n self.num_layers = num_layers/n self.batch_first = batch_first/n self.bias = bias/n self.return_all_layers = return_all_layers/n/n cell_list = []/n for i in range(0, self.num_layers):/n cur_input_dim = self.input_dim if i == 0 else self.hidden_dim[i - 1]/n cell_list.append(ConvLSTMCell(input_dim=cur_input_dim,/n hidden_dim=self.hidden_dim[i],/n kernel_size=self.kernel_size[i],/n bias=self.bias))/n/n self.cell_list = nn.ModuleList(cell_list)/n/n def forward(self, input_tensor, hidden_state=None):/n if not self.batch_first:/n input_tensor = input_tensor.permute(1, 0, 2, 3, 4)/n/n b, t, _, h, w = input_tensor.size()/n/n if hidden_state is not None:/n raise NotImplementedError()/n else:/n hidden_state = self._init_hidden(batch_size=b,/n image_size=(h, w))/n/n layer_output_list = []/n last_state_list = []/n/n seq_len = input_tensor.size(1)/n cur_layer_input = input_tensor/n/n for layer_idx in range(self.num_layers):/n h, c = hidden_state[layer_idx]/n output_inner = []/n for t in range(seq_len):/n h, c = self.cell_list[layer_idx](input_tensor=cur_layer_input[:, t, :, :, :], cur_state=[h, c])/n output_inner.append(h)/n layer_output = torch.stack(output_inner, dim=1)/n cur_layer_input = layer_output/n layer_output_list.append(layer_output)/n last_state_list.append([h, c])/n/n if not self.return_all_layers:/n outputs = layer_output_list[-1][:,-1,:,:,:]/n else:/n outputs = layer_output_list/n outputs = torch.reshape(outputs, (b, -1, self.hidden_dim[-1], h, w))/n/n return outputs, last_state_list/n/n def _init_hidden(self, batch_size, image_size):/n init_states = []/n for i in range(self.num_layers):/n init_states.append(self.cell_list[i].init_hidden(batch_size, image_size))/n return init_states/n/n @staticmethod/n def _check_kernel_size_consistency(kernel_size):/n if not (isinstance(kernel_size, tuple) or/n (isinstance(kernel_size, list) and all([isinstance(elem, tuple) for elem in kernel_size]))):/n raise ValueError('`kernel_size` must be tuple or list of tuples')/n/n @staticmethod/n def _extend_for_multilayer(param, num_layers):/n if not isinstance(param, list):/n param = [param] * num_layers/n return param/n/n#实例化对象/nmodel = ConvLSTM(input_dim=1, hidden_dim=[64, 64], kernel_size=[(1, 55), (1, 55)], num_layers=2)/n#设置优化参数/ncriterion = nn.CrossEntropyLoss()/noptimizer = optim.SGD(list(model.parameters()), lr=0.001, momentum=0.9)/n# 读取Excel文件/ndf = pd.read_excel(r'C:/Users/19738/Desktop/数据集/01.xlsx')/ndf = df.sort_values(by=['日期', '经度', '深度'])/n# 将数据转换为numpy数组/ndata = df.values/ntime = df.iloc[:, 0] # 提取时间列/nlongitude = df.iloc[:, 1] # 提取经度列/ndepth = df.iloc[:, 2] # 提取深度列/n# 假设您的数据张量大小为 (num_samples,num_time_steps,1, num_lon, num_dep, )/nnum_samples = 100/nnum_lon = 2/nnum_dep = 55/nnum_time_steps = 11/n# 构建新的数据张量,初始化为 0/ntemp_data = np.zeros((num_samples, num_time_steps, 1, num_lon, num_dep))/nfor i in range(num_samples):/n for j in range(num_lon):/n for k in range(num_dep):/n for t in range(num_time_steps):/n date_str = time[t].strftime('%Y/%m/%d')/n lon_str = str(longitude[j])/n dep_str = str(depth[k])/n index = (df['日期'] == date_str) & (df['经度'] == lon_str) & (df['深度'] == dep_str)/n temp_value = df.loc[index, '温度'].values[0]/n temp_data[i, t, 0,j, k] = temp_value/ntemp_data_tensor = torch.from_numpy(temp_data)/n# 定义训练数据和标签/ntrain_data = temp_data_tensor[:, :6, :, :, :]/ntrain_label = temp_data_tensor[:, 6:, :, :, :]/nprint(train_data.shape)/nprint(train_label.shape)/ntrain_dataset = TensorDataset(train_data, train_label)/ntrain_loader = DataLoader(train_dataset, batch_size=100, shuffle=True)/n# 定义训练循环/nfor epoch in range(10):/n for i, (inputs, labels) in enumerate(train_loader):/n inputs, labels = inputs.to(device), labels.to(device)/n model.to(device)/n optimizer.zero_grad()/n outputs, _ = model(inputs)/n print(inputs.shape)/n loss = criterion(outputs, labels)/n loss.backward()/n optimizer.step()/n print(f'Epoch {epoch+1}, Batch {i+1}, Loss: {loss.item()}')/n/n/n代码修改说明:/n/n1. 在ConvLSTM的forward函数中,添加了对输出的处理,根据return_all_layers参数决定是否返回所有层LSTM的输出。/n2. 如果return_all_layers为False,则仅返回最后一层LSTM的最后一个时间步的输出,并使用outputs[0][:,-1,:,:,:]获取该输出。/n3. 如果return_all_layers为True,则返回所有层LSTM的输出,并使用torch.reshape将输出重塑为(batch_size, -1, self.hidden_dim[-1], h, w)的形状。/n/n通过以上修改,您可以根据自己的需求选择输出结果,并使用正确的形状计算损失函数。/n/n总结:/n/nConvLSTM模型的输出包含了每一层LSTM的输出和最后一个时间步的隐藏状态和细胞状态。您可以根据需要选择使用outputs[0][:,-1,:,:,:]获取最后一个时间步的预测结果,或者使用outputs[0]获取所有时间步的预测结果。/n/n希望以上解释能够帮助您理解ConvLSTM模型的输出含义,并顺利进行温度预测任务。
原文地址: http://www.cveoy.top/t/topic/mWjc 著作权归作者所有。请勿转载和采集!