1.nullprt与NULL

代码:

void f(int i) {    cout << "f(int)" << endl;}

void f(char* c) {    cout << "f(char*)" << endl;}int main () {
    f();
    f('\0');
    // f(NULL); 在我的GCC中编译不过,因为NULL被认为是(void*)0,二义性
    // f((void*)0); 二义性
    f(nullptr);

    ;
}

结果:

因为NULL容易出问题,因此出现了nullptr。nullptr是有类型的,并且可以被隐式转化为指针类型。

额外:NULL,0,'\0' 的区别

NULL在 "stdio.h" 中的定义如下:

#ifndef __cplusplus
#define NULL ((void *)0)
#else   /* C++ */
#define NULL 0
#endif  /* C++ */

说明NULL在C中是(void*)0,而在C++中是0.

注意:0对于指针来说是一个特别对待的值,"当常量0处于应该作为指针使用的上下文中时,它就作为空指针使用“

如下程序所示;

int main () {
    void* p;
    p = 0;
    //p = 1; 这里报错了,虽然与0都是int类型
    cout << p << endl;
    return 0;
}

而'\0'是所有位都是0的字节,在ASCII码表中所示如下:

有趣的情况:C的字符串是用'\0'结尾的,而有的系统(比如Ubuntu14.04)用NULL也能结束(不过这在其他系统不一定能行得通):

int main () {
    char str[3];
    str[0] = 'a';
    str[1] = 'b';
    cout << str << endl;
    str[2] = '\0';
    cout << str << endl;
    str[2] = 0;
    cout << str << endl;
    return 0;
}

结果:

这三个都是表现出0,但是你的0不是我的0,如下代码:

int main () {
    cout << "\\0的大小:  " << sizeof('\0') << endl;
    cout << "0的大小:   " << sizeof(0) << endl;
    cout << "NULL的大小:" << sizeof(NULL) << endl;
    return 0;
}

结果:

总:这三者都是零,但是NULL是16进制的0(对于C来说这是(void*)0,是指针),0是10进制的0,'\0'是8进制的0

PS:上面NULL可以替代'\0'的情况可以理解为,在二进制中,NULL是16个0而'\0'是8个0,NULL可以截断最后8位成为'\0'。


  2.nullptr是关键字


  3.nullptr的类型是nullptr_t

使用nullptr_t需要包含<cstddef>,其定义为typedef decltype(nullptr) nullptr_t;

使用nullptr_t可以定义多个指针空值,而不仅仅是nullptr一个(虽然nullptr已经够用了)

int main () {
    nullptr_t myNull;
    f(myNull);

    ;
}

规则:

  • nullptr_t类型数据可以隐式转换成任意一个指针类型。
  • nullptr_t类型数据不能转换为非指针类型,即使reinterpret_cast
  • nullptr_t类型数据不适用于算术运算表达式。
  • nullptr_t类型数据适用于关系算术表达式,但仅能与nullptr_t和指针类型比较。

  

  4.nullptr与模板

模板只把nullptr作为一个普通的类型进行推导(并不会视为T* 指针)

template<typename T> void g(T* t) { cout << sizeof(T) << endl;}

template<typename T> void h(T t) { cout << sizeof(T) << endl;}

