当前位置: 首页 > 编程日记 > 正文

技术图文:如何利用 C# 实现 误差反向传播 学习规则?

背景

我们在 如何利用 C# 对神经网络模型进行抽象? 中完成了神经网络的抽象结构:

  • 三个接口:激活函数、有监督学习、无监督学习

接口

  • 三个抽象类:神经元、网络层、网络拓扑

抽象类

我们在 如何利用 C# 实现神经网络的感知器模型? 中对神经网络的结构进行了扩展,完成了感知器神经网络:

  • 实现了三个接口:利用符号函数和阈值函数实现激活函数接口;利用感知器学习规则实现有监督学习接口。

接口实现

  • 覆写了三个抽象类中的抽象方法:用实体类 ActivationLayer 继承 Layer;用实体类 ActivationNeuron 继承 Neuron;用实体类 ActivationNetwork 继承 Network

实体类继承

实体类继承

我们在 如何利用 C# 实现 Delta 学习规则? 中对神经网络进行进一步的扩展。

  • 实现了两个接口:利用Sigmoid函数实现激活函数接口;利用 Delta 学习规则实现有监督学习接口。

接口实现


技术分析

今天,我们进一步扩展神经网络的学习规则,利用误差反向传播学习规则来实现有监督学习接口;增加BP神经网络另外一个常用的Sigmoid函数。

接口实现

有关BP神经网络的原理,参见图文 基于Matlab的BP神经网络在语音特征信号识别中的应用,这篇图文中有详细的权值、阈值推导过程。


代码实现

双极性Sigmoid函数

public class BipolarSigmoidFunction : IActivationFunction
{public double Alpha { get; set; }public BipolarSigmoidFunction(double alpha=2){Alpha = alpha;}public double Function(double x){return 2 / (1 + Math.Exp(-Alpha * x)) - 1;}public double Derivative(double x){double y = Function(x);return Alpha * (1 - y * y) / 2;}public double Derivative2(double y){return Alpha * (1 - y * y) / 2;}
}

误差反传算法的实现

