最近一直在看Deep Learning,各类博客、论文看得不少

但是说实话,这样做有些疏于实现,一来呢自己的电脑也不是很好,二来呢我目前也没能力自己去写一个toolbox

只是跟着Andrew Ng的UFLDL tutorial 写了些已有框架的代码(这部分的代码见github)

后来发现了一个matlab的Deep Learning的toolbox,发现其代码很简单,感觉比较适合用来学习算法

再一个就是matlab的实现可以省略掉很多数据结构的代码,使算法思路非常清晰

所以我想在解读这个toolbox的代码的同时来巩固自己学到的,同时也为下一步的实践打好基础

(本文只是从代码的角度解读算法,具体的算法理论步骤还是需要去看paper的

我会在文中给出一些相关的paper的名字,本文旨在梳理一下算法过程,不会深究算法原理和公式)

==========================================================================================

使用的代码:DeepLearnToolbox  ,下载地址:点击打开,感谢该toolbox的作者

==========================================================================================

今天是CNN的内容啦,CNN讲起来有些纠结,你可以事先看看convolution和pooling(subsampling),还有这篇:tornadomeet的博文

下面是那张经典的图:

======================================================================================================

打开\tests\test_example_CNN.m一观

[cpp] view plaincopy
  1. cnn.layers = {
  2. struct('type', 'i') %input layer
  3. struct('type', 'c', 'outputmaps', 6, 'kernelsize', 5) %convolution layer
  4. struct('type', 's', 'scale', 2) %sub sampling layer
  5. struct('type', 'c', 'outputmaps', 12, 'kernelsize', 5) %convolution layer
  6. struct('type', 's', 'scale', 2) %subsampling layer
  7. };
  8. cnn = cnnsetup(cnn, train_x, train_y);        //here!!!
  9. opts.alpha = 1;
  10. opts.batchsize = 50;
  11. opts.numepochs = 1;
  12. cnn = cnntrain(cnn, train_x, train_y, opts);  //here!!!

似乎这次要复杂了一些啊,首先是layer,有三种,i是input,c是convolution,s是subsampling

'c'的outputmaps是convolution之后有多少张图,比如上(最上那张经典的))第一层convolution之后就有六个特征图

'c'的kernelsize 其实就是用来convolution的patch是多大

's'的scale就是pooling的size为scale*scale的区域

接下来似乎就是常规思路了,cnnsetup()和cnntrain()啦,我们来看代码

\CNN\cnnsetup.m

