Cblas_sgemm 产生了错误的结果
Cblas_sgemm produces wrong results
我在使用 cblas 的 sgemm 函数时遇到问题。
代码如下:
#include <math.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <cblas.h>
#define MATRIX_DIM 5
int main(){
float *a_mat = calloc(MATRIX_DIM*MATRIX_DIM, sizeof(float));
float *b_mat = calloc(MATRIX_DIM, sizeof(float));
float *c_mat = calloc(MATRIX_DIM, sizeof(float));
int i,j;
for(i=0; i<MATRIX_DIM*MATRIX_DIM; i++) {
a_mat[i] = 1.0f;
b_mat[i] = 1.0f;
c_mat[i] = 0.0f;
}
cblas_sgemm(CblasRowMajor, CblasNoTrans,
CblasNoTrans, MATRIX_DIM, MATRIX_DIM,
MATRIX_DIM, 1.0, a_mat,
MATRIX_DIM, b_mat, MATRIX_DIM,
1.0, c_mat, MATRIX_DIM);
//RESULT PRINTING
printf("Printing A MATRIX: \n");
for(i=0; i<MATRIX_DIM; i++) {
for(j=0; j<MATRIX_DIM; j++){
printf("%0.1f ", a_mat[i*MATRIX_DIM+j]);
}
printf("\n");
}
printf("Printing B MATRIX: \n");
for(i=0; i<MATRIX_DIM; i++) {
for(j=0; j<MATRIX_DIM; j++){
printf("%0.1f ", b_mat[i*MATRIX_DIM+j]);
}
printf("\n");
}
printf("\nPrinting the Results: \n");
for(i=0; i<MATRIX_DIM;i++){
for(j=0; j<MATRIX_DIM; j++){
printf("%0.1f ", c_mat[i*MATRIX_DIM+j]);
}
printf("\n");
}
free(a_mat);
free(b_mat);
free(c_mat);
return 0;
}
我很清楚我提出的一些论点是错误的,但我真的不知道是哪个。结果应该是一个用 5.0 填充的 5x5 矩阵。相反,程序会这样响应:
6.0 6.0 6.0 16.0 86.0
6.0 6.0 6.0 16.0 86.0
16.0 36.0 6.0 46.0 86.0
16.0 36.0 5.0 45.0 85.0
20.0 80.0 5.0 45.0 85.0
我知道 rowmajor 顺序或转置参数可能是错误的,我稍后会解决这些问题,但在这个特定的乘法中,无论哪种方式,答案都应该是 5.0。
感谢@AndrasDeak 的评论,我所需要的只是在两个矩阵上分配更多 space,这是我之前忽略的。
所以基本上改变:
float *b_mat = calloc(MATRIX_DIM, sizeof(float));
float *c_mat = calloc(MATRIX_DIM, sizeof(float));
收件人:
float *b_mat = calloc(MATRIX_DIM*MATRIX_DIM, sizeof(float));
float *c_mat = calloc(MATRIX_DIM*MATRIX_DIM, sizeof(float));
因为这些应该是二维矩阵而不是向量。
我在使用 cblas 的 sgemm 函数时遇到问题。
代码如下:
#include <math.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <cblas.h>
#define MATRIX_DIM 5
int main(){
float *a_mat = calloc(MATRIX_DIM*MATRIX_DIM, sizeof(float));
float *b_mat = calloc(MATRIX_DIM, sizeof(float));
float *c_mat = calloc(MATRIX_DIM, sizeof(float));
int i,j;
for(i=0; i<MATRIX_DIM*MATRIX_DIM; i++) {
a_mat[i] = 1.0f;
b_mat[i] = 1.0f;
c_mat[i] = 0.0f;
}
cblas_sgemm(CblasRowMajor, CblasNoTrans,
CblasNoTrans, MATRIX_DIM, MATRIX_DIM,
MATRIX_DIM, 1.0, a_mat,
MATRIX_DIM, b_mat, MATRIX_DIM,
1.0, c_mat, MATRIX_DIM);
//RESULT PRINTING
printf("Printing A MATRIX: \n");
for(i=0; i<MATRIX_DIM; i++) {
for(j=0; j<MATRIX_DIM; j++){
printf("%0.1f ", a_mat[i*MATRIX_DIM+j]);
}
printf("\n");
}
printf("Printing B MATRIX: \n");
for(i=0; i<MATRIX_DIM; i++) {
for(j=0; j<MATRIX_DIM; j++){
printf("%0.1f ", b_mat[i*MATRIX_DIM+j]);
}
printf("\n");
}
printf("\nPrinting the Results: \n");
for(i=0; i<MATRIX_DIM;i++){
for(j=0; j<MATRIX_DIM; j++){
printf("%0.1f ", c_mat[i*MATRIX_DIM+j]);
}
printf("\n");
}
free(a_mat);
free(b_mat);
free(c_mat);
return 0;
}
我很清楚我提出的一些论点是错误的,但我真的不知道是哪个。结果应该是一个用 5.0 填充的 5x5 矩阵。相反,程序会这样响应:
6.0 6.0 6.0 16.0 86.0
6.0 6.0 6.0 16.0 86.0
16.0 36.0 6.0 46.0 86.0
16.0 36.0 5.0 45.0 85.0
20.0 80.0 5.0 45.0 85.0
我知道 rowmajor 顺序或转置参数可能是错误的,我稍后会解决这些问题,但在这个特定的乘法中,无论哪种方式,答案都应该是 5.0。
感谢@AndrasDeak 的评论,我所需要的只是在两个矩阵上分配更多 space,这是我之前忽略的。
所以基本上改变:
float *b_mat = calloc(MATRIX_DIM, sizeof(float));
float *c_mat = calloc(MATRIX_DIM, sizeof(float));
收件人:
float *b_mat = calloc(MATRIX_DIM*MATRIX_DIM, sizeof(float));
float *c_mat = calloc(MATRIX_DIM*MATRIX_DIM, sizeof(float));
因为这些应该是二维矩阵而不是向量。