public class BackPropagationLearning : ISupervisedLearning
{private readonly ActivationNetwork _network;private double _learningRate = 0.1;private readonly double[][] _neuronErrors;private readonly double[][][] _weightsUpdates;private readonly double[][] _thresholdsUpdates;public double LearningRate{get { return _learningRate; }set { _learningRate = Math.Max(0.0, Math.Min(1.0, value)); }}public BackPropagationLearning(ActivationNetwork network){_network = network;_neuronErrors = new double[network.Layers.Length][];_weightsUpdates = new double[network.Layers.Length][][];_thresholdsUpdates = new double[network.Layers.Length][];for (int i = 0; i < network.Layers.Length; i++){Layer layer = network.Layers[i];_neuronErrors[i] = new double[layer.Neurons.Length];_weightsUpdates[i] = new double[layer.Neurons.Length][];_thresholdsUpdates[i] = new double[layer.Neurons.Length];for (int j = 0; j < _weightsUpdates[i].Length; j++){_weightsUpdates[i][j] = new double[layer.InputsCount];}}}public double Run(double[] input, double[] output){// 计算网络的输出_network.Compute(input);// 计算各层的误差double error = CalculateError(output);// 计算权值和阈值的改变量CalculateUpdates(input);// 更新整个网络UpdateNetwork();return error;}public double RunEpoch(double[][] input, double[][] output){double error = 0.0;for (int i = 0; i < input.Length; i++){error += Run(input[i], output[i]);}return error;}
}    

计算各层的误差与网络的总体误差

private double CalculateError(double[] desiredOutput)
{double error = 0;int layersCount = _network.Layers.Length;ActivationNeuron activationNeuron = _network.Layers[0].Neurons[0] as ActivationNeuron;if (activationNeuron != null){IActivationFunction function = activationNeuron.ActivationFunction;Layer layer = _network.Layers[layersCount - 1];double[] errors = _neuronErrors[layersCount - 1];for (int i = 0; i < layer.Neurons.Length; i++){double output = layer.Neurons[i].Output;double e = desiredOutput[i] - output;errors[i] = e * function.Derivative2(output);error += e * e;}for (int j = layersCount - 2; j >= 0; j--){layer = _network.Layers[j];Layer layerNext = _network.Layers[j + 1];errors = _neuronErrors[j];double[] errorsNext = _neuronErrors[j + 1];for (int i = 0; i < layer.Neurons.Length; i++){double sum = 0.0;for (int k = 0; k < layerNext.Neurons.Length; k++){sum += errorsNext[k] * layerNext.Neurons[k].Weights[i];}errors[i] = sum * function.Derivative2(layer.Neurons[i].Output);}}}return error / 2.0;
}

计算网络权值和阈值的调整值

private void CalculateUpdates(double[] input)
{Layer layer = _network.Layers[0];double[] errors = _neuronErrors[0];double[][] layerWeightsUpdates = _weightsUpdates[0];double[] layerThresholdUpdates = _thresholdsUpdates[0];for (int i = 0; i < layer.Neurons.Length; i++){double error = errors[i];double[] neuronWeightUpdates = layerWeightsUpdates[i];for (int j = 0; j < neuronWeightUpdates.Length; j++){neuronWeightUpdates[j] = _learningRate * error * input[j];}layerThresholdUpdates[i] = _learningRate * error;}for (int k = 1; k < _network.Layers.Length; k++){Layer layerPrev = _network.Layers[k - 1];layer = _network.Layers[k];errors = _neuronErrors[k];layerWeightsUpdates = _weightsUpdates[k];layerThresholdUpdates = _thresholdsUpdates[k];for (int i = 0; i < layer.Neurons.Length; i++){double error = errors[i];double[] neuronWeightUpdates = layerWeightsUpdates[i];for (int j = 0; j < neuronWeightUpdates.Length; j++){neuronWeightUpdates[j] = _learningRate * error * layerPrev.Neurons[j].Output;}layerThresholdUpdates[i] = _learningRate * error;}}
}

调整整个网络的权值和阈值

private void UpdateNetwork()
{for (int i = 0; i < _network.Layers.Length; i++){Layer layer = _network.Layers[i];double[][] layerWeightsUpdates = _weightsUpdates[i];double[]  layerThresholdUpdates = _thresholdsUpdates[i];for (int j = 0; j < layer.Neurons.Length; j++){double[] neuronWeightUpdates = layerWeightsUpdates[j];ActivationNeuron neuron = layer.Neurons[j] as ActivationNeuron;for (int k = 0; k < neuron.Weights.Length; k++){neuron.Weights[k] += neuronWeightUpdates[k];}neuron.Threshold += layerThresholdUpdates[j];}}
}

总结

我们仍然应用 基于Matlab的BP神经网络在语音特征信号识别中的应用 中的语音信号分类的例子,来说明 BP 神经网络的应用。

获取数据集的方法

static List<double[]> GetData()
{List<double[]> result = new List<double[]>();string[] files = Directory.GetFiles(@".\data");foreach (string file in files){string[] dataset = File.ReadAllLines(file);foreach (string d in dataset){if (string.IsNullOrEmpty(d))continue;string[] data = d.Split(',');double[] item = new double[data.Length];for (int i = 0; i < data.Length; i++){item[i] = Convert.ToDouble(data[i]);}result.Add(item);}}result = result.OrderBy(c => Guid.NewGuid()).ToList();return result;
}

获取输入、输出样本的方法

private static void GetDataset(List<double[]> lst, out double[][] inputs, out double[][] outputs)
{int count = lst.Count;inputs = new double[count][];outputs = new double[count][];for (int i = 0; i < count; i++){double[] data = lst[i];int k = (int) data[0];switch (k){case 1:outputs[i] = new double[] {1, 0, 0, 0};break;case 2:outputs[i] = new double[] {0, 1, 0, 0};break;case 3:outputs[i] = new double[] { 0, 0, 1, 0 };break;case 4:outputs[i] = new double[] { 0, 0, 0, 1 };break;}inputs[i] = new double[data.Length - 1];for (int j = 0; j < inputs[i].Length; j++){inputs[i][j] = data[j + 1];}}
}

主应用程序