主要是一些参数的作用的解释,详细的参看代码里的注释啊
[cpp] view plaincopy
  1. function net = cnnsetup(net, x, y)
  2. inputmaps = 1;
  3. mapsize = size(squeeze(x(:, :, 1)));
  4. //尤其注意这几个循环的参数的设定
  5. //numel(net.layers)  表示有多少层
  6. for l = 1 : numel(net.layers)   //  layer
  7. if strcmp(net.layers{l}.type, 's')
  8. mapsize = mapsize / net.layers{l}.scale;
  9. //subsampling层的mapsize,最开始mapsize是每张图的大小28*28(这是第一次卷积后的结果,卷积前是32*32)
  10. //这里除以scale,就是pooling之后图的大小,这里为14*14
  11. assert(all(floor(mapsize)==mapsize), ['Layer ' num2str(l) ' size must be integer. Actual: ' num2str(mapsize)]);
  12. for j = 1 : inputmaps //inputmap就是上一层有多少张特征图,通过初始化为1然后依层更新得到
  13. net.layers{l}.b{j} = 0;
  14. end
  15. end
  16. if strcmp(net.layers{l}.type, 'c')
  17. mapsize = mapsize - net.layers{l}.kernelsize + 1;
  18. //这里的mapsize可以参见UFLDL里面的那张图下面的解释
  19. fan_out = net.layers{l}.outputmaps * net.layers{l}.kernelsize ^ 2;
  20. //隐藏层的大小,是一个(后层特征图数量)*(用来卷积的patch图的大小)
  21. for j = 1 : net.layers{l}.outputmaps  //  output map
  22. fan_in = inputmaps * net.layers{l}.kernelsize ^ 2;
  23. //对于每一个后层特征图,有多少个参数链到前层
  24. for i = 1 : inputmaps  //  input map
  25. net.layers{l}.k{i}{j} = (rand(net.layers{l}.kernelsize) - 0.5) * 2 * sqrt(6 / (fan_in + fan_out));
  26. end
  27. net.layers{l}.b{j} = 0;
  28. end
  29. inputmaps = net.layers{l}.outputmaps;
  30. end
  31. end
  32. // 'onum' is the number of labels, that's why it is calculated using size(y, 1). If you have 20 labels so the output of the network will be 20 neurons.
  33. // 'fvnum' is the number of output neurons at the last layer, the layer just before the output layer.
  34. // 'ffb' is the biases of the output neurons.
  35. // 'ffW' is the weights between the last layer and the output neurons. Note that the last layer is fully connected to the output layer, that's why the size of the weights is (onum * fvnum)
  36. fvnum = prod(mapsize) * inputmaps;
  37. onum = size(y, 1);
  38. //这里是最后一层神经网络的设定
  39. net.ffb = zeros(onum, 1);
  40. net.ffW = (rand(onum, fvnum) - 0.5) * 2 * sqrt(6 / (onum + fvnum));
  41. end

\CNN\cnntrain.m

cnntrain就和nntrain是一个节奏了:

[cpp] view plaincopy
  1. net = cnnff(net, batch_x);
  2. net = cnnbp(net, batch_y);
  3. net = cnnapplygrads(net, opts);

cnntrain是用back propagation来计算gradient的,我们一次来看这三个函数:

cnnff.m

这部分计算还比较简单,可以说是有迹可循,大家最好看看tornadomeet的博文的步骤,说得比较清楚
[cpp] view plaincopy
  1. function net = cnnff(net, x)
  2. n = numel(net.layers);
  3. net.layers{1}.a{1} = x;
  4. inputmaps = 1;
  5. for l = 2 : n   //  for each layer
  6. if strcmp(net.layers{l}.type, 'c')
  7. //  !!below can probably be handled by insane matrix operations
  8. for j = 1 : net.layers{l}.outputmaps   //  for each output map
  9. //  create temp output map
  10. z = zeros(size(net.layers{l - 1}.a{1}) - [net.layers{l}.kernelsize - 1 net.layers{l}.kernelsize - 1 0]);
  11. for i = 1 : inputmaps   //  for each input map
  12. //  convolve with corresponding kernel and add to temp output map
  13. //  做卷积,参考UFLDL,这里是对每一个input的特征图做一次卷积,再加起来
  14. z = z + convn(net.layers{l - 1}.a{i}, net.layers{l}.k{i}{j}, 'valid');
  15. end
  16. //  add bias, pass through nonlinearity
  17. //  加入bias
  18. net.layers{l}.a{j} = sigm(z + net.layers{l}.b{j});
  19. end
  20. //  set number of input maps to this layers number of outputmaps
  21. inputmaps = net.layers{l}.outputmaps;
  22. elseif strcmp(net.layers{l}.type, 's')
  23. //  downsample
  24. for j = 1 : inputmaps
  25. //这里有点绕绕的,它是新建了一个patch来做卷积,但我们要的是pooling,所以它跳着把结果读出来,步长为scale
  26. //这里做的是mean-pooling
  27. z = convn(net.layers{l - 1}.a{j}, ones(net.layers{l}.scale) / (net.layers{l}.scale ^ 2), 'valid');   //  !! replace with variable
  28. net.layers{l}.a{j} = z(1 : net.layers{l}.scale : end, 1 : net.layers{l}.scale : end, :);
  29. end
  30. end
  31. end
  32. //  收纳到一个vector里面,方便后面用~~
  33. //  concatenate all end layer feature maps into vector
  34. net.fv = [];
  35. for j = 1 : numel(net.layers{n}.a)
  36. sa = size(net.layers{n}.a{j});
  37. net.fv = [net.fv; reshape(net.layers{n}.a{j}, sa(1) * sa(2), sa(3))];
  38. end
  39. //  最后一层的perceptrons,数据识别的结果
  40. net.o = sigm(net.ffW * net.fv + repmat(net.ffb, 1, size(net.fv, 2)));
  41. end

