前言:

  很多朋友看到我写的《算法导论》系列,可能会觉得云里雾里,不知所云。这里我再次说明,本系列博文时配合《算法导论》一书,给出该书涉及的算法的c++实现。请结合《算法导论》一书阅读该系列博文。我这里有该书的电子版,有需要的朋友可以留言。

正题:

  今天讨论的算法是矩阵乘法的Strassen算法,该算法的精髓在于减少n/2矩阵*n/2矩阵的次数。首先,作一些写该算法的基础工作:

/*
* 矩阵的加法运算
*/
void Add(int** matrixA, int** matrixB, int** matrixResult,int length)
{
for(int i = ; i < length; i++) {
for(int j = ; j < length; j++) {
matrixResult[i][j] = matrixA[i][j] + matrixB[i][j];
}
}
} /*
* 矩阵的减法运算
*/
void Sub(int** matrixA, int** matrixB, int** matrixResult,int length)
{
for(int i = ; i < length; i++) {
for(int j = ; j < length; j++) {
matrixResult[i][j] = matrixA[i][j] - matrixB[i][j];
}
}
} /*
* 矩阵乘法
*/
void Mul(int** matrixA, int** matrixB, int** matrixResult){
for(int i = ; i < ; ++i) {
for(int j = ; j < ; ++j) {
matrixResult[i][j] = ;
for(int k = ; k < ; ++k) {
matrixResult[i][j] += matrixA[i][k] * matrixB[k][j];
}
}
}
}

接着进入核心部分:

void Strassen(int** matrixA, int** matrixB, int** matrixResult,int length)
{
int halfLength=length/;
int** a11=new int*[halfLength];
int** a12=new int*[halfLength];
int** a21=new int*[halfLength];
int** a22=new int*[halfLength]; int** b11=new int*[halfLength];
int** b12=new int*[halfLength];
int** b21=new int*[halfLength];
int** b22=new int*[halfLength]; int** s1=new int*[halfLength];
int** s2=new int*[halfLength];
int** s3=new int*[halfLength];
int** s4=new int*[halfLength];
int** s5=new int*[halfLength];
int** s6=new int*[halfLength];
int** s7=new int*[halfLength]; int** matrixResult11=new int*[halfLength];
int** matrixResult12=new int*[halfLength];
int** matrixResult21=new int*[halfLength];
int** matrixResult22=new int*[halfLength]; int** temp=new int*[halfLength];
int** temp1=new int*[halfLength];
if(halfLength==){
Mul(matrixA, matrixB, matrixResult);
}else{
//首先将矩阵A,B 分为4块
for(int i = ; i < halfLength; i++) {
a11[i]=new int[halfLength];
a12[i]=new int[halfLength];
a21[i]=new int[halfLength];
a22[i]=new int[halfLength]; b11[i]=new int[halfLength];
b12[i]=new int[halfLength];
b21[i]=new int[halfLength];
b22[i]=new int[halfLength]; s1[i]=new int[halfLength];
s2[i]=new int[halfLength];
s3[i]=new int[halfLength];
s4[i]=new int[halfLength];
s5[i]=new int[halfLength];
s6[i]=new int[halfLength];
s7[i]=new int[halfLength]; matrixResult11[i]=new int[halfLength];
matrixResult12[i]=new int[halfLength];
matrixResult21[i]=new int[halfLength];
matrixResult22[i]=new int[halfLength]; temp[i]=new int[halfLength];
temp1[i]=new int[halfLength];
for(int j = ; j < halfLength; j++) {
a11[i][j]=matrixA[i][j];
a12[i][j]=matrixA[i][j+halfLength];
a21[i][j]=matrixA[i+halfLength][j];
a22[i][j]=matrixA[i+halfLength][j+halfLength];
b11[i][j]=matrixB[i][j];
b12[i][j]=matrixB[i][j+halfLength];
b21[i][j]=matrixB[i+halfLength][j];
b22[i][j]=matrixB[i+halfLength][j+halfLength];
}
} //计算s1
Sub(b12, b22, temp,halfLength);
Strassen(a11, temp, s1,halfLength);
//计算s2
Add(a11, a12, temp,halfLength);
Strassen(temp, b22, s2,halfLength);
//计算s3
Add(a21, a22, temp,halfLength);
Strassen(temp, b11, s3,halfLength);
//计算s4
Sub(b21, b11, temp,halfLength);
Strassen(a22, temp, s4,halfLength);
//计算s5
Add(a11, a22, temp1,halfLength);
Add(b11, b22, temp,halfLength);
Strassen(temp1, temp, s5,halfLength);
//计算s6
Sub(a12, a22, temp1,halfLength);
Add(b21, b22, temp,halfLength);
Strassen(temp1, temp, s6,halfLength);
//计算s7
Sub(a11, a21, temp1,halfLength);
Add(b11, b12, temp,halfLength);
Strassen(temp1, temp, s7,halfLength); //计算matrixResult11
Add(s5, s4, temp1,halfLength);
Sub(temp1, s2, temp,halfLength);
Add(temp, s6, matrixResult11,halfLength);
//计算matrixResult12
Add(s1, s2, matrixResult12,halfLength);
//计算matrixResult21
Add(s3, s4, matrixResult21,halfLength);
//计算matrixResult22
Add(s5, s1, temp1,halfLength);
Sub(temp1, s3, temp,halfLength);
Sub(temp, s7, matrixResult22,halfLength); //结果送回matrixResult中
for(int i = ; i < halfLength; i++) {
for(int j = ; j < halfLength; j++) {
matrixResult[i][j]=matrixResult11[i][j];
matrixResult[i][j+halfLength]=matrixResult12[i][j];
matrixResult[i+halfLength][j]=matrixResult21[i][j];
matrixResult[i+halfLength][j+halfLength]=matrixResult22[i][j];
}
delete(a11[i]);
delete(a12[i]);
delete(a21[i]);
delete(a22[i]); delete(b11[i]);
delete(b12[i]);
delete(b21[i]);
delete(b22[i]); delete(s1[i]);
delete(s2[i]);
delete(s3[i]);
delete(s4[i]);
delete(s5[i]);
delete(s6[i]);
delete(s7[i]); delete(matrixResult11[i]);
delete(matrixResult12[i]);
delete(matrixResult21[i]);
delete(matrixResult22[i]); delete(temp[i]);
delete(temp1[i]);
}
delete(a11);
delete(a12);
delete(a21);
delete(a22); delete(b11);
delete(b12);
delete(b21);
delete(b22); delete(s1);
delete(s2);
delete(s3);
delete(s4);
delete(s5);
delete(s6);
delete(s7); delete(matrixResult11);
delete(matrixResult12);
delete(matrixResult21);
delete(matrixResult22); delete(temp);
delete(temp1);
}
}