  • 输入层 24 个神经元
  • 中间层 25 个神经元
  • 输出层 4 个神经元
  • 训练样本 1500 个
  • 测试样本 500 个
  • 最大迭代次数 1000 次
static void Main(string[] args)
{List<double[]> lst = GetData();double[][] input;double[][] output;int trainNum = 1500;List<double[]> trainData = lst.GetRange(0, trainNum);GetDataset(trainData, out input, out output);ActivationNetwork network = new ActivationNetwork(new SigmoidFunction(),24, new int[] {25, 4});ISupervisedLearning bp = new BackPropagationLearning(network);for (int i = 0; i < 1000; i++){double error = bp.RunEpoch(input, output);Console.WriteLine("{0}:{1}",i,error);if (error <= 0.01)break;}List<double[]> testData = lst.GetRange(trainNum, lst.Count - trainNum);GetDataset(testData, out input, out output);int num = 0;for (int i = 0; i < testData.Count; i++){double[] result = network.Compute(input[i]);int index = MaxIndex(result);if (output[i][index] == 1)num++;}Console.WriteLine("准确率为:{0}", 1.0*num/testData.Count);
}

最后得到的准确率为:91.2%,与用 Matlab 计算的近似。好了,今天就到这里吧!See You!


对了,到目前为止已经有 10 名同学 通过解码 Huffman Code 得到团队的报名方式。

  • 如何加入 LSGO 软件技术团队?

团队的招新仍在进行中,对我们感兴趣的同学欢迎与我联系,我会亲自带着大家学习,一起成长!


相关图文

