Как я могу утверждать () без использования abort ()? - PullRequest
24 голосов
/ 01 сентября 2008

Если я использую assert() и утверждение не выполняется, то assert() вызовет abort(), что приведет к внезапному завершению работающей программы. Я не могу позволить себе это в моем производственном коде. Есть ли способ утверждать во время выполнения, но иметь возможность отлавливать ошибочные утверждения, поэтому у меня есть возможность обработать их изящно?

Ответы [ 6 ]

24 голосов
/ 01 сентября 2008

Да, на самом деле есть. Вам нужно будет написать собственную функцию assert самостоятельно, так как assert() в C ++ - это в точности C assert() с включенной в нее * функцией '1003 *. К счастью, это удивительно просто.

Assert.hh

template <typename X, typename A>
inline void Assert(A assertion)
{
    if( !assertion ) throw X();
}

Приведенная выше функция выдаст исключение, если предикат не выполняется. Тогда у вас будет шанс поймать исключение. Если вы не поймаете исключение, будет вызван terminate(), который завершит программу аналогично abort().

Вы можете спросить, как насчет оптимизации утверждения, когда мы создаем для производства. В этом случае вы можете определить константы, которые будут означать, что вы создаете для производства, а затем обратиться к константе, когда вы Assert().

debug.hh

#ifdef NDEBUG
    const bool CHECK_WRONG = false;
#else
    const bool CHECK_WRONG = true;
#endif

main.cc

#include<iostream>

struct Wrong { };

int main()
{
    try {
        Assert<Wrong>(!CHECK_WRONG || 2 + 2 == 5);
        std::cout << "I can go to sleep now.\n";
    }
    catch( Wrong e ) {
        std::cerr << "Someone is wrong on the internet!\n";
    }

    return 0;
}

Если CHECK_WRONG является константой, то вызов Assert() будет скомпилирован в производственном процессе, даже если утверждение не является константным выражением. Есть небольшой недостаток в том, что, ссылаясь на CHECK_WRONG, мы печатаем немного больше. Но взамен мы получаем преимущество в том, что мы можем классифицировать различные группы утверждений и включать и отключать каждое из них по своему усмотрению. Так, например, мы можем определить группу утверждений, которые мы хотим включить даже в рабочем коде, а затем определить группу утверждений, которые мы хотим видеть только в сборках разработки.

Функция Assert() эквивалентна вводу

if( !assertion ) throw X();

но это ясно указывает на намерение программиста: сделать утверждение. Утверждения также легче найти с помощью этого подхода, как простые assert() s.

Подробнее об этой технике см. Бьярн Страуструп «Язык программирования C ++ 3e», раздел 24.3.7.2.

11 голосов
/ 03 октября 2008

функции отчетов об ошибках glib используют метод продолжения после подтверждения. glib - это базовая библиотека независимости платформ, которую использует Gnome (через GTK). Вот макрос, который проверяет предусловие и печатает трассировку стека в случае сбоя предусловия.

#define RETURN_IF_FAIL(expr)      do {                  \
 if (!(expr))                                           \
 {                                                      \
         fprintf(stderr,                                \
                "file %s: line %d (%s): precondition `%s' failed.", \
                __FILE__,                                           \
                __LINE__,                                           \
                __PRETTY_FUNCTION__,                                \
                #expr);                                             \
         print_stack_trace(2);                                      \
         return;                                                    \
 };               } while(0)
#define RETURN_VAL_IF_FAIL(expr, val)  do {                         \
 if (!(expr))                                                       \
 {                                                                  \
        fprintf(stderr,                                             \
                "file %s: line %d (%s): precondition `%s' failed.",     \
                __FILE__,                                               \
                __LINE__,                                               \
                __PRETTY_FUNCTION__,                                    \
                #expr);                                                 \
         print_stack_trace(2);                                          \
         return val;                                                    \
 };               } while(0)

Вот функция, которая печатает трассировку стека, написанная для среды, которая использует цепочку инструментов gnu (gcc):

void print_stack_trace(int fd)
{
    void *array[256];
    size_t size;

    size = backtrace (array, 256);
    backtrace_symbols_fd(array, size, fd);
}

Вот как бы вы использовали макросы:

char *doSomething(char *ptr)
{
    RETURN_VAL_IF_FAIL(ptr != NULL, NULL);  // same as assert(ptr != NULL), but returns NULL if it fails.

    if( ptr != NULL )        // Necessary if you want to define the macro only for debug builds
    {
       ...
    }

    return ptr;
}

void doSomethingElse(char *ptr)
{
    RETURN_IF_FAIL(ptr != NULL);
}
6 голосов
/ 01 сентября 2008

Утверждения в C / C ++ выполняются только в отладочных сборках. Так что это не произойдет во время выполнения. В общем, утверждения должны отмечать вещи, которые в случае их возникновения указывают на ошибку, и, как правило, отображать предположения в вашем коде и т. Д.

Если вы хотите иметь код, который проверяет ошибки во время выполнения (в выпуске), вам, вероятно, следует использовать исключения, а не утверждения, поскольку именно для этого они и предназначены. Ваш ответ в основном заключает исключение в синтаксис assert. Несмотря на то, что это сработает, я не вижу особого преимущества в этом, если просто бросить исключение.

5 голосов
/ 07 ноября 2008

Вот что у меня есть в "assert.h" (Mac OS 10.4):

#define assert(e) ((void) ((e) ? 0 : __assert (#e, __FILE__, __LINE__)))
#define __assert(e, file, line) ((void)printf ("%s:%u: failed assertion `%s'\n", file, line, e), abort(), 0)

Исходя из этого, замените вызов abort () на throw (исключение). И вместо printf вы можете отформатировать строку в сообщение об ошибке исключения. В итоге вы получите что-то вроде этого:

#define assert(e) ((void) ((e) ? 0 : my_assert (#e, __FILE__, __LINE__)))
#define my_assert( e, file, line ) ( throw std::runtime_error(\
   std::string(file:)+boost::lexical_cast<std::string>(line)+": failed assertion "+e))

Я не пытался это скомпилировать, но вы поняли смысл.

Примечание: вам нужно убедиться, что заголовок «исключение» всегда включен, а также boost (если вы решите использовать его для форматирования сообщения об ошибке). Но вы также можете сделать my_assert функцией и объявить только ее прототип. Что-то вроде:

void my_assert( const char* e, const char* file, int line);

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

Оберните его в #ifdef DEBUG, если вам это нужно, или нет, если вы всегда хотите запускать эти проверки.

0 голосов
/ 06 февраля 2016

Если вы хотите выбросить строку символов с информацией об утверждении: http://xll8.codeplex.com/SourceControl/latest#xll/ensure.h

0 голосов
/ 08 октября 2014
_set_error_mode(_OUT_TO_MSGBOX);

Поверьте мне, эта функция может помочь вам.

...