阅读《深度学习入门:基于python的理论与实现》,其中在实现CNN的章节中,提到为了CNN的快速计算需要将输入数据展开是以适合滤波器(权重),对于输入数据,将应用滤波器的区域(3维方块)横向展开为1列(如下图)。im2col会在所有应用滤波器的地方进行这个展开处理。


对输入数据进行处理

对卷积核操作

下面是使用展开后的输入数据和展开后的卷积核做矩阵乘法,得到结果进行col2im操作复原结果(注意不是im2col),关于col2im操作将在后面介绍


解释:
发现这个博主介绍的这种坐标轴表示的方法非常容易理解:
对于数据:

我们使用坐标轴表示(分别沿着0轴,再沿着1轴方向即为矩阵方向):

经过transpose(1, 0)之后,表示交换 ‘0轴’ 和 ‘1轴’,我们可以得到如下结果:

那么我们根据分别沿着0轴,再沿着1轴方向即为矩阵方向,可以得到结果:
0 2
1 3
a = np.arange(24)
a = a.reshape(2, 3, 4)
print(a)
输出:
array([[[ 0, 1, 2, 3],
[ 4, 5, 6, 7],
[ 8, 9, 10, 11]],
[[12, 13, 14, 15],
[16, 17, 18, 19],
[20, 21, 22, 23]]])
b = a.transpose(1, 0, 2)
b
输出:
array([[[ 0, 1, 2, 3],
[12, 13, 14, 15]],
[[ 4, 5, 6, 7],
[16, 17, 18, 19]],
[[ 8, 9, 10, 11],
[20, 21, 22, 23]]])
c = a.transpose(1, 2, 0)
c
输出:
array([[[ 0, 12],
[ 1, 13],
[ 2, 14],
[ 3, 15]],
[[ 4, 16],
[ 5, 17],
[ 6, 18],
[ 7, 19]],
[[ 8, 20],
[ 9, 21],
[10, 22],
[11, 23]]])
A = np.arange(95,99).reshape(2,2)
np.pad(A,((10,4),(2,3)),'constant',constant_values = (1,-1))

输出:
array([[ 1, 1, 1, 1, -1, -1, -1],
[ 1, 1, 1, 1, -1, -1, -1],
[ 1, 1, 1, 1, -1, -1, -1],
[ 1, 1, 1, 1, -1, -1, -1],
[ 1, 1, 1, 1, -1, -1, -1],
[ 1, 1, 1, 1, -1, -1, -1],
[ 1, 1, 1, 1, -1, -1, -1],
[ 1, 1, 1, 1, -1, -1, -1],
[ 1, 1, 1, 1, -1, -1, -1],
[ 1, 1, 1, 1, -1, -1, -1],
[ 1, 1, 95, 96, -1, -1, -1],
[ 1, 1, 97, 98, -1, -1, -1],
[ 1, 1, -1, -1, -1, -1, -1],
[ 1, 1, -1, -1, -1, -1, -1],
[ 1, 1, -1, -1, -1, -1, -1],
[ 1, 1, -1, -1, -1, -1, -1]])
def im2col(input_data, filter_h, filter_w, stride=1, pad=0):
"""
Parameters
----------
input_data :由(数据量,通道,高,长)的4维数组构成的输入数据
filter_h : 滤波器的高
filter_w : 滤波器的长
stride : 步幅
pad : 填充
Returns
-------
col : 2维数组
"""
# 获取 数据量、通道数、图像高度、图像长度
N, C, H, W = input_data.shape
# 对图像进行卷积运算后的输出高度,如图像是7X7,卷积核是5X5 结果是3X3
out_h = (H + 2*pad - filter_h)//stride + 1
# 对图像进行卷积运算后的输出宽度
out_w = (W + 2*pad - filter_w)//stride + 1
# 对图像在4个维度进行填充,默认pad为0
img = np.pad(input_data, [(0,0), (0,0), (pad, pad), (pad, pad)], 'constant')
# 见下面解释1
col = np.zeros((N, C, filter_h, filter_w, out_h, out_w))
# 从左到右,从上到下依次进行遍历
for y in range(filter_h):
# 见解释2
y_max = y + stride*out_h
for x in range(filter_w):
x_max = x + stride*out_w
# 见解释3
col[:, :, y, x, :, :] = img[:, :, y:y_max:stride, x:x_max:stride]
col = col.transpose(0, 4, 5, 1, 2, 3).reshape(N*out_h*out_w, -1)
return col

