当前位置:首页 » 《随便一记》 » 正文

Java Deeplearning4j:基础大纲详细整理

10 人参与  2024年10月06日 08:01  分类 : 《随便一记》  评论

点击全文阅读


? 博主简介:历代文学网(PC端可以访问:https://literature.sinhy.com/#/literature?__c=1000,移动端可微信小程序搜索“历代文学”)总架构师,15年工作经验,精通Java编程高并发设计Springboot和微服务,熟悉LinuxESXI虚拟化以及云原生Docker和K8s,热衷于探索科技的边界,并将理论知识转化为实际应用。保持对新技术的好奇心,乐于分享所学,希望通过我的实践经历和见解,启发他人的创新思维。在这里,我希望能与志同道合的朋友交流探讨,共同进步,一起在技术的世界里不断学习成长。

在这里插入图片描述

在这里插入图片描述

Java Deeplearning4j:基础大纲详细整理

DeepLearning4J(DL4J)是一个专为 JavaScala 环境设计的开源深度学习框架,旨在帮助开发者构建、训练和部署深度学习模型

DL4J 不仅提供了丰富的功能和工具,还与 HadoopSpark 等大数据技术无缝集成,适用于企业级应用。其强大的跨平台支持使得开发者能够在 WindowsLinuxmacOS 等多种操作系统上运行深度学习任务

DL4J 的核心优势之一是其高性能。它使用 ND4J一个用于科学计算的 Java 库)作为底层计算引擎,支持多线程和 GPU 加速,从而显著提高计算效率。此外,DL4J 还支持分布式训练,可以在多台机器上并行处理大规模数据集,适用于处理海量数据的企业级应用场景。

DL4J 提供了丰富的深度学习算法和模型,包括卷积神经网络(CNN)、循环神经网络(RNN)、长短期记忆网络(LSTM)和门控循环单元(GRU)等。开发者可以轻松定义和配置这些模型,并通过简单的 API 进行训练和评估。DL4J 还支持迁移学习,允许开发者利用预训练模型进行新任务的训练,从而减少训练时间和计算资源的需求。

在数据处理方面,DL4J 集成了 DataVec,一个用于数据预处理和转换的库。DataVec 支持多种数据格式和数据源,包括 CSV图像文本音频等,使得数据加载和预处理变得简单高效。DL4J 还提供了丰富的评估指标和工具,帮助开发者监控和分析模型的性能。

DL4J 的社区支持也非常活跃,提供了大量的文档、教程和示例代码,帮助开发者快速上手。此外,DL4J 还与多个深度学习框架和工具集成,如 TensorFlowKerasONNX,使得开发者可以灵活选择和切换不同的工具和模型。

总之,DeepLearning4J 是一个功能强大、易于使用且高度集成的深度学习框架,适用于 Java 和 Scala 开发者。无论是在学术研究还是企业应用中,DL4J 都能帮助开发者高效地构建和部署深度学习模型,解决复杂的数据分析和预测问题。

本大纲涵盖了DeepLearning4J环境配置基础操作模型构建训练与评估高级应用等每个角落。

1. 环境配置与基础入门

1.1 环境配置

在开始学习 DeepLearning4J 之前,首先需要配置开发环境。以下是配置环境的步骤:

1.1.1 安装 Java 和 Maven

DeepLearning4J 是基于 Java 的框架,因此需要安装 Java 开发工具包(JDK)。推荐使用 JDK 8 或更高版本。安装完成后,配置环境变量 JAVA_HOMEPATH

Maven 是一个项目管理工具,用于管理项目的依赖和构建过程。安装 Maven 后,配置环境变量 MAVEN_HOMEPATH

1.1.2 配置 DeepLearning4J 项目依赖

创建一个新的 Maven 项目,并在 pom.xml 文件中添加 DeepLearning4J 的依赖项。以下是一个示例 pom.xml 文件:

<dependencies>    <dependency>        <groupId>org.deeplearning4j</groupId>        <artifactId>deeplearning4j-core</artifactId>        <version>1.0.0-beta7</version>    </dependency>    <dependency>        <groupId>org.nd4j</groupId>        <artifactId>nd4j-native-platform</artifactId>        <version>1.0.0-beta7</version>    </dependency></dependencies>
1.1.3 使用 IntelliJ IDEA 或 Eclipse 创建 DL4J 项目

