一、断言:运行时与预处理时

断言(assertion)是一种编程常用的手段。想必大家都见过 assert 吧。今天我们就来了解一下它。

通常情况下,断言就是将一个返回值总是需要为真的判别式放在语句中,用于排除在设计的逻辑上不应该产生的情况。

比如一个函数总需要输入在一定的范围内的参数,那么程序员就额可以对该参数使用断言,以迫使在该参数发生异常的时候程序退出,从而避免程序陷入逻辑的混乱。

从一些意义上讲,断言并不是正常程序所必须的,不过对于程序调试来说,通常断言能够帮助程序开发者快速定位那些违反了某些前提条件的程序错误。

例子:

#include<cassert>
using namespace std; char* ArrayAlloc(int n)
{
assert(n > );
return new char[n];
} int main()
{
char* a = ArrayAlloc();
}

结果显示,很明显程序崩溃了。

这一点上,虽然异常可以防止程序崩溃,使程序得以继续执行,但是该结束程序的时候,还是要果断地用assert 结束程序

我们应该合理应用断言和异常。

在C++中,程序员也可以定义宏NDEBUG来禁用assert 宏。这对发布程序来说还是必要的。

因为程序用户退出总是敏感的,而且部分的程序错误也未必会导致程序全部功能失效。(比如上一段提到的异常就可以避免这种情况。)

那么通过定义NDEBUG宏发布程序也可以尽量避免程序退出的状况。而当程序有问题时,通过没有定义宏NDEBUG的版本,程序员则可以比较容易地找到出问题的位置。

事实上,assert 宏在<cassert>中的实现方式类似如下形式:

#ifdef NDEBUG
# define assert(expression) (static_cast<void>) //!!!!!
#else
...
#endif

可以看到,一旦定义了NDEBUG宏,assert 宏将被展开为一条无意义的C语句(通常被编译器优化掉)

但是有一个问题,定义NDEBUG宏的时候必须将其定义在相应头文件之前才行!!

在上一篇(三)中,我们还看到 #error 这样的预处理指令,而事实上,通过预处理指令#if 和#error 的配合,也可以让程序员在预处理阶段进行断言。

这样的用法也是极为常见的。

书上对于这方面是这样说的,原话如下:(由于没有相关的设备支持,所以后面自己写了个简单的工程测试了一下)

比如GNU的cmathcalls.h 头文件中(在我们的实验机上,该文件位于/user/include/bits/cmathcalls.h),我们会看到如下代码:
#ifndef _COMPLEX_h
#error "Never use <bits/cmathcalls.h> directly; include<complex.h>instead."
#endif
如果程序员直接包含头文件<bits/cmathcalls.h>并进行编译,就会引发错误。#error 指令会将后面的语句输出,从而提醒用户不要直接使用这个头文件,而应该包含头文件<complex.h>。

我们用这种方法测试一下:

我们首先写一个头文件  test.h 里面有一个类 _Test 文件定义宏为 _TEST

//test.h
#define _TEST class _Test
{
int m_int;
public:
_Test():m_int(){}
const int get_int()const { return m_int; }
};

我们再写另一个头文件 test1.h

//test1.h
#ifndef _TEST #error "请引用正确的头文件 test.h" #endif

意思是,如果该文件没有重定义宏 _TEST ,那么就用原版本

主调函数测试:

#include<iostream>
#include"test1.h"
using namespace std;
int main()
{
_Test a;
cout << a.get_int() << endl;
}

我们试图调用test1.h文件中的_TEST类,但是该文件没有重新定义该类

触发错误。

在次测试:

#include<iostream>
#include"test.h"
#include"test1.h"
using namespace std;
int main()
{
_Test a;
cout << a.get_int() << endl;
}

这样一来,通过预处理时的断言,库发布者就可以避免一些头文件的引用问题。

感谢您的阅读,生活愉快~

