博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
用JS实现简单的神经网络算法
阅读量:5820 次
发布时间:2019-06-18

本文共 3081 字,大约阅读时间需要 10 分钟。

hot3.png

笔者尝试用JavaScript实现最简单的神经网络算法。

神经网络简介

神经网络试图模拟大脑的神经元之间的关系来处理信息。它的计算模型通常需要大量彼此连接的节点。每个神经元通过某种特殊的输出函数来处理来自其它相邻神经元的加权输入值。

神经元之间的信息传递的强度,用所谓的加权值来定义,算法会不断的调整加权值来实现自我的学习过程。

神经网络分为多层,如上图,有输入层,隐藏层和输出层。

JS线性代数包

神经网络的计算涉及到大量的矩阵计算,有许多的线性代数的开源的软件,Python下有著名的,非常有名。 Javascript也有几个:

我使用了numericjs,效果还不错。推荐大家可以试试。

两层神经网络

我们有一些简单的输入输出的数据用来训练神经网络。这里每一行代表一条数据。输入有三个参数,输出是一个。

Inputs 0 Inputs 1 Inputs 2 Output
0 0 1 0
1 1 1 1
1 0 1 1
0 1 1 0

首先我们实现一个最简单的神经网络,没有隐藏层,输入直连输出。

因为输入是三个参数,输出是一个,所以我们的神经网络输入层是三个节点,输出是1个。

// Sigmod functionfunction nonlin(x, deriv) {  if (deriv) {    return numeric.mul(x, numeric.sub(1, x));  }  return numeric.div(1, numeric.add(1, numeric.exp(numeric.neg(x))));}function train_neural(X, y, iteration) {  // initialize weights  var syn0 = numeric.sub(numeric.mul(2, numeric.random([3, 1])), 1);  //Training loop  var i = 0;  for (; i < iteration; i++) {    var l0 = X;    var l1 = nonlin(numeric.dot(l0, syn0));    var l1_error = numeric.sub(y, l1);    var l1_delta = numeric.mul(l1_error, nonlin(l1, true));    syn0 = numeric.add(syn0, numeric.dot(numeric.transpose(l0), l1_delta));    }   }}//Initial input/ouput valuesvar X = [  [0, 0, 1],  [0, 1, 1],  [1, 0, 1],  [1, 1, 1]];var y = [  [0],  [0],  [1],  [1]];train_neural(X, y, 1000);

简单介绍一下训练的代码和过程

  • X 输入数据
  • y 输出数据
  • nonlin, S函数
  • l0,网络第一层,这是等于输入数据
  • l1,网络第二层,这里就是输出层
  • syn0,第一层网络的权重

训练的迭代过程就是先给出一个初始的权重,利用这个权重算出一个输出值,用目标结果减去这个值,得到一个差异值,再利用这个差异值对权重进行修正。

1000次迭代后的网络输出: [0.03,0.02, 0.979, 0.974]

1000次迭代后的syn0权重值: [7.266,-0.221,-3.415]

这里我们发现第一个节点的权重较大,这个和我们的数据是一致的,通过观察数据我们也可以发现,输出值和第一列的输入值是强相关。如果增加迭代的次数,这个值会更大。

三层神经网络

Inputs 0 Inputs 1 Inputs 2 Output
0 0 1 0
0 1 1 1
1 0 1 1
1 1 1 0

现在我们有了一组新的数据,通过观察发现,第三列和结果完全无关,第一列和第二列相同时结果为0,否则为1。这是一种非线性的关系,为了有效学习我们增加一层,网络变成了这个样子。

// Sigmod functionfunction nonlin(x, deriv) {  if (deriv) {    return numeric.mul(x, numeric.sub(1, x));  }  return numeric.div(1, numeric.add(1, numeric.exp(numeric.neg(x))));}function train_neural(X, y, iteration) {  // initialize weights  var syn0 = [    [-0.1653904, 0.11737966, -0.71922612, -0.60379702],    [0.60148914, 0.93652315, -0.37315164, 0.38464523],    [0.7527783, 0.78921333, -0.82991158, -0.92189043]  ];  var syn1 = [    [-0.66033916],    [0.75628501],    [-0.80330633],    [-0.15778475]  ];  //Training loop  var i = 0;  for (; i < 1000; i++) {    var l0 = X;    var l1 = nonlin(numeric.dot(l0, syn0));    var l2 = nonlin(numeric.dot(l1, syn1));    var l2_error = numeric.sub(y, l2);    var l2_delta = numeric.mul(l2_error, nonlin(l2, true));    var l1_error = numeric.dot(l2_delta, numeric.transpose(syn1));    var l1_delta = numeric.mul(l1_error, nonlin(l1, true));    syn1 = numeric.add(syn1, numeric.dot(numeric.transpose(l1), l2_delta));    syn0 = numeric.add(syn0, numeric.dot(numeric.transpose(l0), l1_delta));  }}//Initial input/output valuesvar X = [  [0, 0, 1],  [0, 1, 1],  [1, 0, 1],  [1, 1, 1]];var y = [  [0],  [1],  [1],  [0]];train_neural(X, y, 1000);

训练的过程和之前的两层差别不大,只是多了一层。通过增加的这一层,可以有效的学习数据中的复杂非线性的关联关系。

经过1000次迭代, 输出值为:[0.02,0.95,0.94,0.05]

syn0 : 

 

如果大家对训练的过程有兴趣,可以运行我的code

 

转载于:https://my.oschina.net/taogang/blog/686568

你可能感兴趣的文章
通用代码引擎生成工具 XDoclet 介绍
查看>>
闭包,闭包用途,call、apply、bind 的用法
查看>>
约会 倍增lca
查看>>
单元测试(Unit testing)
查看>>
UVA 810 A Dicey Promblem 筛子难题 (暴力BFS+状态处理)
查看>>
echarts图调用多个接口
查看>>
建筑装修保温(一)
查看>>
常用排序算法(归并排序)
查看>>
迷你MVVM框架 avalonjs1.5 入门教程
查看>>
C# 讀取Excel、xlsx文件Excel2007
查看>>
LeetCode 10. Regular Expression Matching / 44. Wildcard Matching
查看>>
爬虫IP被禁的简单解决方法——切换UserAgent
查看>>
VUE入门
查看>>
JavaScript按IP地址排序
查看>>
正则表达式的基础组成部分
查看>>
python os模块常用方法总结
查看>>
二进制字符串匹配
查看>>
Uva 12299 带循环移动的RMQ(线段树)
查看>>
Android深度探索(卷1)HAL与驱动开发学习笔记(3)
查看>>
(转) UIALertView的基本用法与UIAlertViewDelegate对对话框的事件处理方法
查看>>