IntelliJ IDEAEclipse 是常用的 Java 集成开发环境(IDE)。使用这些 IDE 可以更方便地管理和开发 DL4J 项目。创建一个新的 Maven 项目,并导入之前配置的 pom.xml 文件。

1.2 基础操作

在配置好开发环境后,开始学习 DeepLearning4J 的基础操作。

1.2.1 创建和操作 NDArray

NDArrayDeepLearning4J 中的核心数据结构,类似于 NumPy 中的 ndarray。学习如何创建不同形状和数据类型的 NDArray,并进行基本的数学运算和操作。

1.2.2 数据加载与预处理

数据加载和预处理是深度学习模型训练的重要步骤。学习如何使用 DataVec 库加载和预处理不同类型的数据,如 CSV、图像和文本数据。

1.2.3 简单线性回归模型

通过构建一个简单的线性回归模型,学习如何使用 DeepLearning4J 构建、训练和评估模型。理解模型的基本组件,如层、损失函数和优化器。


2. 模型构建与训练

2.1 构建多层感知器(MLP)

在这里插入图片描述

多层感知器(MLP)是最简单的神经网络模型之一。学习如何使用 DeepLearning4J 构建和训练一个 MLP 模型,用于分类或回归任务。

2.1.1 定义模型结构

学习如何定义 多层感知器 (MLP) 的层结构,包括输入层隐藏层输出层。理解不同类型的层(如 DenseLayerOutputLayer)及其参数。

2.1.1.1 多层感知器 (MLP) 的层结构基本概念
输入层:接收输入数据的层。隐藏层:位于输入层和输出层之间的层,用于提取特征。输出层:生成最终输出的层,通常用于分类或回归任务。
2.1.1.2 定义 MLP 层结构

在 DeepLearning4J (DL4J) 中,多层感知器 (MLP) 是一种常见的前馈神经网络,通常用于分类和回归任务。MLP 由输入层、一个或多个隐藏层和输出层组成。每层可以是全连接层(DenseLayer)或输出层(OutputLayer)。

2.1.1.3 导入必要的库

首先,确保你已经导入了 DL4J 的核心库。

import org.deeplearning4j.nn.conf.NeuralNetConfiguration;import org.deeplearning4j.nn.conf.layers.DenseLayer;import org.deeplearning4j.nn.conf.layers.OutputLayer;import org.deeplearning4j.nn.multilayer.MultiLayerNetwork;import org.deeplearning4j.nn.weights.WeightInit;import org.nd4j.linalg.activations.Activation;import org.nd4j.linalg.lossfunctions.LossFunctions;
2.1.1.4 配置 MLP 层结构

使用 NeuralNetConfiguration.Builder 来配置 MLP 的层结构。

// 定义 MLP 的层结构NeuralNetConfiguration.Builder builder = new NeuralNetConfiguration.Builder()    .seed(123)  // 随机种子,确保结果可重复    .updater(new Adam())  // 使用 Adam 优化器    .list()  // 开始定义层    .layer(0, new DenseLayer.Builder()  // 第一个隐藏层        .nIn(784)  // 输入层的节点数(例如,MNIST 数据集的 28x28 图像)        .nOut(100)  // 输出层的节点数        .activation(Activation.RELU)  // 激活函数        .weightInit(WeightInit.XAVIER)  // 权重初始化方法        .build())    .layer(1, new DenseLayer.Builder()  // 第二个隐藏层        .nIn(100)  // 输入层的节点数        .nOut(50)  // 输出层的节点数        .activation(Activation.RELU)  // 激活函数        .weightInit(WeightInit.XAVIER)  // 权重初始化方法        .build())    .layer(2, new OutputLayer.Builder(LossFunctions.LossFunction.NEGATIVELOGLIKELIHOOD)  // 输出层        .nIn(50)  // 输入层的节点数        .nOut(10)  // 输出层的节点数(例如,MNIST 数据集的 10 个类别)        .activation(Activation.SOFTMAX)  // 激活函数        .weightInit(WeightInit.XAVIER)  // 权重初始化方法        .build())    .backprop(true)  // 启用反向传播    .pretrain(false);  // 不进行预训练
2.1.1.4 构建和初始化 MLP

使用 MultiLayerNetwork 来构建和初始化 MLP。

// 构建和初始化 MLPMultiLayerNetwork model = new MultiLayerNetwork(builder.build());model.init();
2.1.1.5 不同类型的层及其参数