  • 如何利用 C# 实现 K 最邻近算法?
  • 如何利用 C# 实现 K-D Tree 结构?
  • 如何利用 C# + KDTree 实现 K 最邻近算法?
  • 如何利用 C# 对神经网络模型进行抽象?
  • 如何利用 C# 实现神经网络的感知器模型?
  • 如何利用 C# 实现 Delta 学习规则?
  • 如何利用 C# 爬取带 Token 验证的网站数据?
  • 如何利用 C# 向 Access 数据库插入大量数据?
  • 如何利用 C# 开发「桌面版百度翻译」软件!
  • 如何利用 C# 开发「股票数据分析软件」(上)
  • 如何利用 C# 开发「股票数据分析软件」(中)
  • 如何利用 C# 开发「股票数据分析软件」(下)
  • 如何利用 C# 爬取「财报说」中的股票数据?
  • 如何利用 C# 爬取 One 持有者返利数据!
  • 如何利用 C# 爬取Gate.io交易所的公告!
  • 如何利用 C# 爬取BigOne交易所的公告!
  • 如何利用 C# 爬取 ONE 的交易数据?
  • 如何利用 C# 爬取「猫眼电影:热映口碑榜」及对应影片信息!
  • 如何利用 C# 爬取「猫眼电影专业版:票房」数据!
  • 如何利用 C# 爬取「猫眼电影:最受期待榜」及对应影片信息!
  • 如何利用 C# 爬取「猫眼电影:国内票房榜」及对应影片信息!
  • 如何利用 C# + Python 破解猫眼电影的反爬虫机制?
  • 如何利用BigOne的API制作自动化交易系统 – 身份验证
  • 如何利用BigOne的API制作自动化交易系统 – 获取账户资产
  • 如何利用BigOne的API制作自动化交易系统 – 订单系统

相关文章:

SQL注入漏洞全接触--入门篇

随着B/S模式应用开发的发展&#xff0c;使用这种模式编写应用程序的程序员也越来越多。但是由于这个行业的入门门槛不高&#xff0c;程序员的水平及经验也参差不齐&#xff0c;相当大一部分程序员在编写代码的时候&#xff0c;没有对用户输入数据的合法性进行判断&#xff0c;使…

WannaCry 不相信眼泪 它需要你的安全防御与响应能力

在过去的几天里&#xff0c;WannaCry恶意软件及其变体影响了全球数百家组织与机构。 尽管每个组织都会因各种各样的原因没能及时对存在漏洞的系统做更新保护&#xff0c;或者担心更新实时系统的风险&#xff0c;两个月对于任何组织来用于采取措施保证系统安全也并不算太短的时间…

小白阶段如何学习Web前端知识

学会了UI设计技术&#xff0c;接下来的计划就是要找工作了&#xff0c;UI设计在面试环节的自我介绍很重要&#xff0c;有时候一分钟的自我介绍已经足够让HR判断出你适不适合他们公司&#xff0c;那做为一名UI设计师面试时如何自我介绍呢?来看看下面的详细介绍。 UI设计师面试时…

康泰瑞影推高性能3D/4D超声可视化方案

本文讲的是康泰瑞影推高性能3D/4D超声可视化方案,康泰瑞影(ContextVision)推出的业界首款超声实时3D立体图像增强产品已经配备全新的影像可视化功能。所推出的产品REALiCE?将提供逼真的3D超声影像&#xff0c;提高了诊断质量。 REALiCE软件将GOPiCE?自适应3D/4D立体图像增强产…

New Video Game Controlled By Kissing

unassimilatible writes "Artist Hye Yeon Nam has put her video game where her mouth is — literally — with the creation of a new bowling game thats controlled only by passionate (and awkward) French kissing. The Kiss Controller, as its called, has two…

资料分享:数学建模资料分享 -- 图论部分

背景 今天上午&#xff0c;在教六第一阶梯教室为数学建模俱乐部的同学们分享了有关图论的基本知识和应用。 课后&#xff0c;为同学们留了一个算法实现的小练习&#xff0c;大家可以先做一下。在本图文的末尾处&#xff0c;我把上课的资料以及代码分享出来&#xff0c;供大家…

免费学习编程的10个好工具

互联网时代的快速发展&#xff0c;很多人都在学习编程技术&#xff0c;小编今天为大家推荐的就是学习编程技术会用到的一些编程工具&#xff0c;免费学习编程的10个好工具!希望能够帮助到正在学习的小伙伴们。 免费学习编程的10个好工具&#xff1a; 1. Code/Racer Code/Racer是…

【ZT】我家宝宝不会哭----分享在美国养孩子的妈妈经(必看)

我在美国生了两个孩子&#xff0c;生育前后都有培训班&#xff0c;家庭医生每次洗脑让我受益匪浅&#xff0c;我的两个宝宝在婴儿时期乖巧得好像家里没有小婴儿&#xff0c;我甚至疑心她们会不会哭&#xff1f;如今外婆常拿这句傻话笑我。回国后看到朋友或邻居们被小祖宗折磨得…

资料分享:数学建模资料分享 -- 神经网络部分

背景 周日的时候&#xff0c;为数学建模俱乐部的同学们进行了一场有关人工神经网络方面的分享。虽然在这个方面有一些积累&#xff0c;但过于零散&#xff0c;所以拿了一堆文件拼凑成整体的内容。 幸亏自己有分享的习惯&#xff0c;学会一些知识就写下来&#xff0c;这次就是…

配置文件的简单使用

常见的配置文件格式:1.properties里面内容的格式 keyvalue2.xml后期详细解释若我们的配置文件为properties,并且放在src目录下.我们可以通过 ResourceBundle工具快速获取里面的配置信息使用步骤:1.获取ResourceBundle 对象:static ResourceBundle getBundle("文件名称不带…

软件测试工程师如何提升自己?

在软件测试行业&#xff0c;尤其是今年疫情的发生&#xff0c;想要在这个行业站稳脚跟&#xff0c;有着扎实的技术是非常重要的&#xff0c;今天小编要为大家介绍的内容就是软件测试工程师如何提升自己?希望能够给大家带来帮助。 软件测试工程师如何提升自己?熟练掌握IT核心技…

一些简单的SQL语句

1.创建数据库&#xff1a;create database database-name 2.删除数据库&#xff1a;delete database database-name 3.选择:select * from table where ... 4.插入&#xff1a;insert into table(field1,field2) values(value1,value2) 5.更新&#xff1a;update table set fie…

技术图文:进一步完善自动化交易系统 - 01

背景 最初&#xff0c;我们介绍了 如何利用网格交易法对数字资产进行交易。只要价格存在波动&#xff0c;通过这种方法就能赚取利润。 接着&#xff0c;我们介绍了 如何通过三角套利来交易数字资产。只要三种数字资产&#xff0c;两两存在交易对&#xff0c;就可以建立套利的…

String、StringBuffer与StringBuilder之间区别 (转载)

最近学习到StringBuffer&#xff0c;心中有好些疑问&#xff0c;搜索了一些关于String&#xff0c;StringBuffer&#xff0c;StringBuilder的东西&#xff0c;现在整理一下。 关于这三个类在字符串处理中的位置不言而喻&#xff0c;那么他们到底有什么优缺点&#xff0c;到底什…

女生可以学习Web前端吗?

随着时代的不断发展&#xff0c;女性独立意识越来越高&#xff0c;在职场这方面&#xff0c;很多岗位都有了女性的存在&#xff0c;尤其是IT互联网行业&#xff0c;广阔的发展前景和优厚的福利待遇吸引了很多女性小伙伴想要学习Web前端开发&#xff0c;那么女生可以学习Web前端…

adobe就不敢把融合做得更好一点?

体验了一把flash bulider4 和 flash cs5的结合。 之前说得挺牛B的协同开发也不过如此。还以为会给开发者更多的便利。。。 其也就是让建立一个有外链类的mc的时候&#xff0c;能自动为你在bulider里面提示新建一个类而已。 在flash bulider(flex bulider)中依然不认识flash中的…

apache开启虚拟主机 并进行配置

sudo vi /etc/apache2/httpd.conf 进入 apache 配置文件 在配置文件中搜索 Virtual hosts 如图 把前面的#删掉 #为注释 sudo vi /etc/apache2/extra/httpd-vhosts.conf 打开虚拟主机配置文件对下面的一下代码进行复制并粘贴在下方并更改里面的一些内容 DocumentRoot …

技术图文:进一步完善自动化交易系统 - 02

背景 自己的自动化交易系统起初只有网格交易法一个版本&#xff1a;如何利用网格交易法对数字资产进行交易。后面&#xff0c;加入了ONE、USDT、EOS的三角套利版本&#xff1a;如何通过三角套利来交易数字资产。接着&#xff0c;发现三角套利的个别订单在未全部成交的情况下被…

女士做软件测试的利弊有哪些?

社会在不断的进步&#xff0c;在如今的智能时代&#xff0c;女性独立意识越来越高&#xff0c;很多女性都把重心放在职场上&#xff0c;IT行业出现了越来越多的女性职业者&#xff0c;尤其是软件测试这一块&#xff0c;那么女士做软件测试的利弊有哪些呢?来看看下面的详细介绍…

刻意练习:Python基础 -- Task01. 变量、运算符与数据类型

背景 我们准备利用17天时间&#xff0c;将 Python 基础的刻意练习分为如下任务&#xff1a; Task01&#xff1a;变量、运算符与数据类型&#xff08;1day&#xff09;Task02&#xff1a;条件与循环&#xff08;1day&#xff09;Task03&#xff1a;列表与元组&#xff08;2day…

linux ramdisk与tmpfs的深入分析

一)ramdisk 1)概念 ramdisk就是指使用一部分内存空间来模拟硬盘分区,也就是说ramdisk是一个块设备,要用mkfs格式化,才能真正使用它. .ramdisk在内核2.0/2.2版本就已经支持. .ramdisk设备是它不允许重新声明它的内存空间,所以ramdisk块通常会一直占用空间的内存直到系统重启. .r…