int main () {
    // g(nullptr); 类型是nullptr_t而不是指针
    g(();
    g((int*)nullptr);

    h();
    h(nullptr);
    h((int*)nullptr);

    ;
}


  5.规定sizeof(nullptr_t) == sizeof(void*)

nullptr是一个编译时期的常量,是编译时期的关键字,能被编译器识别。(void*)0需要经过类型转换才能变成其他指针类型。


  6.nullprt地址

  • 可以打印nullptr_t对象的地址
  • 不能直接打印nullptr的地址
  • 可以打印nullptr的右值引用的地址

代码:

int main () {
    nullptr_t my_null;
    cout << &my_null << endl; //可以打印nullptr_t的地址

    // cout << nullptr << endl; 错误

    const nullptr_t &nullptr_r = nullptr; //可以打印nullptr的右值引用地址
    cout << &nullptr_r << endl;

    ;
}

参考书籍:《深入理解C++11》

C++11---nullptr的更多相关文章

  1. C++ 11 nullptr关键字

    熟悉C++的童鞋都知道,为了避免“野指针”(即指针在首次使用之前没有进行初始化)的出现,我们声明一个指针后最好马上对其进行初始化操作.如果暂时不明确该指针指向哪个变量,则需要赋予NULL值.除了NUL ...

  2. C++11 并发指南一(C++11 多线程初探)

    引言 C++11 自2011年发布以来已经快两年了,之前一直没怎么关注,直到最近几个月才看了一些 C++11 的新特性,今后几篇博客我都会写一些关于 C++11 的特性,算是记录一下自己学到的东西吧, ...

  3. C++11 并发指南一(C++11 多线程初探)(转)

    引言 C++11 自2011年发布以来已经快两年了,之前一直没怎么关注,直到最近几个月才看了一些 C++11 的新特性,今后几篇博客我都会写一些关于 C++11 的特性,算是记录一下自己学到的东西吧, ...

  4. 【C/C++开发】C++11 并发指南一(C++11 多线程初探)

    引言 C++11 自2011年发布以来已经快两年了,之前一直没怎么关注,直到最近几个月才看了一些 C++11 的新特性,今后几篇博客我都会写一些关于 C++11 的特性,算是记录一下自己学到的东西吧, ...

  5. C++学习笔记28:运行期型式信息

    RTTI 运行期标识对象的型式信息 优势:允许使用指向基类的指针或引用自如地操作派生类的对象 typeid:获取表达式的型式:type_info:型式信息类 头文件:typeinfo 对象转型模板 d ...

  6. 译注(3): NULL-计算机科学上最糟糕的失误

    原文:the worst mistake of computer science 注释:有些术语不知道怎么翻译,根据自己理解的意思翻译了,如有不妥,敬请提出:) 致谢: @vertextao @fra ...

  7. 地区sql

    /*Navicat MySQL Data Transfer Source Server : localhostSource Server Version : 50136Source Host : lo ...

  8. C++11特性——变量部分(using类型别名、constexpr常量表达式、auto类型推断、nullptr空指针等)

    #include <iostream> using namespace std; int main() { using cullptr = const unsigned long long ...

  9. C++11 之 nullptr

    C++11 中,nullptr 是空指针,可用来给 (指向任意对象类型的) 指针赋值 广义整型 (integral types) = char, short, int, long, long long ...

  10. c++11编码规范 NULL还是nullptr

    0和nullptr/NULL 至于指针(地址值),根据实际选择用0.NULL还是nullptr.对使用了C++11特性的项目,选用nullptr:对于C++03项目,推荐NULL,因为它像是一个指针

随机推荐

  1. Python与Hack之守护进程

    1.什么是守护进程: 在linux或者unix操作系统中,守护进程(Daemon)是一种运行在后台的特殊进程,它独立于控制终端并且周期性的执行某种任务或等待处理某些发生的事件.由于在linux中,每个 ...

  2. 我的c++学习(8)运算符重载和友元

    运算符的重载,实际是一种特殊的函数重载,必须定义一个函数,并告诉C++编译器,当遇到该运算符时就调用此函数来行使运算符功能.这个函数叫做运算符重载函数(常为类的成员函数). 方法与解释 ◆ 1.定义运 ...

  3. 二叉搜索树 POJ 2418 Hardwood Species

    题目传送门 题意:输入一大堆字符串,问字典序输出每个字符串占的百分比 分析:二叉搜索树插入,然后中序遍历就是字典序,这里root 被new出来后要指向NULL,RE好几次.这题暴力sort也是可以过的 ...

  4. Robotium ant 报错Unable to find instrumentation info for: ComponentInfo{project/android.test.InstrumentationTestRunner}

    [echo] Running tests ... [exec] INSTRUMENTATION_STATUS: id=ActivityManagerService [exec] INSTRUMENTA ...

  5. ssl原理,非对称加密握手,对称加密传输

    SSL的基本思想是用非对称加密来建立链接(握手阶段),用对称加密来传输数据(传输阶段).这样既保证了密钥分发的安全,也保证了通信的效率. SSL握手,单方服务器认证(一般的浏览器上网) SSL握手,双 ...

  6. Java NIO之通道Channel

    channel与流的区别: 流基于字节,且读写为单向的. 通道基于快Buffer,可以异步读写.除了FileChannel之外都是双向的. channel的主要实现: FileChannel Data ...

  7. HDU - Pseudoforest

    Description In graph theory, a pseudoforest is an undirected graph in which every connected componen ...

  8. 【HDU】4089 Activation

    http://acm.hdu.edu.cn/showproblem.php?pid=4089 题意: 有n个人排队等着在官网上激活游戏.主角排在第m个. 对于队列中的第一个人.有以下情况:1.激活失败 ...

  9. usaco silver刷水~其实是回顾一下,补题解

    [BZOJ1606][Usaco2008 Dec]Hay For Sale 裸01背包 ;i<=n;i++) for(int j=m;j>=a[i];j--) f[j]=max(f[j], ...

  10. ObjectContext,DataContext和DBContext 分别获取linq 的sql方法

    ObjectContext 先定义一个扩展方法: public static string ToTraceString<T>(this IQueryable<T> t) { s ...