全连接层DenseLayer

DenseLayer 是全连接层,每个输入节点都连接到每个输出节点。

nIn:输入层的节点数。nOut:输出层的节点数。activation:激活函数,如 Activation.RELUActivation.TANHActivation.SIGMOID 等。weightInit:权重初始化方法,如 WeightInit.XAVIERWeightInit.UNIFORM 等。

输出层OutputLayer

OutputLayer 是输出层,通常用于分类或回归任务。

nIn:输入层的节点数。nOut:输出层的节点数。activation:激活函数,如 Activation.SOFTMAXActivation.IDENTITY 等。weightInit:权重初始化方法,如 WeightInit.XAVIERWeightInit.UNIFORM 等。LossFunction:损失函数,如 LossFunctions.LossFunction.NEGATIVELOGLIKELIHOODLossFunctions.LossFunction.MSE 等。
2.1.2 配置训练参数

配置训练过程中的参数,如学习率、批次大小、迭代次数等。理解这些参数对模型训练的影响。

2.1.3 训练和评估模型

使用训练数据训练模型,并使用测试数据评估模型的性能。学习如何计算模型的准确率、损失值等指标。


2.2 构建卷积神经网络(CNN)

在这里插入图片描述

卷积神经网络(CNN)是处理图像数据的常用模型。学习如何使用 DeepLearning4J 构建和训练一个 CNN 模型。

2.2.1 定义模型结构

DeepLearning4J (DL4J) 中,卷积神经网络 (CNN) 是一种常用的深度学习模型,特别适用于图像处理任务。CNN 由多个层组成,包括卷积层、池化层和全连接层。每层都有不同的参数,如卷积核、步幅和填充。

学习如何定义 CNN 的层结构,包括卷积层池化层全连接层。理解卷积核步幅填充等参数的作用。

基本概念:

卷积层:用于提取图像的特征,通过卷积核在输入图像上滑动并计算特征图。池化层:用于减少特征图的尺寸,通常使用最大池化或平均池化。全连接层:将卷积层或池化层的输出展平并连接到输出层,通常用于分类或回归任务。
2.2.1.2 导入必要的库

首先,确保你已经导入了 DL4J 的核心库。

import org.deeplearning4j.nn.conf.NeuralNetConfiguration;import org.deeplearning4j.nn.conf.layers.ConvolutionLayer;import org.deeplearning4j.nn.conf.layers.SubsamplingLayer;import org.deeplearning4j.nn.conf.layers.DenseLayer;import org.deeplearning4j.nn.conf.layers.OutputLayer;import org.deeplearning4j.nn.multilayer.MultiLayerNetwork;import org.deeplearning4j.nn.weights.WeightInit;import org.nd4j.linalg.activations.Activation;import org.nd4j.linalg.lossfunctions.LossFunctions;
2.2.1.3 配置 CNN 层结构

使用 NeuralNetConfiguration.Builder 来配置 CNN 的层结构。

// 定义 CNN 的层结构NeuralNetConfiguration.Builder builder = new NeuralNetConfiguration.Builder()    .seed(123)  // 随机种子,确保结果可重复    .updater(new Adam())  // 使用 Adam 优化器    .list()  // 开始定义层    .layer(0, new ConvolutionLayer.Builder(5, 5)  // 第一个卷积层        .nIn(1)  // 输入通道数(例如,MNIST 数据集的灰度图像)        .nOut(20)  // 输出通道数        .stride(1, 1)  // 步幅        .padding(2, 2)  // 填充        .activation(Activation.RELU)  // 激活函数        .weightInit(WeightInit.XAVIER)  // 权重初始化方法        .build())    .layer(1, new SubsamplingLayer.Builder(SubsamplingLayer.PoolingType.MAX)  // 第一个池化层        .kernelSize(2, 2)  // 池化窗口大小        .stride(2, 2)  // 步幅        .build())    .layer(2, new ConvolutionLayer.Builder(5, 5)  // 第二个卷积层        .nIn(20)  // 输入通道数        .nOut(50)  // 输出通道数        .stride(1, 1)  // 步幅        .padding(2, 2)  // 填充        .activation(Activation.RELU)  // 激活函数        .weightInit(WeightInit.XAVIER)  // 权重初始化方法        .build())    .layer(3, new SubsamplingLayer.Builder(SubsamplingLayer.PoolingType.MAX)  // 第二个池化层        .kernelSize(2, 2)  // 池化窗口大小        .stride(2, 2)  // 步幅        .build())    .layer(4, new DenseLayer.Builder()  // 全连接层        .nIn(800)  // 输入节点数(计算方式:(28 + 2*2 - 5 + 1) / 2 * (28 + 2*2 - 5 + 1) / 2 * 50)        .nOut(500)  // 输出节点数        .activation(Activation.RELU)  // 激活函数        .weightInit(WeightInit.XAVIER)  // 权重初始化方法        .build())    .layer(5, new OutputLayer.Builder(LossFunctions.LossFunction.NEGATIVELOGLIKELIHOOD)  // 输出层        .nIn(500)  // 输入节点数        .nOut(10)  // 输出节点数(例如,MNIST 数据集的 10 个类别)        .activation(Activation.SOFTMAX)  // 激活函数        .weightInit(WeightInit.XAVIER)  // 权重初始化方法        .build())    .backprop(true)  // 启用反向传播    .pretrain(false);  // 不进行预训练
2.2.1.4 构建和初始化 CNN

