我几天来一直在研究类似的问题,但仍未找到解决方案。谢谢任何帮助:
我有两个文件,一个包含处理有理数的方法,另一个用二维数组处理它们。我的问题是matrix.c不能识别fraction.c中包含的分数结构。我相信我的问题与我宣布我的二维数组的方式有某种关系。
在fraction.c中:
struct fraction {
int integer;
int num;
int den;
};
typedef struct fraction* fractionRef; //This line is in fraction.h
在matrix.c中:
#include "fraction.h"
typedef struct matrix* matrixRef;
struct matrix {
int rows;
int columns;
fractionRef *m;
}matrix;
matrixRef new_matrix ( int rows, int columns ) {
matrixRef matrix;
matrix = (matrixRef)malloc( sizeof( matrix ) );
matrix->m = (fractionRef*)calloc( rows, sizeof( fractionRef ) );
int i;
for ( i=0; i<=rows; i++ )
matrix->m[i] = (fractionRef)calloc( columns, sizeof( fractionRef ) );
assert( matrix->m );
return matrix;
}
void free_matrix ( matrixRef freeMe ) {
if ( freeMe != NULL ){
int i, j;
for( i = 0; i <= freeMe->rows; i++ ){
for ( j = 0; j <= freeMe->columns; j++ ){
free_fraction( freeMe->m[i][j] ); //ERROR OCCURS HERE
}
}
freeMe->rows = 0;
freeMe->columns = 0;
free(freeMe);
freeMe = NULL;
}
}
我得到的错误对应于matrix.c中标记的行。
matrix.c:47: error: invalid use of undefined type ‘struct fraction’
matrix.c:47: error: dereferencing pointer to incomplete type
这可能都是因为我学会了java BEFORE c,大错!再次感谢您的帮助。
编辑:谢谢大家。所以我现在看到它的方式是标题.h文件中的所有内容都类似于java中的public。我的分数结构定义不是c编译器的“公共”,所以我的matrix.c无法访问它。答案 0 :(得分:10)
您必须将struct
的定义移至fraction.h
文件中。与Java不同,编译器和链接器不会“神奇地”导致一个.c
文件引用另一个.c
文件中的信息。 (如果你在另一个中#include
一个.c文件就可以做到 - 这是一个坏主意。)
指令
#include“fraction.h”
导致头文件的文本内容(就像通过剪切和粘贴一样)放在具有该指令的行上。编译器一次处理一个输入文件,读入#include
'd文件,并且在编译一个.c
文件时必须存在所有必需的信息。
为了帮助您理解,我将指出一种可行的方法来完成您的需求:只需将struct fraction
的结构定义剪切并粘贴到matrix.c
的顶部,紧接着#include "fraction.h"
- 结果将编译。在Java中,编译器可能会抱怨您已声明了一些重复类型。在C中,您所做的是定义两个不同的结构,它们恰好具有相同的内存布局和相同的名称。从链接目标文件的角度来看,只需要相同的内存布局就可以使它们互换。
是的,这是你从Java那里得到的误解。你学习C语真是太棒了!
答案 1 :(得分:6)
将struct fraction
定义移至“fraction.h”和
#include "fraction.h"
在“fraction.c”和“matrix.c”
中答案 2 :(得分:3)
很难猜出你的意图,但我猜你需要传递指向free_fraction的指针。
free_fraction( &(freeMe->m[i][j]) );
答案 3 :(得分:2)
在那里,您可能还想更改定义。
更好的方式,让您了解正在发生的事情:
struct fraction {
int integer;
int num;
int den;
};
typedef struct fraction fractionType; // We can now use fractionType and it'll be struct fraction
typedef fractionType * fractionTypePtr; // We can now use fractionTypePtr and it will be a pointer to a fractionType which is a struct fraction
答案 4 :(得分:1)
或者,如果你想完全隐藏分数结构的实现,你可以在fraction.h中使用:
typedef struct fraction* fractionRef;
和fraction.c文件中结构分数的实际声明。
然而,当您隐藏实现时,您还必须为分数上的所有操作定义函数,例如,获取分子和分母的值,即编写如下内容:
get_numerator(f);
而不是:
f->num;