(四)静态断言(上),assert,NDEBUG, 以及通过宏定义处理文件包含关系的更多相关文章

  1. C 常用库函数memset,编译器宏定义assert

    一. 总览 1.1库函数 函数名 头文件 功能 原型 说明 syslog syslog.h 记录至系统记录(日志) void    syslog(int, const char *, ...) __p ...

  2. iOS 静态库中使用宏定义区分iPhone模拟器与真机---备用

    问题描述 一般项目中,可以使用宏定义来判断模拟器还是真机,这无疑是有效的. #if TARGET_IPHONE_SIMULATOR #define SIMULATOR 1 #elif TARGET_O ...

  3. Swift入门(四)——可选类型(Optionals)与断言(Assert)

    可选类型是什么? 首先看一个问题,Swift中String类型的变量有一个叫做toInt的方法,能够把String类型变量转换为Int类型变量. var stringValue = "5&q ...

  4. c++11 静态断言

    c++11 静态断言 #define _CRT_SECURE_NO_WARNINGS #include <iostream> #include <string> #includ ...

  5. C++断言与静态断言

    断言是很早之前就有的东西了,只需要引入cassert头文件即可使用.往往assert被用于检查不可能发生的行为,来确保开发者在调试阶段尽早发现“不可能”事件真的发生了,如果真的发生了,那么就表示代码的 ...

  6. C++11 静态断言(static_assert)

      简介 C++0x中引入了static_assert这个关键字,用来做编译期间的断言,因此叫做静态断言. 其语法很简单:static_assert(常量表达式,提示字符串). 如果第一个参数常量表达 ...

  7. python接口自动化(二十三)--unittest断言——上(详解)

    简介 在测试用例中,执行完测试用例后,最后一步是判断测试结果是 pass 还是 fail,自动化测试脚本里面一般把这种生成测试结果的方法称为断言(assert).用 unittest 组件测试用例的时 ...

  8. (五)静态断言(下),static_assert

    二.静态断言与static_assert 通过上一篇,我们可以看到,断言assert宏只有在程序运行的时候才能起作用.而#error值在编译器预处理时才能起作用. 有时候,我们希望在编译时候能做一些断 ...

  9. c语言静态断言

    在php中可以通过xdebug来显示详细的错误信息,可以细化到哪个文件哪行代码引起的报错.在C语言里面也可以通过静态断言(assert)来使得调试代码更加方便.关于断言,可以作为一种很强大的调试方式或 ...

随机推荐

  1. [整理]C语言函数说明和定义

    函数的一般形式是:type-specifier function_name(parameter list) parameter declarations{   body of the function ...

  2. C语言的内存对齐

    从一个例子开始 象下面这样定义的结构体占几个字节? typedef struct{ char a; int i; } Sample; char占1个字节,int占4个字节,答案是5个字节? 错了.如果 ...

  3. CF 1008C Reorder the Array

    You are given an array of integers. Vasya can permute (change order) its integers. He wants to do it ...

  4. sklearn_SVC_支持向量机

    # coding:utf-8 import numpy as np from sklearn.svm import SVC import matplotlib.pyplot as plt #生成数据 ...

  5. 自动检测SOCKET链接断开

    如何判断SOCKET已经断开 最近在做一个服务器端程序,C/S结构.功能方面比较简单就是client端与server端建立连接,然后发送消息给server.我在server端会使用专门的线程处理一条s ...

  6. [转]KMP 算法

    KMP 算法,俗称“看毛片”算法,是字符串匹配中的很强大的一个算法,不过,对于初学者来说,要弄懂它确实不易.整个寒假,因为家里没有网,为了理解这个算法,那可是花了九牛二虎之力!不过,现在我基本上对这个 ...

  7. [LeetCode] #112 #113 #437 Path Sum Series

    首先要说明二叉树的问题就是用递归来做,基本没有其他方法,因为这数据结构基本只能用递归遍历,不要把事情想复杂了. #112 Path Sum 原题链接:https://leetcode.com/prob ...

  8. zTree静态树与动态树的用法——(七)

    0.[简介] zTree 是利用 JQuery 的核心代码,实现一套能完成大部分常用功能的 Tree 插件 兼容 IE.FireFox.Chrome 等浏览器 在一个页面内可同时生成多个 Tree 实 ...

  9. Hibernate5笔记2--单表的增删改查操作

    单表的增删改查操作: (1)定义获取Session和SessionFactory的工具类: package com.tongji.utils; import org.hibernate.Session ...

  10. Java IO,硬骨头也能变软

    开胃菜 先看一张网上流传的http://java.io包的类结构图: 当你看到这幅图的时候,我相信,你跟我一样内心是崩溃的. 有些人不怕枯燥,不怕寂寞,硬着头皮看源码,但是,能坚持下去全部看完的又有几 ...