Unique Paths II

注意一个容易犯的错误&#xff1a;判断obstacleGrid是否为1时&#xff0c;else那部分不能少。因为如果不加&#xff0c;就会默认把那些值设置为0。 class Solution { public:int uniquePathsWithObstacles(vector<vector<int>>& obstacleGrid) {int height ob…

零基础java培训如何规划学习路线

学习java的同学越来越多&#xff0c;很多小伙伴都想知道零基础如何学习java?如何安排学习路线&#xff0c;零基础java培训如何规划学习路线?那么针对以上问题&#xff0c;今天小编特意分享这篇Java学习指南&#xff0c;希望对那些想学Java&#xff0c;想零基础快速入门Java有…

CCIE-MPLS基础篇-实验手册

又一部前期JUSTECH&#xff08;南京捷式泰&#xff09;工程师职业发展系列丛书完整拷贝。 MPLS&#xff08;Multi-Protocol Label Switching&#xff09; 目录 1&#xff1a;MPLS 基础实验.... 3 1.1实验拓扑... 3 1.2实验需求&#xff1a;... 3 1.3实验步骤... 3 1.4校验…

[转载]Oracle 11g新特征之形式料理(2)

不可见的索引 您经常感应利诱&#xff0c;索引能否真的有利于用户的盘考&#xff1f;它也许有利于一个盘考&#xff0c;但会影响 10 个其他盘考。索引肯定会对 INSERT 语句形成负面影响&#xff0c;也会执行潜伏的删除和更新操纵&#xff0c;这取决于 WHERE 条件能否在索引中包…

