C中实现矩阵乘法的一种高效的方法

缘雨5

缘雨5

2016-02-19 09:52

今天图老师小编要跟大家分享C中实现矩阵乘法的一种高效的方法,精心挑选的过程简单易学,喜欢的朋友一起来学习吧!

如何计算矩阵乘法,这个大家都知道。通常情况下,我们都是用以下代码实现的
代码如下:

for(i=0;in;++i)
    for(j=0;jn;++j){
        sum=0;
        for(k=0;kn;++k)
            sum+=A[i][k]*B[k][j];
        C[i][j]+=sum;
}

但是考虑了高速缓存的问题后,其实有一种更好的实现方式:
代码如下:

(本文来源于图老师网站,更多请访问http://m.tulaoshi.com/bianchengyuyan/)

for(i=0;in;++i)
    for(k=0;kn;++k){
        r=A[i][k];
        for(j=0;jn;++j)
            C[i][j]+=r*B[k][j];
}


细看一番就会发现这两种实现语义是等价的,但是后者的实际运行效率却比前者高。

那为什么会如此呢?

那是因为CPU读数据时,并不是直接访问内存,而是先查看缓存中是否有数据,有的话直接从缓存读取。而从缓存读取数据比从内存读数据快很多。

当数据不在缓存中时,CPU会将包含数据在内的一个数据块读到缓存,如果程序具有良好空间局部性,那么第一次cache miss后,之后的几次数据访问就可以直接在缓存中完成。除了空间局部性(程序倾向于引用与当前数据邻近的数据)之外,还有时间局部性(程序倾向于引用最近被引用过的数据)。

回到矩阵乘法。(我们只考虑内循环)

前者对矩阵A,有良好的空间局部性,假设一次能缓存四个元素,则每次迭代对于A只有0.25次miss,但是对于B,则不然,因此B是按列访问的,每次访问都会miss,因此每次迭代总的miss数是1.25。

后者对于矩阵C和矩阵B都有良好的局部性,每次迭代都只有0.25词miss,因此总的miss数是0.5。后者每次迭代多了一次存储(对C[i][j]写入),但是即便如此,后者的运行效率也比前者高。

(本文来源于图老师网站,更多请访问http://m.tulaoshi.com/bianchengyuyan/)

总而言之,要想程序跑得快,就要在程序中多利用局部性,让缓存hold住你的数据,减少访存次数。要知道CPU可以在3个时钟周期内访问到L1 cache,10个时钟周期左右的时间访问到L2 cache。访问内存却要上百个时钟周期,孰快孰慢,很清楚了吧?

展开更多 50%)
分享

猜你喜欢

C中实现矩阵乘法的一种高效的方法

编程语言 网络编程
C中实现矩阵乘法的一种高效的方法

java 二维数组矩阵乘法的实现方法

编程语言 网络编程
java 二维数组矩阵乘法的实现方法

s8lol主宰符文怎么配

英雄联盟 网络游戏
s8lol主宰符文怎么配

vb实现图像缩放的一种方法

编程语言 网络编程
vb实现图像缩放的一种方法

C++消息连接的一种系统方法

编程语言 网络编程
C++消息连接的一种系统方法

lol偷钱流符文搭配推荐

英雄联盟 网络游戏
lol偷钱流符文搭配推荐

网站登录记忆跳转实现的一种方法

Web开发
网站登录记忆跳转实现的一种方法

Linux下Socket连接超时的一种实现方法

Linux Linux命令 Linux安装 Linux编程 Linux桌面 Linux软件 Linux内核 Linux管理
Linux下Socket连接超时的一种实现方法

lolAD刺客新符文搭配推荐

英雄联盟
lolAD刺客新符文搭配推荐

用正则删除不包含某个字符串的行的代码

用正则删除不包含某个字符串的行的代码

使用正则表达式替换表情符号核心代码

使用正则表达式替换表情符号核心代码
下拉加载更多内容 ↓