np.zeros((1, 1, 2, 2, 2, 3))
可以得到
array([[[[[[0., 0., 0.],
[0., 0., 0.]],
[[0., 0., 0.],
[0., 0., 0.]]],
[[[0., 0., 0.],
[0., 0., 0.]],
[[0., 0., 0.],
[0., 0., 0.]]]]]])
我们可以看见初始化后的结果包含了4个(卷积窗口滑动次数)大小为out_houtw大小的矩阵,这个out_h*out_w是与输出矩阵具有同样大小的感受夜,后面再reshape的时候我们可以将不同列变成同一行
y_max = y + stride*out_h ,获取纵轴方向的最大取值x_max = x + stride*out_w,获取横轴方向的最大取值col[:, :, y, x, :, :] = img[:, :, y:y_max:stride, x:x_max:stride]这里主要有两部分难点知识:
col[:, :, y, x, :, :]的赋值y:y_max:stride 跳跃取值col[:, :, y, x, :, :]的赋值我们通过代码来解释这段是如何赋值的:

其中col[:, 1, :]的含义为把[[8, 88, 888]]赋给col[][1][], 也就是把col第二维索引值为1的数组(此例中为[0,0,0]),更改为一个尺寸正好为col第一维和第三维的数组(此例为[8,88,888])
同理col[:, 0, :]的含义为把[[9, 99, 999]]赋给col[][1][]
回到col[:, :, y, x, :, :] = img[:, :, y:y_max:stride, x:x_max:stride], col数组为六维,img数组为四维,固定col数组的第三维为y,第四维为x; img数组的四维与col数组的第1维,第2维,第5维,第6维是相对应的,其中y:y_max:stride的长度为(y_max-y)/stride,也就等于out_h;x:x_max:stride的长度为(x_max-x)/stride,也就等于out_w,所以img的第3维与col的第5维长度一直,img的第6维与col的第6维长度一致
所以col[:, :, y, x, :, :] = img[:, :, y:y_max:stride, x:x_max:stride]赋值的意思是:依次把输入数据按照滤波器的尺寸进行分割,并存储到对应的位置。
y:y_max:stride 跳跃取值
step_data[0:10]是按照间隔为1进行取值step_data[0:10:2]是按照间隔为2进行取值y:y_max:stride是在[y, y_max) 范围内以间隔为stride进行取值看到这里,我最大的疑问:便是如果这样跳跃的取值那么不是和滑动窗口的概念不一致了么?
于是我在原有的函数中加上了几句打印,观察col在赋值中的变化
def im2col(input_data, filter_h, filter_w, stride=1, pad=0):
"""
Parameters
----------
input_data :由(数据量,通道,高,长)的4维数组构成的输入数据
filter_h : 滤波器的高
filter_w : 滤波器的长
stride : 步幅
pad : 填充
Returns
-------
col : 2维数组
"""
# 获取 数据量、通道数、图像高度、图像长度
N, C, H, W = input_data.shape
# 对图像进行卷积运算后的输出高度,如图像是7X7,卷积核是5X5 结果是3X3
out_h = (H + 2*pad - filter_h)//stride + 1
# 对图像进行卷积运算后的输出宽度
out_w = (W + 2*pad - filter_w)//stride + 1
# 对图像在4个维度进行填充,默认pad为0
img = np.pad(input_data, [(0,0), (0,0), (pad, pad), (pad, pad)], 'constant')
# 见下面解释1
col = np.zeros((N, C, filter_h, filter_w, out_h, out_w))
# 从左到右,从上到下依次进行遍历
for y in range(filter_h):
# 见解释2
y_max = y + stride*out_h
for x in range(filter_w):
x_max = x + stride*out_w
# 见解释3
col[:, :, y, x, :, :] = img[:, :, y:y_max:stride, x:x_max:stride]
print("x:", x, "y:", y, "x_max:", x_max, "y_max:", y_max)
print('col',y,x,':\n',col)
print("\ncol.transpose res is:\n",col.transpose(0, 4, 5, 1, 2, 3))
col = col.transpose(0, 4, 5, 1, 2, 3).reshape(N*out_h*out_w, -1)
return col

我们首先根据这幅图来进行模拟stride=1时候的情况:

input_data进行如下初始化:input_data = np.arange(12).reshape(1, 1, 3, 4)
input_data