技术图文:进一步完善自动化交易系统 - 03

背景 在 进一步完善自动化交易系统 - 02 这篇图文中&#xff0c;向大家详细介绍了&#xff0c;满足以下两个条件下&#xff1a; 拥有数字资产&#xff1a;BTC、EOS、USDT、ONEBigOne 提供交易对&#xff1a;ONE-USDT、ONE-EOS、ONE-BTC、EOS-BTC、EOS-USDT、BTC-USDT 我们可…

HTML5培训后好就业吗

HTML5技术在移动互联网行业的发展前景是非常可观的&#xff0c;以至于很多人都非常看好HTML5行业&#xff0c;想要学习HTML5技术的人越来越多&#xff0c;但大家都比较关心HTML5培训后好就业吗?就这个问题我们来看看下面的详细介绍吧。 HTML5培训后好就业吗?首先&#xff0c;…

股市复盘:本周交易数据分析

特别说明&#xff1a; 每周我会统计「股票市场」的数据为大家的投资提供辅助支持&#xff0c;每个人站的角度不同&#xff0c;面对同一份数据得到的观点亦或不同&#xff0c;我只提供数据&#xff0c;不提供观点&#xff0c;欢迎大家在图文下方留言&#xff0c;我们理性探讨。 …

Strategy Analytics:物联网安全令人担忧,成本节约效益也不确定

Strategy Analytics物联网战略研究报告《2016年物联网部署趋势及使用调查》发现&#xff0c;物联网安全问题令人担忧&#xff0c;同时&#xff0c;超过半数企业对物联网带来的成本节约效益也不太确定。56%的受访公司认为&#xff0c;大数据分析是推动物联网部署的首要驱动力。 …

SQLServer中设置XML索引

XML索引分为主XML索引和次XML索引&#xff1b;1.主XML索引&#xff1a;为了完整、一致的表示XML的值&#xff0c;格式&#xff1a;Create PRIMARY XML INDEX indexName ON tableName(columName);2.次XML索引&#xff1a;对于XPath和XQuery表达式&#xff0c;可以使用XML次索引&…