使用 MultiLayerNetwork 来构建和初始化 CNN。

// 构建和初始化 CNNMultiLayerNetwork model = new MultiLayerNetwork(builder.build());model.init();
2.2.1.5 不同类型的层及其参数

ConvolutionLayer 是卷积层,用于提取图像的特征:

kernelSize:卷积核的大小,例如 5, 5nIn:输入通道数。nOut:输出通道数。stride:卷积核的步幅,例如 1, 1padding:填充大小,例如 2, 2activation:激活函数,如 Activation.RELUActivation.TANH 等。weightInit:权重初始化方法,如 WeightInit.XAVIERWeightInit.UNIFORM 等。

SubsamplingLayer 是池化层,用于减少特征图的尺寸:

PoolingType:池化类型,如 SubsamplingLayer.PoolingType.MAX(最大池化)或 SubsamplingLayer.PoolingType.AVG(平均池化)。kernelSize:池化窗口的大小,例如 2, 2stride:池化窗口的步幅,例如 2, 2

DenseLayer 是全连接层,将卷积层或池化层的输出展平并连接到输出层:

nIn:输入节点数。nOut:输出节点数。activation:激活函数,如 Activation.RELUActivation.TANH 等。weightInit:权重初始化方法,如 WeightInit.XAVIERWeightInit.UNIFORM 等。

OutputLayer 是输出层,通常用于分类或回归任务:

nIn:输入节点数。nOut:输出节点数。activation:激活函数,如 Activation.SOFTMAXActivation.IDENTITY 等。weightInit:权重初始化方法,如 WeightInit.XAVIERWeightInit.UNIFORM 等。LossFunction:损失函数,如 LossFunctions.LossFunction.NEGATIVELOGLIKELIHOODLossFunctions.LossFunction.MSE 等。
2.2.2 数据预处理

学习如何对图像数据进行预处理,如归一化、缩放、数据增强等。理解这些预处理步骤对模型性能的影响。

2.2.3 训练和评估模型

使用训练数据训练模型,并使用测试数据评估模型的性能。学习如何计算模型的准确率、损失值等指标。

2.3 构建循环神经网络(RNN)

在这里插入图片描述

循环神经网络(RNN)是处理序列数据的常用模型。学习如何使用 DeepLearning4J 构建和训练一个 RNN 模型。

2.3.1 定义模型结构

DeepLearning4J (DL4J) 中,循环神经网络 (RNN) 是一种用于处理序列数据的深度学习模型。RNN 由多个层组成,包括输入层隐藏层输出层。常见的 RNN 单元包括长短期记忆网络 (LSTM) 和门控循环单元 (GRU)。

我们需要学习如何定义 RNN 的层结构,包括输入层隐藏层输出层。理解 LSTMGRU 等不同类型的 RNN 单元。

2.3.1.1 定义 RNN 层结构
输入层:接收序列数据的层。隐藏层:位于输入层和输出层之间的层,用于处理序列数据并提取特征。输出层:生成最终输出的层,通常用于分类或回归任务。
2.3.1.2 导入必要的库

首先,确保你已经导入了 DL4J 的核心库。