cnnbp.m

这个就哭了,代码有些纠结,不得已又找资料看啊,《Notes on Convolutional Neural Networks》要好一些

只是这个toolbox的代码和《Notes on Convolutional Neural Networks》里有些不一样的是这个toolbox在subsampling(也就是pooling层)没有加sigmoid激活函数,只是单纯地pooling了一下,所以这地方还需仔细辨别,这个toolbox里的subsampling是不用计算gradient的,而在Notes里是计算了的

还有这个toolbox没有Combinations of Feature Maps,也就是tornadomeet的博文里这张表格:

具体就去看看上面这篇论文吧

然后就看代码吧:

[cpp] view plaincopy
  1. function net = cnnbp(net, y)
  2. n = numel(net.layers);
  3. //  error
  4. net.e = net.o - y;
  5. //  loss function
  6. net.L = 1/2* sum(net.e(:) .^ 2) / size(net.e, 2);
  7. //从最后一层的error倒推回来deltas
  8. //和神经网络的bp有些类似
  9. backprop deltas
  10. net.od = net.e .* (net.o .* (1 - net.o));   //  output delta
  11. net.fvd = (net.ffW' * net.od);              //  feature vector delta
  12. if strcmp(net.layers{n}.type, 'c')         //  only conv layers has sigm function
  13. net.fvd = net.fvd .* (net.fv .* (1 - net.fv));
  14. end
  15. //和神经网络类似,参看神经网络的bp
  16. //  reshape feature vector deltas into output map style
  17. sa = size(net.layers{n}.a{1});
  18. fvnum = sa(1) * sa(2);
  19. for j = 1 : numel(net.layers{n}.a)
  20. net.layers{n}.d{j} = reshape(net.fvd(((j - 1) * fvnum + 1) : j * fvnum, :), sa(1), sa(2), sa(3));
  21. end
  22. //这是算delta的步骤
  23. //这部分的计算参看Notes on Convolutional Neural Networks,其中的变化有些复杂
  24. //和这篇文章里稍微有些不一样的是这个toolbox在subsampling(也就是pooling层)没有加sigmoid激活函数
  25. //所以这地方还需仔细辨别
  26. //这这个toolbox里的subsampling是不用计算gradient的,而在上面那篇note里是计算了的
  27. for l = (n - 1) : -1 : 1
  28. if strcmp(net.layers{l}.type, 'c')
  29. for j = 1 : numel(net.layers{l}.a)
  30. net.layers{l}.d{j} = net.layers{l}.a{j} .* (1 - net.layers{l}.a{j}) .* (expand(net.layers{l + 1}.d{j}, [net.layers{l + 1}.scale net.layers{l + 1}.scale 1]) / net.layers{l + 1}.scale ^ 2);
  31. end
  32. elseif strcmp(net.layers{l}.type, 's')
  33. for i = 1 : numel(net.layers{l}.a)
  34. z = zeros(size(net.layers{l}.a{1}));
  35. for j = 1 : numel(net.layers{l + 1}.a)
  36. z = z + convn(net.layers{l + 1}.d{j}, rot180(net.layers{l + 1}.k{i}{j}), 'full');
  37. end
  38. net.layers{l}.d{i} = z;
  39. end
  40. end
  41. end
  42. //参见paper,注意这里只计算了'c'层的gradient,因为只有这层有参数
  43. calc gradients
  44. for l = 2 : n
  45. if strcmp(net.layers{l}.type, 'c')
  46. for j = 1 : numel(net.layers{l}.a)
  47. for i = 1 : numel(net.layers{l - 1}.a)
  48. net.layers{l}.dk{i}{j} = convn(flipall(net.layers{l - 1}.a{i}), net.layers{l}.d{j}, 'valid') / size(net.layers{l}.d{j}, 3);
  49. end
  50. net.layers{l}.db{j} = sum(net.layers{l}.d{j}(:)) / size(net.layers{l}.d{j}, 3);
  51. end
  52. end
  53. end
  54. //最后一层perceptron的gradient的计算
  55. net.dffW = net.od * (net.fv)' / size(net.od, 2);
  56. net.dffb = mean(net.od, 2);
  57. function X = rot180(X)
  58. X = flipdim(flipdim(X, 1), 2);
  59. end
  60. end

cnnapplygrads.m

这部分就轻松了,已经有grads了,依次进行梯度更新就好了

[cpp] view plaincopy
  1. function net = cnnapplygrads(net, opts)
  2. for l = 2 : numel(net.layers)
  3. if strcmp(net.layers{l}.type, 'c')
  4. for j = 1 : numel(net.layers{l}.a)
  5. for ii = 1 : numel(net.layers{l - 1}.a)
  6. net.layers{l}.k{ii}{j} = net.layers{l}.k{ii}{j} - opts.alpha * net.layers{l}.dk{ii}{j};
  7. end
  8. net.layers{l}.b{j} = net.layers{l}.b{j} - opts.alpha * net.layers{l}.db{j};
  9. end
  10. end
  11. end
  12. net.ffW = net.ffW - opts.alpha * net.dffW;
  13. net.ffb = net.ffb - opts.alpha * net.dffb;
  14. end

cnntest.m

好吧,我们得知道最后结果怎么来啊
[cpp] view plaincopy
  1. function [er, bad] = cnntest(net, x, y)
  2. //  feedforward
  3. net = cnnff(net, x);
  4. [~, h] = max(net.o);
  5. [~, a] = max(y);
  6. bad = find(h ~= a);
  7. er = numel(bad) / size(y, 2);
  8. end

就是这样~~cnnff一次后net.o就是结果

总结

just code !

这是一个89年的模型啊~~~,最近还和RBM结合起来了,做了一个Imagenet的最好成绩(是这个吧?):

Alex Krizhevsky.ImageNet Classification with Deep  Convolutional Neural Networks. Video and Slides, 2012
               http://www.cs.utoronto.ca/~rsalakhu/papers/dbm.pdf

【参考】:

【Deep learning:三十八(Stacked CNN简单介绍)】

【UFLDL】

【Notes on Convolutional Neural Networks】

【Convolutional Neural Networks (LeNet)】  这是deeplearning 的theano库的

【面向代码】学习 Deep Learning(三)Convolution Neural Network(CNN)相关推荐

  1. 【面向代码】学习 Deep Learning Convolution Neural Network(CNN)

    转载自: [面向代码]学习 Deep Learning(三)Convolution Neural Network(CNN) - DarkScope从这里开始 - 博客频道 - CSDN.NET htt ...

  2. 卷积神经网络Convolution Neural Network (CNN) 原理与实现

    本文结合Deep learning的一个应用,Convolution Neural Network 进行一些基本应用,参考Lecun的Document 0.1进行部分拓展,与结果展示(in pytho ...

  3. 转【面向代码】学习 Deep Learning(二)Deep Belief Nets(DBNs)

    [面向代码]学习 Deep Learning(二)Deep Belief Nets(DBNs) http://blog.csdn.net/dark_scope/article/details/9447 ...

  4. 2021李宏毅 机器学习 Convolutional Neural Network (CNN)

    应用1:影像辨识 一张图片是一个三维的Tensor,一维是长,一维是宽,一维是channel (channel=3代表RGB的三种颜色,长和宽代表解析度,像素的数目) 模型的弹性越大,越容易过拟合. ...

  5. 机器学习(Machine Learning)、深度学习(Deep Learning)、NLP面试中常考到的知识点和代码实现

    网址:https://github.com/NLP-LOVE/ML-NLP 此项目是机器学习(Machine Learning).深度学习(Deep Learning).NLP面试中常考到的知识点和代 ...

  6. 机器学习(Machine Learning)深度学习(Deep Learning)资料(Chapter 2)

    机器学习(Machine Learning)&深度学习(Deep Learning)资料(Chapter 2) - tony的专栏 - 博客频道 - CSDN.NET 注:机器学习资料篇目一共 ...

  7. 【深度学习Deep Learning】资料大全

    感谢关注天善智能,走好数据之路↑↑↑ 欢迎关注天善智能,我们是专注于商业智能BI,人工智能AI,大数据分析与挖掘领域的垂直社区,学习,问答.求职一站式搞定! 对商业智能BI.大数据分析挖掘.机器学习, ...

  8. 【github】机器学习(Machine Learning)深度学习(Deep Learning)资料

    转自:https://github.com/ty4z2008/Qix/blob/master/dl.md# <Brief History of Machine Learning> 介绍:这 ...

  9. 机器学习(Machine Learning)深度学习(Deep Learning)资料汇总

    本文来源:https://github.com/ty4z2008/Qix/blob/master/dl.md 机器学习(Machine Learning)&深度学习(Deep Learning ...

  10. 深度学习Deep Learning 资料大全

    转自:http://www.cnblogs.com/charlotte77/ [深度学习Deep Learning]资料大全 最近在学深度学习相关的东西,在网上搜集到了一些不错的资料,现在汇总一下: ...

最新文章

  1. SAP PM交叉申请时间表
  2. 数字孪生智慧城市建设加速5G+产业AI的应用,全流程数据透明化及实时可控成为升级的大方向,构建智慧城市生态圈在路上!...
  3. 剑指offer 面试题三 找出数组中重复的数字
  4. oracle故障级别划分,jfinal 2.2 oracle 事务级别问题
  5. 【译】怎样处理 Safari 移动端对图片资源的限制
  6. zk pivottable java_zookeeper节点类型和java客户端创建zk节点
  7. 如何自学python-如何自学python语言
  8. 数组遍历,判断数组中的对象中某一属性值时候为空
  9. VS2010 在Win 7 附加w3wp.exe进程进行调试
  10. NetXray嗅探器介绍
  11. 机器人抓取中物体3D定位算法介绍
  12. typecho图片插件_Typecho 文章缩略图插件
  13. 产品数据管理(PDM)技术概述
  14. 脚手架开发(1)-准备阶段
  15. MediaInfo与wmv3 wma3 的解码器
  16. R语言编程的高效方法
  17. 深度学习(计算机视觉)面试问题:
  18. Ubuntu下tar命令使用详解 .tar解压、.tar压缩
  19. Hadoop 基本常用命令
  20. 数据仓库之缓慢变化维度处理

热门文章

  1. 【mybatis深度历险系列】mybatis中的动态sql
  2. 字符串是通过“引用”传递的
  3. 工作那些事(四)大公司VS小公司
  4. hdu 1213 “How Many Tables”(并查集基本到优化)
  5. linux操作系统备份及恢复,备份Linux操作系统的几种方法
  6. 不等式解集怎么取_6.初中数学:一个不等式的解集,都是另一个不等式的解,求a的取值范围?...
  7. 1106 Lowest Price in Supply Chain (25 分) 静态树vector+DFS
  8. h5保存图片到手机_如何快速制作一个【H5】作品?
  9. 字符串替换(NYOJ)
  10. ajax php 数组参数传递参数,jquery中ajax传递中文参数和js数组问题