2012-03-14 24 views
7
00001 /* assert.h 
00002 Copyright (C) 2001, 2003 Free Software Foundation, Inc. 
00003 Written by Stephane Carrez ([email protected])  
00004 
00005 This file is free software; you can redistribute it and/or modify it 
00006 under the terms of the GNU General Public License as published by the 
00007 Free Software Foundation; either version 2, or (at your option) any 
00008 later version. 
00009 
00010 In addition to the permissions in the GNU General Public License, the 
00011 Free Software Foundation gives you unlimited permission to link the 
00012 compiled version of this file with other programs, and to distribute 
00013 those programs without any restriction coming from the use of this 
00014 file. (The General Public License restrictions do apply in other 
00015 respects; for example, they cover modification of the file, and 
00016 distribution when not linked into another program.) 
00017 
00018 This file is distributed in the hope that it will be useful, but 
00019 WITHOUT ANY WARRANTY; without even the implied warranty of 
00020 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 
00021 General Public License for more details. 
00022 
00023 You should have received a copy of the GNU General Public License 
00024 along with this program; see the file COPYING. If not, write to 
00025 the Free Software Foundation, 59 Temple Place - Suite 330, 
00026 Boston, MA 02111-1307, USA. */ 
00027 
00028 #ifndef _ASSERT_H 
00029 #define _ASSERT_H 
00030 
00031 #ifdef NDEBUG 
00032 # define assert(EX) 
00033 #else 
00034 # define assert(EX) (void)((EX) || (__assert (#EX, __FILE__, __LINE__),0)) 
00035 #endif 
00036 
00037 #ifdef __cplusplus 
00038 extern "C" { 
00039 #endif 
00040 
00041 extern void __assert (const char *msg, const char *file, int line); 
00042 
00043 #ifdef __cplusplus 
00044 }; 
00045 #endif 
00046 #endif 

La pregunta es: ¿qué es para el "(vacío)" en la línea 34 y qué es __assert?Implementación de C++ assert en assert.h

Respuesta

11

Look az esta línea:

extern void __assert (const char *msg, const char *file, int line); 

__assert es función que toma un mensaje de afirmación, un nombre de archivo y un número de línea como argumentos. Básicamente, este es el método que imprime el mensaje de error y finaliza el programa cuando la afirmación falla.

Luego, busquen en la definición macro anterior:

#define assert(EX) (void)((EX) || (__assert (#EX, __FILE__, __LINE__),0)) 

Define el assert(EX) macro así, comprueba en primer lugar la expresión EX y (debido a la operación de cortocircuito del operador C++ ||) sólo si falla, llama a la función __assert y pasa la excepción de aserción fallida como una cadena y la ubicación exacta de la llamada al método assert() en los archivos de origen. Con este engaño preprocesador la biblioteca afirmación se consigue que cuando se escribe lo siguiente en su programa

assert(a == 0); 

y su afirmación de falla durante la ejecución del programa, se obtiene el mensaje

Assertion failed: a == 0 at program.c, line 23 

error detallado que ayuda a identifique la ubicación exacta donde falló la aserción en su código.

La parte (void) es solo para asegurarse de que el compilador no advierte sobre el resultado no utilizado de la expresión (EX) || 0, consulte las otras respuestas, los chicos lo explicaron bien.

El preprocesador restante define NDEBUG se utiliza para activar la generación de aserciones en todo el tiempo de compilación, el ejecutable resultante será más pequeño y más rápido.

+0

Muy bien, perfecto, gracias por la explicación detallada. Solo dos preguntas: 1) Por qué (__ASsert (#EX, __FILE__, __LINE __), 0) se usa ", 0" y 2) ¿Por qué (qué mecanismo) suprime la advertencia del compilador cuando se convierte bool en void? ¿Es porque el vacío no es nada? – Narek

+4

1) El [operador de coma] (http://en.wikipedia.org/wiki/Comma_operator) (',') primero ejecuta la expresión a su izquierda (llamada '__assert') luego la de su derecha (literal' 0') y devuelve el resultado de este último. Es necesario porque el operador '||' espera un operando booleano-compatible, pero la función '__assert' devuelve' void'. – buc

+4

2) Imagine una afirmación '(a == 3) || (0); 'solo en una línea. No tiene mucho sentido realizar una comparación y luego no hacer nada con el resultado. El compilador también reconoce esto y sospecha que ha olvidado completar la declaración y le advierte al respecto. Al convertir toda la expresión en 'void':' (void) ((a == 3) || (0)) 'le dices al compilador que esta expresión no tiene un valor de resultado razonable (como una llamada a un' void'- la función de retorno tampoco debería), por lo que no se quejará de ello. – buc

1

Suprime las advertencias del compilador sobre valores o variables no utilizados.

También tenga en cuenta el operador de coma a la derecha, que hace que ambos lados de la || algo que es convertible a bool.

__assert es una función interna que presumiblemente imprimirá un mensaje e implementará el comportamiento de aserción requerido (es decir, llame al abort()).

7

__assert es parte de la implementación; en este caso, una función en la biblioteca a la que se llamará en caso de fallo de aserción. Y el (void) es simplemente para cerrar las advertencias del compilador sobre los resultados no utilizados del operador ||.

1

Casi ...

considerar: assert(a == 0); Esto es ampliado para

(void)((a == 0) || (__assert (#a == 0, __FILE__, __LINE__),0))

Si usted tiene la expresión (x || y) - En este caso, tanto x como y se evalúan como Bools, por lo que si 'a' evals a 0, la definición de || dice probar el próximo bool en línea, es decir, 'y'

Entonces, si su afirmación es falsa, se evalúa (__assert (#a == 0, __FILE__, __LINE__),0), lo que significa que se llama a __assert().

¿Por qué el (__assert(), 0) ?? __assert es una función definida como void __assert() - bueno ... la mayoría de las veces. ver http://www.opensource.apple.com/source/Libc/Libc-825.26/include/assert.h para un conjunto de macros assert() - tenga en cuenta que todas ellas terminan llamando a una función ...

El operador de coma le permite hacer dos cosas en la misma instrucción, es decir, i++, j++ - pero recuerde que una expresión tiene que evalúe a algún valor - en este caso al "0" - por lo que la declaración completa evalúa a (void)(0 || 0) que es una expresión válida. Como efecto secundario, su expresión se evalúa y posiblemente se llama a una función.

Nota en (void)(0 || 0) el primer 0 es de su afirmación fallida. Esto puede crearse en tiempo de compilación si su afirmación evalúa algo que el compilador puede crear como una constante o una expresión que se evalúa, es decir, (void)((a == 0) || 0)

tenga en cuenta que la coma ", 0" es para compiladores realmente pedantes. no lo veo Pero el (a || __assert()) es muy común. Puede escribir la macro como

#define assert(x) { if(! (x)) __assert(.....); }