对神经网络的理解,以及代码的手动实现,以鸾尾花数据集为例

本文主要是介绍对神经网络的理解,以及代码的手动实现,以鸾尾花数据集为例,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

“”"

这是一个神经网络的示意图。
在这里插入图片描述
大家看到这里可能会感到非常的困惑,什么输入层,输出层,隐藏层,权重,阈值什么的到底是什么。接下来我为大家举个例子,去理解它。
这里我们从线性回归开始一步步引入神经网络,因为我们可以把线性回归看成是一个单层的神经网络,它的结构就应该是这样:
在这里插入图片描述

在线性回归中,我们尝试使用一条y_pred=wx+b的直线来拟合真实值y_true。使用梯度下降来一步步逼近他的真实的结果,也就是求出与真实值最接近的[w,b]来拟合最终结果,如下图所示
在这里插入图片描述
但是当我们数据之间的关系是非线性的时候,例如这样:
在这里插入图片描述
我们就无法通过一条直线来拟合它,这时候,我们可以尝试为他的结构中加入隐藏层,例如这样:

在这里插入图片描述
通过每一个隐藏层神经元来模拟它的每一条直线,第一个隐藏层学到单方向的直线,第二个隐藏层将两条线拼接,以此类推直到最后都拼接起来。这样我们就能完全模拟它的规律了,也就是能拟合出一个非线性的y。

从上边我们就可以得到这样一个结论:在神经网络中,当数据线性不可分的时候才需要隐藏层。

那么问题来了,隐藏层的数量和每个隐藏层中神经元的数量应该为多少呢?·········这里我们可以把它理解为一个可以调整的超参数。

想要确定神经网络中隐藏层的数量和每个隐藏层中神经元的数量是一个关键的问题,对于神经网络的性能和训练效果有重要影响。这个问题通常没有一个通用的答案,取决于具体的任务和数据。这里是一些我的建议。
简单任务: 对于相对简单的任务,例如基本的图像分类或回归问题,可以尝试只使用一个隐藏层。
复杂任务: 对于更复杂的任务,如图像识别、自然语言处理等,可能需要深层网络,即多个隐藏层。深度神经网络通常能够学习到更复杂的特征和表示。
避免过拟合: 大型神经网络可能会导致过拟合,尤其是在训练数据有限的情况下。可以使用正则化技术、丢弃(Dropout)等方法来防止过拟合。
尝试不同的配置: 尝试不同的隐藏层神经元数量的配置,使用交叉验证来评估模型性能,选择效果最好的配置。

接下来我们尝试代码实现。

当把神经网络理解为,加入隐藏层的线性回归,我们在梯度下降的时候,就可以使用以下四个步骤

这里提到了线性回归,对线性回归的理解我们可以参考我的上一篇文章:
想要实现神经网络应该从以下四个步骤开始
1.先随机初始化一个权重和阈值

#初始化权重,以及阈值
weight_input_hidden=np.random.uniform(size=(input_layer_number,hidden_layer_number))
bias_hidden=np.random.uniform(size=(1,hidden_layer_number))
weight_hidden_output=np.random.uniform(size=(hidden_layer_number,output_layer_number))
bias_output=np.random.uniform(size=(1,output_layer_number))

开始梯度下降:

2.前向传播
1.输入层到隐藏层:
定义变量 hidden_layer_input,含义为隐藏的输入,对输入层输入的数据乘以输入层到隐藏层的权重再加上隐藏层阈值bias,这里实际上就是使用y=wx+b来拟合最终结果

 hidden_layer_input=np.dot(X_train,weight_input_hidden)+bias_hidden

再用激活函数sigmod()把数据映射到(0,1)区间上
这里hidden_layer_output为隐藏层的输出

  hidden_layer_output=sigmoid(hidden_layer_input)
2.隐藏层到输出层:对输入层映射过来的数据乘以权重w再加上输出层的阈值bias然后把它通过激活函数sigmod()映射到(0,1)区间上
  	output_layer_input=np.dot(hidden_layer_output,weight_hidden_output)+bias_outputoutput_layer_output=sigmoid(output_layer_input)

3。计算误差
使用目标值y减去输出层的输出,得到误差

error=y_train-output_layer_output

