Библиотека переопределяет NULL - PullRequest
1 голос
/ 30 ноября 2009

Я работаю с библиотекой, которая переопределяет NULL. Это вызывает некоторые проблемы с другими частями моей программы. Я не уверен, что я могу с этим поделать. Любая идея? Моя программа на C ++, библиотека на C .

#ifdef NULL
#undef NULL
#endif

/**
 * NULL define.
 */
#define NULL    ((void *) 0)

О, и он выдает следующие ошибки:

Generic.h:67: error: default argument for parameter of type 'LCD::LCDBase*' has type 'void*'
Generic.cpp: In constructor 'LCD::Generic::Generic(std::string, Json::Value*, int, LCD::LCDBase*)':
Generic.cpp:44: error: invalid conversion from 'void*' to 'QObject*'
Generic.cpp:44: error:   initializing argument 2 of 'LCD::LCDWrapper::LCDWrapper(LCD::LCDInterface*, QObject*)'
Generic.cpp: In member function 'void LCD::Generic::BuildLayouts()':
Generic.cpp:202: error: invalid conversion from 'void*' to 'LCD::Widget*'
Generic.cpp: In member function 'void LCD::Generic::AddWidget(std::string, unsigned int, unsigned int, std::string)':
Generic.cpp:459: error: invalid conversion from 'void*' to 'LCD::Widget*'
scons: *** [Generic.o] Error 1

Вот первый:

Generic(std::string name, Json::Value *config, int type, LCDBase *lcd = NULL);

Редактировать: Хорошо, приведение явно работает, но как преобразовать указатель на функцию?

Ответы [ 5 ]

5 голосов
/ 30 ноября 2009

Можете ли вы перестроить библиотеку без этого определения? Это то, что я попробую в первую очередь. NULL - довольно стандартный макрос, и предполагается, что он определен везде.

В данный момент ваша проблема в том, что C ++ не разрешает автоматическое приведение от void * к другим типам указателей, как это делает C.

Из C ++ Ссылка :

В C ++ NULL расширяется до 0 или 0L.

Если это не сработает, просто выполните глобальную замену в библиотеке: от NULL до LIBDEFINEDNULL или что-то в этом роде. Таким образом вы сохраните код библиотеки в целости и избежите конфликта макросов.

3 голосов
/ 30 ноября 2009

В одном из ваших комментариев говорится, что вы рассматривали возможность его переопределения самостоятельно, но не знаете, к чему его переопределить.

Многие реализации будут определять NULL следующим образом:

#undef NULL

#ifdef __cplusplus
#define NULL 0
#else
#define NULL ((void*)0)
#endif

Это потому, что в C имеет смысл иметь его как указатель void, потому что это тип указателя, который можно неявно приводить к другим типам.

C ++ не позволяет этого (что вызывает ваши проблемы), но использование 0 вместо NULL работает.

Я думаю, что во всех последних версиях GCC определит его как __null, который является непереносимым расширением.

3 голосов
/ 30 ноября 2009

Наиболее общим подходом было бы обернуть оскорбительные включения и сохранить и восстановить предыдущее определение макроса. Это, однако, зависит от компилятора.
Вот как вы могли бы сделать это с VC:

#pragma push_macro("NULL")
#include <offendinglib.h>
#pragma pop_macro("NULL")

В качестве альтернативы, установите макрос таким, каким он должен быть после:

#include <offendinglib.h>
#undef NULL
#define NULL 0
3 голосов
/ 30 ноября 2009

У вас есть доступ к этому источнику библиотек? Если это так, я думаю, что поиск и замена по их коду в порядке. (Замените их NULL на LIBNAME_NULL или что-то подобное.) Если это просто не вариант, я бы рекомендовал использовать 0 в вашем коде вместо NULL.

Мне любопытно: какие проблемы это вызывает? Они не изменяют значение null, только приведение по умолчанию.

0 голосов
/ 30 ноября 2009

Да, вам просто нужно правильно разыграть:

Generic(std::string name, Json::Value *config, int type,
    LCDBase *lcd = (LCDBase *)NULL);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...