import org.deeplearning4j.nn.conf.NeuralNetConfiguration;import org.deeplearning4j.nn.conf.layers.LSTM;import org.deeplearning4j.nn.conf.layers.RnnOutputLayer;import org.deeplearning4j.nn.multilayer.MultiLayerNetwork;import org.deeplearning4j.nn.weights.WeightInit;import org.nd4j.linalg.activations.Activation;import org.nd4j.linalg.lossfunctions.LossFunctions;
2.3.1.3 配置 RNN 层结构

使用 NeuralNetConfiguration.Builder 来配置 RNN 的层结构。

// 定义 RNN 的层结构NeuralNetConfiguration.Builder builder = new NeuralNetConfiguration.Builder()    .seed(123)  // 随机种子,确保结果可重复    .updater(new Adam())  // 使用 Adam 优化器    .list()  // 开始定义层    .layer(0, new LSTM.Builder()  // 第一个 LSTM 层        .nIn(10)  // 输入层的节点数(例如,序列的特征维度)        .nOut(100)  // 输出层的节点数        .activation(Activation.TANH)  // 激活函数        .weightInit(WeightInit.XAVIER)  // 权重初始化方法        .build())    .layer(1, new LSTM.Builder()  // 第二个 LSTM 层        .nIn(100)  // 输入层的节点数        .nOut(100)  // 输出层的节点数        .activation(Activation.TANH)  // 激活函数        .weightInit(WeightInit.XAVIER)  // 权重初始化方法        .build())    .layer(2, new RnnOutputLayer.Builder(LossFunctions.LossFunction.MCXENT)  // 输出层        .nIn(100)  // 输入层的节点数        .nOut(10)  // 输出层的节点数(例如,分类任务的类别数)        .activation(Activation.SOFTMAX)  // 激活函数        .weightInit(WeightInit.XAVIER)  // 权重初始化方法        .build())    .backprop(true)  // 启用反向传播    .pretrain(false);  // 不进行预训练
2.3.1.4 构建和初始化 RNN

使用 MultiLayerNetwork 来构建和初始化 RNN。

// 构建和初始化 RNNMultiLayerNetwork model = new MultiLayerNetwork(builder.build());model.init();
2.3.1.5 不同类型的 RNN 单元

LSTM 是一种特殊的 RNN 单元,具有长短期记忆能力,能够处理长序列数据。

nIn:输入层的节点数。nOut:输出层的节点数。activation:激活函数,如 Activation.TANHActivation.SIGMOID 等。weightInit:权重初始化方法,如 WeightInit.XAVIERWeightInit.UNIFORM 等。

GRU 是一种简化的 LSTM 单元,具有更少的参数和更快的训练速度。

nIn:输入层的节点数。nOut:输出层的节点数。activation:激活函数,如 Activation.TANHActivation.SIGMOID 等。weightInit:权重初始化方法,如 WeightInit.XAVIERWeightInit.UNIFORM 等。

RnnOutputLayer 是 RNN 的输出层,通常用于分类或回归任务。

nIn:输入层的节点数。nOut:输出层的节点数。activation:激活函数,如 Activation.SOFTMAXActivation.IDENTITY 等。weightInit:权重初始化方法,如 WeightInit.XAVIERWeightInit.UNIFORM 等。LossFunction:损失函数,如 LossFunctions.LossFunction.MCXENT(多分类交叉熵)、LossFunctions.LossFunction.MSE(均方误差)等。
2.3.2 数据预处理

学习如何对序列数据进行预处理,如分词、向量化、填充等。理解这些预处理步骤对模型性能的影响。

2.3.3 训练和评估模型

使用训练数据训练模型,并使用测试数据评估模型的性能。学习如何计算模型的准确率、损失值等指标。

3. 高级应用

3.1 迁移学习

迁移学习是一种利用预训练模型进行新任务训练的技术。通过迁移学习,我们可以利用在大规模数据集上预训练的模型,并将其应用于新的、可能较小的数据集上。这种方法可以显著减少训练时间和计算资源的需求。

我们需要深入学习如何使用 DeepLearning4J 进行迁移学习。
在这里插入图片描述

3.1.1 加载预训练模型

学习如何加载预训练的模型,并使用其作为新模型的基础。理解如何冻结和解冻模型的某些层。

3.1.2 微调模型

学习如何对预训练模型进行微调,以适应新任务。理解微调过程中的参数设置和训练策略。

3.1.3 评估迁移学习效果