4.反向传播
(1)计算输出层的误差对结果的贡献程度,用于调整隐藏层到输出层的权重,d_output表示输出层每个单元的局部梯度
(2) 定义变量error_hidden 表示隐藏层整体的误差,用于计算隐藏层的权重的梯度,
(3)用于调整输入层和隐藏层的权重,d_hidden 表示隐藏层中每个单元的局部梯度

	d_output=error*derivative_sigmoid(output_layer_output)error_hidden=d_output.dot(weight_hidden_output.T)d_hidden=error_hidden*derivative_sigmoid(hidden_layer_output)

5.更新权重和偏置
1.隐藏层到输出层的权重=原始权重+学习率a*(隐藏层的输出乘输出层每个单元的局部梯度)
主要是为了更新了连接隐藏层和输出层的权重矩阵,根据隐藏层的输出、输出层的局部梯度和学习率来调整权重,以减小输出层的误差。
2.输出层的阈值就等于原始阈值+学习率a*(输出层局部梯度之和)
主要是为了更新了输出层的偏差项,根据输出层的局部梯度和学习率来调整偏差项,以减小输出层的误差
3.输入层到隐藏层的权重=原始权重+学习率a*(输入样本乘隐藏层每个单元的局部梯度)
主要是为了更新了连接输入层和隐藏层的权重矩阵,根据输入数据、隐藏层的局部梯度和学习率来调整权重,以减小隐藏层的误差
4.隐藏层的阈值就等于原始阈值+学习率a*(隐藏层局部梯度之和)
主要是为了更新隐藏层的偏差项,根据输出层的局部梯度和学习率调整偏差,以减小隐藏层的误差。

	weight_hidden_output+=learning_rate*np.dot(hidden_layer_output.T,d_output)bias_output+=np.sum(d_output,axis=0,keepdims=True)*learning_rateweight_input_hidden+=learning_rate*np.dot(X_train.T,d_hidden)bias_hidden+=np.sum(d_hidden,axis=0,keepdims=True)*learning_rate

以下是完整代码,写的不好没有使用函数式编程的写法,但是便于理解一点,使用我上面的步骤可以很方便的实现它:

import numpy as np
from sklearn import datasets
from sklearn.model_selection import train_test_split
# 导入鸾尾花数据集
iris=datasets.load_iris()
X=iris.data
print(X)
y=iris.target# 使用one-hot对类别进行独热编码
num_class=len(np.unique(y))
y_one_hot=np.eye(num_class)[y]#划分数据
X_train,X_test,y_train,y_test=train_test_split(X,y_one_hot,test_size=0.2,random_state=1)# 定义激活函数
def sigmoid(x):return 1/(1+np.exp(-x))#定义激活函数的导数
def derivative_sigmoid(x):return x*(1-x)#定义输入层,隐藏层,以及输出层的个数
input_layer_number=4
hidden_layer_number=8
output_layer_number=num_class#初始化权重,以及阈值
weight_input_hidden=np.random.uniform(size=(input_layer_number,hidden_layer_number))
bias_hidden=np.random.uniform(size=(1,hidden_layer_number))
weight_hidden_output=np.random.uniform(size=(hidden_layer_number,output_layer_number))
bias_output=np.random.uniform(size=(1,output_layer_number))#定义超参数
learning_rate=0.001
epochs=100001
for epoch in range(1,epochs):#前向传播hidden_layer_input=np.dot(X_train,weight_input_hidden)+bias_hiddenhidden_layer_output=sigmoid(hidden_layer_input)output_layer_input=np.dot(hidden_layer_output,weight_hidden_output)+bias_outputoutput_layer_output=sigmoid(output_layer_input)#计算误差error=y_train-output_layer_output# 反向传播d_output=error*derivative_sigmoid(output_layer_output)error_hidden=d_output.dot(weight_hidden_output.T)d_hidden=error_hidden*derivative_sigmoid(hidden_layer_output)# 更新权重和偏置weight_hidden_output+=learning_rate*np.dot(hidden_layer_output.T,d_output)bias_output+=np.sum(d_output,axis=0,keepdims=True)*learning_rateweight_input_hidden+=learning_rate*np.dot(X_train.T,d_hidden)bias_hidden+=np.sum(d_hidden,axis=0,keepdims=True)*learning_rateif(epoch%1000==0):print("模型训练{0}次的结果是{1}".format(epoch,np.mean(np.abs(error))))
#模型训练结束,使用测试集对模型进行测试
hidden_layer_input = np.dot(X_test, weight_input_hidden) + bias_hidden
hidden_layer_output = sigmoid(hidden_layer_input)
output_layer_input = np.dot(hidden_layer_output, weight_hidden_output) + bias_output
output_layer_output = sigmoid(output_layer_input)predicted_class = np.argmax(output_layer_output, axis=1)
print(predicted_class)
true_class = np.argmax(y_test, axis=1)
accuracy = np.mean(predicted_class == true_class)
print(f'Test Accuracy: {accuracy * 100}%')

