C: «недопустимое использование неопределенного типа« struct X »и разыменование указателя на неполный тип» ошибки - PullRequest
6 голосов
/ 10 мая 2011

Я уже пару дней просматривал подобные вопросы, но пока не нашел решения. Спасибо за любую помощь:

У меня есть два файла, один из которых содержит методы для работы с рациональными числами, а другой обрабатывает их в двумерном массиве. Моя проблема заключается в том, что matrix.c не распознает структуру дроби, которая содержится в fra.c. Я считаю, что моя проблема как-то связана с тем, как я объявил свой 2d массив.

В дроби.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 ДО c, большая ошибка !!! Еще раз спасибо за любую помощь.

РЕДАКТИРОВАТЬ: Спасибо всем. Поэтому, как я вижу это сейчас, все в заголовочных файлах .h похоже на public в java. Мое определение структуры дроби не было «публичным» для компилятора c, поэтому мой matrix.c не смог получить к нему доступ.

Ответы [ 5 ]

10 голосов
/ 10 мая 2011

Вы должны переместить определение struct в файл fraction.h.В отличие от Java, компилятор и компоновщик не «волшебным образом» заставляют один файл .c ссылаться на информацию внутри другого файла .c.(Это может быть сделано, если вы #include один файл .c в другом - плохая идея.)

Директива

# включает в себя «дробный.h"

вызывает текстовое содержимое файла заголовка, как если бы оно было вырезано и вставлено, в строку с этой директивой.Компилятор обрабатывает один входной файл за раз, считывая файлы #include 'd, и вся необходимая информация должна присутствовать, пока компилируется один файл .c.

Чтобы помочь вам понять,Я укажу ужасный способ выполнить то, что вам нужно: просто вырежьте и вставьте определение структуры struct fraction в верхнюю часть matrix.c, непосредственно перед #include "fraction.h" - результат скомпилируется.В Java компилятор может жаловаться, что вы объявили несколько дублирующих типов.В C вы определили две разные структуры, которые имеют одинаковую структуру памяти и одно и то же имя.Только одна и та же схема памяти необходима для того, чтобы сделать их взаимозаменяемыми с точки зрения связывания объектных файлов.

Да, это недоразумение, которое вы уловили в Java.Замечательно, что ты учишь C!

6 голосов
/ 10 мая 2011

Переместите определение struct fraction в "дробный.ч" и

#include "fraction.h"

как в "дробном.c", так и в "matrix.c"

3 голосов
/ 10 мая 2011

Трудно угадать ваши намерения, но я предполагаю, что вам нужно передать указатель на free_fraction.

free_fraction( &(freeMe->m[i][j]) );
2 голосов
/ 10 мая 2011

Находясь там, вы также можете изменить свои определения.

Лучше, чтобы вы понимали, что происходит:

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
1 голос
/ 10 мая 2011

В качестве альтернативы, если вы хотите полностью скрыть реализацию своей структуры дроби, вы можете получить в дробном .h:

typedef struct fraction* fractionRef;

и фактическое объявление вашей структурной дроби в файле fra.c.

Однако, поскольку вы скрываете реализацию, вы также должны определить функцию для всех действий, которые вы можете выполнить над дробью, например, получить значение числителя и знаменателя, т.е. написать такие вещи, как:

get_numerator(f);

вместо:

f->num;
...