col = im2col(input_data, 2, 2)
print('\ncol res is:\n', col)
我们得到
x: 0 y: 0 x_max: 3 y_max: 2
col 0 0 :
[[[[[[0. 1. 2.]
[4. 5. 6.]]
[[0. 0. 0.]
[0. 0. 0.]]]
[[[0. 0. 0.]
[0. 0. 0.]]
[[0. 0. 0.]
[0. 0. 0.]]]]]]
x: 1 y: 0 x_max: 4 y_max: 2
col 0 1 :
[[[[[[0. 1. 2.]
[4. 5. 6.]]
[[1. 2. 3.]
[5. 6. 7.]]]
[[[0. 0. 0.]
[0. 0. 0.]]
[[0. 0. 0.]
[0. 0. 0.]]]]]]
x: 0 y: 1 x_max: 3 y_max: 3
col 1 0 :
[[[[[[ 0. 1. 2.]
[ 4. 5. 6.]]
[[ 1. 2. 3.]
[ 5. 6. 7.]]]
[[[ 4. 5. 6.]
[ 8. 9. 10.]]
[[ 0. 0. 0.]
[ 0. 0. 0.]]]]]]
x: 1 y: 1 x_max: 4 y_max: 3
col 1 1 :
[[[[[[ 0. 1. 2.]
[ 4. 5. 6.]]
[[ 1. 2. 3.]
[ 5. 6. 7.]]]
[[[ 4. 5. 6.]
[ 8. 9. 10.]]
[[ 5. 6. 7.]
[ 9. 10. 11.]]]]]]
col.transpose res is:
[[[[[[ 0. 1.]
[ 4. 5.]]]
[[[ 1. 2.]
[ 5. 6.]]]
[[[ 2. 3.]
[ 6. 7.]]]]
[[[[ 4. 5.]
[ 8. 9.]]]
[[[ 5. 6.]
[ 9. 10.]]]
[[[ 6. 7.]
[10. 11.]]]]]]
col res is
[[ 0. 1. 4. 5.]
[ 1. 2. 5. 6.]
[ 2. 3. 6. 7.]
[ 4. 5. 8. 9.]
[ 5. 6. 9. 10.]
[ 6. 7. 10. 11.]]
col.transpose 之后,每个“块”中的内容便是卷积核要与img矩阵进行计算的内容
transpose(0, 4, 5, 1, 2, 3)最后再变换轴,变换后前三个维度刚好就是N*out_h*out_w这三个数。reshape(N*out_h*out_w, -1)这里是指第二个维度程序自己推理出来。N*out_h*out_w,表示为卷积核和原始输入数据的计算次数(也就是卷积窗口滑动了几次)
im2col运算之后,得到了形状为(6,4)的矩阵,其中6表示卷积核和原始输入数据的计算次数为6次,而4表示每次计算从输入数据中取出的值个数为4个执行:
col2=im2col(input_data, 2, 2, stride=2)
print('\ncol2 res is\n', col2)
输出:
x: 0 y: 0 x_max: 4 y_max: 2
col 0 0 :
[[[[[[0. 2.]]
[[0. 0.]]]
[[[0. 0.]]
[[0. 0.]]]]]]
x: 1 y: 0 x_max: 5 y_max: 2
col 0 1 :
[[[[[[0. 2.]]
[[1. 3.]]]
[[[0. 0.]]
[[0. 0.]]]]]]
x: 0 y: 1 x_max: 4 y_max: 3
col 1 0 :
[[[[[[0. 2.]]
[[1. 3.]]]
[[[4. 6.]]
[[0. 0.]]]]]]
x: 1 y: 1 x_max: 5 y_max: 3
col 1 1 :
[[[[[[0. 2.]]
[[1. 3.]]]
[[[4. 6.]]
[[5. 7.]]]]]]
col.transpose res is:
[[[[[[0. 1.]
[4. 5.]]]
[[[2. 3.]
[6. 7.]]]]]]
col2 res is
[[0. 1. 4. 5.]
[2. 3. 6. 7.]]
stride>1时候,虽然在循环的时候,矩阵img是跳跃式的取值,但是通过后面的transpose操作,成功的将结果变换回来了,只能感叹能写出这样程序的人,真的是思维太强了!!

到此,im2col函数实现的解释全部解说完毕,不知道为大家说清楚了么?