该算法看着或许有些冗长,几乎一半都在进行动态指针的初始化和删除。利用该算法计算矩阵乘的时间复杂度为θ(n^lg7)。

测试一下吧:

#include "stdafx.h"
#include <iostream>
#include "SquareMatrix.h" using namespace std;
using namespace dksl; //STRASSEN矩阵乘法算法 const int N=; //常量N用来定义矩阵的大小
int _tmain(int argc, _TCHAR* argv[])
{
int **a=new int*[];
int **b=new int*[];
int **c=new int*[];
for(int i=;i<;i++)
{
a[i]=new int[];
b[i]=new int[];
c[i]=new int[];
for(int j=;j<;j++)
{
a[i][j]=;
b[i][j]=;
}
}
Strassen(a,b,c,);
for(int i=;i<;i++)
{
for(int j=;j<;j++)
cout<<c[i][j]<<" ";
cout<<endl;
}
system("PAUSE");
return ;
}

《算法导论》——矩阵乘法的Strassen算法的更多相关文章

  1. 4-2.矩阵乘法的Strassen算法详解

    题目描述 请编程实现矩阵乘法,并考虑当矩阵规模较大时的优化方法. 思路分析 根据wikipedia上的介绍:两个矩阵的乘法仅当第一个矩阵B的列数和另一个矩阵A的行数相等时才能定义.如A是m×n矩阵和B ...

  2. 【算法导论C++代码】Strassen算法

    简单方阵矩乘法 SQUARE-MATRIX-MULTIPLY(A,B) n = A.rows let C be a new n*n natrix to n to n cij = to n cij=ci ...

  3. 算法导论-矩阵乘法-strassen算法

    目录 1.矩阵相乘的朴素算法 2.矩阵相乘的strassen算法 3.完整测试代码c++ 4.性能分析 5.参考资料 内容 1.矩阵相乘的朴素算法 T(n) = Θ(n3) 朴素矩阵相乘算法,思想明了 ...

  4. 第四章 分治策略 4.2 矩阵乘法的Strassen算法

    package chap04_Divide_And_Conquer; import static org.junit.Assert.*; import java.util.Arrays; import ...

  5. 【算法导论】--分治策略Strassen算法(运用下标运算)【c++】

    由于偷懒不想用泛型,所以直接用了整型来写了一份 ①首先你得有一个矩阵的class Matrix ②Matrix为了方便用下标进行运算, Matrix的结构如图:(我知道我的字丑...) Matrix. ...

  6. 算法笔记_081:蓝桥杯练习 算法提高 矩阵乘法(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 有n个矩阵,大小分别为a0*a1, a1*a2, a2*a3, ..., a[n-1]*a[n],现要将它们依次相乘,只能使用结合率,求最 ...

  7. 蓝桥 ADV-232 算法提高 矩阵乘法 【区间DP】

      算法提高 矩阵乘法   时间限制:3.0s   内存限制:256.0MB      问题描述 有n个矩阵,大小分别为a0*a1, a1*a2, a2*a3, ..., a[n-1]*a[n],现要 ...

  8. Java实现 蓝桥杯 算法提高 矩阵乘法(暴力)

    试题 算法提高 矩阵乘法 问题描述 小明最近刚刚学习了矩阵乘法,但是他计算的速度太慢,于是他希望你能帮他写一个矩阵乘法的运算器. 输入格式 输入的第一行包含三个正整数N,M,K,表示一个NM的矩阵乘以 ...

  9. Java实现 蓝桥杯 算法训练 矩阵乘法

    算法训练 矩阵乘法 时间限制:1.0s 内存限制:512.0MB 提交此题 问题描述 输入两个矩阵,分别是ms,sn大小.输出两个矩阵相乘的结果. 输入格式 第一行,空格隔开的三个正整数m,s,n(均 ...

随机推荐

  1. Map、Set、List集合差别及联系详解

    提到集合之前,先说说数组Array和集合的区别:   (1)数组是大小固定的,并且同一个数组只能存放类型一样的数据(基本类型/引用类型)   (2)JAVA集合可以存储和操作数目不固定的一组数据. ( ...

  2. 蓝牙协议分析(3)_BLE协议栈介绍

    1. 前言 通过“蓝牙协议分析(2)_协议架构”的介绍,大家对蓝牙协议栈应该有了简单的了解,但是,肯定还有“似懂非懂.欲说还休”的感觉.有这种感觉太正常了,毕竟蓝牙协议是一个历史悠久又比较庞大的协议, ...

  3. Js/使用js来改变css的样式

    1.一般来说我最先想到的是,通过id的方式去改变css的样式,所以有了下面这种写法: 2.第二种我想到的办法是通过改变他的class的名称,去设置 他的样式,其中用的比较多的就是这样的:

  4. 洛谷 P1763 状态压缩dp+容斥原理

    (题目来自洛谷oj) 一天,maze决定对自己的一块n*m的土地进行修建.他希望这块土地共n*m个格子的高度分别是1,2,3,...,n*m-1,n*m.maze又希望能将这一些格子中的某一些拿来建蓄 ...

  5. IC卡_状态码SW1、SW2

    9000 正常 成功执行 61xx 正常 需要发GET RESPONSE命令 6200 警告 信息未提供 6281 警告 回送数据可能出错 6282 警告 文件长度小于Le 6283 警告 选中的文件 ...

  6. Reading Level Assessment Using Support Vector Machines and Statistical Language Models-paper

    Authors: Sarah E. Schwarm University of Washington, Seattle, WAMari Ostendorf University of Washingt ...

  7. 求数组的相邻子数组的最大值(txt文件存储)

    package mypackage; import java.io.BufferedReader; import java.io.BufferedWriter; import java.io.File ...

  8. jquery中filter的用法

    一.filter的参数类型可分为两种 1.传递选择器 $('a').filter('.external') 2.传递过滤函数 $('a').filter(function(index) {       ...

  9. PythonStudy——文件操作习题 Document operation exercises

    # 1.统计文件数据中字母e出现的次数(不区分大小写)# 文件内容:hello friend, can you speak English!# 结果:4# 分析:将文件内容读出,然后统计读出的字符串中 ...

  10. Java高级特性 第4节 输入输出流

    一.使用I/O操作文件 关键步骤: 使用File类操作文件或目录属性 使用FileInputStream类读文本文件 使用FileOutputStram类写文本文件 使用BufferedReader类 ...