我们来看看代码跑出来的结果:
在这里插入图片描述
可以看到我们的误差为0.032,而分类的精准度为100%。

这篇关于对神经网络的理解,以及代码的手动实现,以鸾尾花数据集为例的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/788561

相关文章

基于Go语言实现一个压测工具

《基于Go语言实现一个压测工具》这篇文章主要为大家详细介绍了基于Go语言实现一个简单的压测工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录整体架构通用数据处理模块Http请求响应数据处理Curl参数解析处理客户端模块Http客户端处理Grpc客户端处理Websocket客户端

Java CompletableFuture如何实现超时功能

《JavaCompletableFuture如何实现超时功能》:本文主要介绍实现超时功能的基本思路以及CompletableFuture(之后简称CF)是如何通过代码实现超时功能的,需要的... 目录基本思路CompletableFuture 的实现1. 基本实现流程2. 静态条件分析3. 内存泄露 bug

一文详解Python中数据清洗与处理的常用方法

《一文详解Python中数据清洗与处理的常用方法》在数据处理与分析过程中,缺失值、重复值、异常值等问题是常见的挑战,本文总结了多种数据清洗与处理方法,文中的示例代码简洁易懂,有需要的小伙伴可以参考下... 目录缺失值处理重复值处理异常值处理数据类型转换文本清洗数据分组统计数据分箱数据标准化在数据处理与分析过

C#实现添加/替换/提取或删除Excel中的图片

《C#实现添加/替换/提取或删除Excel中的图片》在Excel中插入与数据相关的图片,能将关键数据或信息以更直观的方式呈现出来,使文档更加美观,下面我们来看看如何在C#中实现添加/替换/提取或删除E... 在Excandroidel中插入与数据相关的图片,能将关键数据或信息以更直观的方式呈现出来,使文档更

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常

C#实现系统信息监控与获取功能

《C#实现系统信息监控与获取功能》在C#开发的众多应用场景中,获取系统信息以及监控用户操作有着广泛的用途,比如在系统性能优化工具中,需要实时读取CPU、GPU资源信息,本文将详细介绍如何使用C#来实现... 目录前言一、C# 监控键盘1. 原理与实现思路2. 代码实现二、读取 CPU、GPU 资源信息1.

SpringBoot实现动态插拔的AOP的完整案例

《SpringBoot实现动态插拔的AOP的完整案例》在现代软件开发中,面向切面编程(AOP)是一种非常重要的技术,能够有效实现日志记录、安全控制、性能监控等横切关注点的分离,在传统的AOP实现中,切... 目录引言一、AOP 概述1.1 什么是 AOP1.2 AOP 的典型应用场景1.3 为什么需要动态插

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

Python脚本实现自动删除C盘临时文件夹

《Python脚本实现自动删除C盘临时文件夹》在日常使用电脑的过程中,临时文件夹往往会积累大量的无用数据,占用宝贵的磁盘空间,下面我们就来看看Python如何通过脚本实现自动删除C盘临时文件夹吧... 目录一、准备工作二、python脚本编写三、脚本解析四、运行脚本五、案例演示六、注意事项七、总结在日常使用

Java实现Excel与HTML互转

《Java实现Excel与HTML互转》Excel是一种电子表格格式,而HTM则是一种用于创建网页的标记语言,虽然两者在用途上存在差异,但有时我们需要将数据从一种格式转换为另一种格式,下面我们就来看看... Excel是一种电子表格格式,广泛用于数据处理和分析,而HTM则是一种用于创建网页的标记语言。虽然两