使用测试数据评估迁移学习模型的性能。学习如何比较迁移学习模型和从头训练模型的效果。

3.2 自定义层和损失函数

在某些情况下,标准层和损失函数可能无法满足需求。学习如何使用 DeepLearning4J 自定义层和损失函数。

3.2.1 自定义层

学习如何创建自定义层,并将其集成到模型中。理解层的输入、输出和参数定义。

3.2.2 自定义损失函数

学习如何创建自定义损失函数,并将其用于模型训练。理解损失函数的计算逻辑和梯度传播。

3.2.3 训练和评估自定义模型

使用自定义层和损失函数训练模型,并评估其性能。学习如何调试和优化自定义模型。

3.3 模型部署

模型训练完成后,需要将其部署到生产环境中。学习如何使用 DeepLearning4J 部署模型。

3.3.1 导出模型

学习如何将训练好的模型导出为可部署的格式,如 PMMLONNX。理解不同格式的优缺点。

3.3.2 加载和使用模型

学习如何在生产环境中加载和使用导出的模型。理解如何处理输入数据和获取预测结果。

3.3.3 监控和维护模型

学习如何监控和维护部署的模型,以确保其性能和稳定性。理解如何处理模型漂移和更新模型。

4. 实战案例

4.1 图像分类

通过一个实际的图像分类任务,综合应用前面所学的知识,使用 DeepLearning4J 构建、训练和评估图像分类模型。

4.1.1 数据集准备

准备图像分类任务的数据集,并进行数据预处理。理解数据集的划分和数据增强技术。

4.1.2 模型构建

构建一个适合图像分类任务的 CNN 模型,并配置训练参数。理解不同层和参数的选择。

4.1.3 模型训练和评估

使用训练数据训练模型,并使用测试数据评估模型的性能。学习如何调整模型和参数以提高性能。

4.2 文本分类

通过一个实际的文本分类任务,综合应用前面所学的知识,使用 DeepLearning4J 构建、训练和评估文本分类模型。

4.2.1 数据集准备

准备文本分类任务的数据集,并进行数据预处理。理解文本数据的向量化和填充技术。

4.2.2 模型构建

构建一个适合文本分类任务的 RNN 模型,并配置训练参数。理解不同层和参数的选择。

4.2.3 模型训练和评估

使用训练数据训练模型,并使用测试数据评估模型的性能。学习如何调整模型和参数以提高性能。

4.3 时间序列预测

通过一个实际的时间序列预测任务,综合应用前面所学的知识,使用 DeepLearning4J 构建、训练和评估时间序列预测模型。

4.3.1 数据集准备

准备时间序列预测任务的数据集,并进行数据预处理。理解时间序列数据的特征提取和归一化技术。

4.3.2 模型构建

构建一个适合时间序列预测任务的 RNN 模型,并配置训练参数。理解不同层和参数的选择。

4.3.3 模型训练和评估

使用训练数据训练模型,并使用测试数据评估模型的性能。学习如何调整模型和参数以提高性能。

5. 总结与展望

通过本大纲的学习,您已经掌握了 DeepLearning4J 从基础到高级的使用方法:

环境配置:成功配置了 Java 和 Maven 环境,并创建了 DL4J 项目。基础操作:掌握了 NDArray 的操作、数据加载与预处理、简单线性回归模型。模型构建与训练:学习了 MLPCNNRNN 模型的构建、训练和评估方法。高级应用:掌握了迁移学习、自定义层和损失函数、模型部署等高级技术。实战案例:通过实际任务,综合应用所学知识,构建、训练和评估了图像分类、文本分类和时间序列预测模型。

5.1 展望

深入研究:深入研究 DL4J 的高级功能,如分布式训练、模型优化等。实际应用:将所学知识应用于实际项目,解决实际问题。社区贡献:参与 DL4J 社区,贡献代码和文档,帮助他人学习和使用 DL4J

通过不断学习和实践,您将能够成为一名熟练的 DeepLearning4J 开发者,并在实际项目中应用深度学习技术。


点击全文阅读


本文链接:http://m.zhangshiyu.com/post/168226.html

<< 上一篇 下一篇 >>

  • 评论(0)
  • 赞助本站

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。

关于我们 | 我要投稿 | 免责申明

Copyright © 2020-2022 ZhangShiYu.com Rights Reserved.豫ICP备2022013469号-1