def col2im(col, input_shape, filter_h, filter_w, stride=1, pad=0):
"""
Parameters
----------
col :
input_shape : 输入数据的形状(例如:(10,1,28,28))
filter_h :
filter_w
stride
pad
Returns
-------
"""
N, C, H, W = input_shape
out_h = (H + 2*pad - filter_h)//stride + 1
out_w = (W + 2*pad - filter_w)//stride + 1
col = col.reshape(N, out_h, out_w, C, filter_h, filter_w).transpose(0, 3, 4, 5, 1, 2)
img = np.zeros((N, C, H + 2*pad + stride - 1, W + 2*pad + stride - 1))
for y in range(filter_h):
y_max = y + stride*out_h
for x in range(filter_w):
x_max = x + stride*out_w
img[:, :, y:y_max:stride, x:x_max:stride] += col[:, :, y, x, :, :]
return img[:, :, pad:H + pad, pad:W + pad]
我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re
我正在尝试用ruby中的gsub函数替换字符串中的某些单词,但有时效果很好,在某些情况下会出现此错误?这种格式有什么问题吗NoMethodError(undefinedmethod`gsub!'fornil:NilClass):模型.rbclassTest"replacethisID1",WAY=>"replacethisID2andID3",DELTA=>"replacethisID4"}end另一个模型.rbclassCheck 最佳答案 啊,我找到了!gsub!是一个非常奇怪的方法。首先,它替换了字符串,所以它实际上修改了
我有一个用户工厂。我希望默认情况下确认用户。但是鉴于unconfirmed特征,我不希望它们被确认。虽然我有一个基于实现细节而不是抽象的工作实现,但我想知道如何正确地做到这一点。factory:userdoafter(:create)do|user,evaluator|#unwantedimplementationdetailshereunlessFactoryGirl.factories[:user].defined_traits.map(&:name).include?(:unconfirmed)user.confirm!endendtrait:unconfirmeddoenden
我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin
我正在阅读SandiMetz的POODR,并且遇到了一个我不太了解的编码原则。这是代码:classBicycleattr_reader:size,:chain,:tire_sizedefinitialize(args={})@size=args[:size]||1@chain=args[:chain]||2@tire_size=args[:tire_size]||3post_initialize(args)endendclassMountainBike此代码将为其各自的属性输出1,2,3,4,5。我不明白的是查找方法。当一辆山地自行车被实例化时,因为它没有自己的initialize方法
如何在Ruby中按名称传递函数?(我使用Ruby才几个小时,所以我还在想办法。)nums=[1,2,3,4]#Thisworks,butismoreverbosethanI'dlikenums.eachdo|i|putsiend#InJS,Icouldjustdosomethinglike:#nums.forEach(console.log)#InF#,itwouldbesomethinglike:#List.iternums(printf"%A")#InRuby,IwishIcoulddosomethinglike:nums.eachputs在Ruby中能不能做到类似的简洁?我可以只
华为OD机试题本篇题目:明明的随机数题目输入描述输出描述:示例1输入输出说明代码编写思路最近更新的博客华为od2023|什么是华为od,od薪资待遇,od机试题清单华为OD机试真题大全,用Python解华为机试题|机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为o
C#实现简易绘图工具一.引言实验目的:通过制作窗体应用程序(C#画图软件),熟悉基本的窗体设计过程以及控件设计,事件处理等,熟悉使用C#的winform窗体进行绘图的基本步骤,对于面向对象编程有更加深刻的体会.Tutorial任务设计一个具有基本功能的画图软件**·包括简单的新建文件,保存,重新绘图等功能**·实现一些基本图形的绘制,包括铅笔和基本形状等,学习橡皮工具的创建**·设计一个合理舒适的UI界面**注明:你可能需要先了解一些关于winform窗体应用程序绘图的基本知识,以及关于GDI+类和结构的知识二.实验环境Windows系统下的visualstudio2017C#窗体应用程序三.
说在前面这部分我本来是合为一篇来写的,因为目的是一样的,都是通过独立按键来控制LED闪灭本质上是起到开关的作用,即调用函数和中断函数。但是写一篇太累了,我还是决定分为两篇写,这篇是调用函数篇。在本篇中你主要看到这些东西!!!1.调用函数的方法(主要讲语法和格式)2.独立按键如何控制LED亮灭3.程序中的一些细节(软件消抖等)1.调用函数的方法思路还是比较清晰地,就是通过按下按键来控制LED闪灭,即每按下一次,LED取反一次。重要的是,把按键与LED联系在一起。我打算用K1来作为开关,看了一下开发板原理图,K1连接的是单片机的P31口,当按下K1时,P31是与GND相连的,也就是说,当我按下去时
MIMO技术的优缺点优点通过下面三个增益来总体概括:阵列增益。阵列增益是指由于接收机通过对接收信号的相干合并而活得的平均SNR的提高。在发射机不知道信道信息的情况下,MIMO系统可以获得的阵列增益与接收天线数成正比复用增益。在采用空间复用方案的MIMO系统中,可以获得复用增益,即信道容量成倍增加。信道容量的增加与min(Nt,Nr)成正比分集增益。在采用空间分集方案的MIMO系统中,可以获得分集增益,即可靠性性能的改善。分集增益用独立衰落支路数来描述,即分集指数。在使用了空时编码的MIMO系统中,由于接收天线或发射天线之间的间距较远,可认为它们各自的大尺度衰落是相互独立的,因此分布式MIMO