c++的const小总结(2)

c++的const小总结(1) http://www.cnblogs.com/MyNameIsPc/p/7091631.html

  • 顶层const和底层cnost

先凭经验区分一下两者的区别?

在primer c++里看到这个章节时,我有点疑惑。

primer c++上面也解释的稀里糊涂

如前所述,指针本身是一个对象,它又可以指向另外一个对象。因此,指针本身是不是常量以及指针所指的是不是一个常量就是两个相互独立的问题。用名词顶层const(top-level const)表示指针本身是个常量,而用名词底层const (low-leve丨const)表示指
针所指的对象是一个常量。

直到我关注了英文:

顶层const(top-level const)  底层const (low-level const)

所以说白了顶层const就是本身是const。自己最重要嘛。

  • 现在我们来总结性的区分一下这两种const

更一般的,顶层const可以表示任意的对象是常量,这一点对任何数据类型都适用,如算术类型、类、指针等。底层const则与指针和引用等复合类型的基本类型部分有关。比较特殊的是,指针类型既可以是顶层const也可以是底层const,这一点和其他类型相比区别明显。

当执行对象的拷贝操作时,常量是顶层const还是底层const区别明显。其中,顶层const不受什么影响。另一方面,底层const的限制却不能忽视。当执行对象的拷贝操作时,拷入和拷出的对象必须具有相同的底层const资格,或者两个对象的数据类型必须能够转换。一般来说,非常量可以转换成常量,反之则不行:

#include <iostream>
using namespace std;

int main(){
    int i = 0;
    int *const p1 = &i; //不能改变p1的值,这是一个顶层const
    const int ci = 42; //不能改变ci的值,这是一个顶层const
    const int *p2 = &ci; //允许改变p2的值,这是一个底层const
    const int *const p3 = p2; //靠右的const是顶层const,靠左的是底层const
    const int &r = ci; //重要:用于声明引用的const都是底层const
    return 0;
    ///
    i = ci; //正确,拷贝ci的值,ci是一个顶层const,对此操作无影响
    p2 = p3; //正确,p2和p3指向的对象类型相同,p3顶层const的部分不影响
    int *p = p3; //错误,p3包含底层const的定义,而p没有
    p2 = &i; //正确,int*能转换成const int*
    int &r = ci; //错误,普通的int&不能绑定到int常量上
    const int &r2 = i; //正确,const int&可以绑定在一个普通int上
}

说白了,也就是(1)中总结的问题。如果一个指针a的底层有const,另一个指针b的底层没有const,那么b就不能赋值为a。

好像这可以通过类型转换来解释,不过我还没懂。

就说到这里啦

时间: 2024-10-23 15:42:39

c++的const小总结(2)的相关文章

c++的const小总结(3)

c++的const小总结(2) 今天来讨论指针或引用的形参与const的关系. 首先说明:c++里规定同名函数中参数必须有明显区别. 例如void reset(int &i) 和 void reset(const int &i)这样的同名函数是通不过编译的. int i = 0; const int ci = i; string::size_type ctr = 0; reset(&i); reset(&ci); //错误,不能用指向const int对象的指针初始化int

状态空间搜索好题UVA10603

题目 分析:注意这里求的是最少流量, 二不是最少步数!!!所以我们用优先队列去维护一个最小流量,然后进行bfs即可,解释一下一个重要的数组ans[i],表示的是杯子中的水为i时的最小流量 1 #include "iostream" 2 #include "cstdio" 3 #include "cstring" 4 #include "queue" 5 #include "algorithm" 6 usin

联合训练图论场

联合训练图论场题解报告 传送门 A.Euler 题意:略 分析: 这题主要是先掌握欧拉通路的概念,然后是如何判断图是否存在欧拉通路. 欧拉通路:通过图中每条边且只通过一次,并且经过每一顶点的通路. 欧拉回路:通过图中每条边且只通过一次,并且经过每一顶点的回路. 无向图: 欧拉通路:连通图+只存在0个或者两个度数为奇数的点. 欧拉回路:连通图+所有节点的度数均为偶数. 有向图: 欧拉通路:连通图+(所有点的入度=出度 || 出两个点之外其他点的入度=出度,一个点的入度-出度=1,一个点的出度-入度

C++ 中有关const引用的一点小知识

在读<C++ Primer>时,发现之前对const的概念不是很清晰,只知道如何去使用,于是翻开const引用部分又阅读了一遍,发现有两点自己要注意的地方 1.const限定的对象不可以初始化非const引用 ex. 1 const int src = 512; 2 const int &ok_dest = src; //ok: 引用和初始化对象都是const 3 int &err_dest = src; //error : 引用为非const 原因很简单, src 是不可以被

ES6小实验-let和const(2)

继续小实验,上次写到块级作用域,那么为什么需要块级作用域呢?书中给了两个场景: 1.没有块级作用域,内层变量可能会覆盖外层变量.举例: var tmp = new Date() function f() { console.log(tmp) if(false) { var tmp = "hello world" } } f();//undefined 内层的tmp变量把外层的tmp变量覆盖,所以输出结果为undefined 2.用来计数的循环变量泄露为全局变量,举例: var s =

请问微信小程序let和var以及const有什么区别

在JavaScript中有三种声明变量的方式:var.let.const. var:声明全局变量,换句话理解就是,声明在for循环中的变量,跳出for循环同样可以使用. [JavaScript] 纯文本查看 复制代码 ? 1 2 3 4 5 for(var i=0;i<=1000;i++){ var sum=0; sum+=i; } alert(sum); 声明在for循环内部的sum,跳出for循环一样可以使用,不会报错正常弹出结果 let:声明块级变量,即局部变量. 在上面的例子中,跳出fo

const 和 extern 用法小結

 雖說不難,但很重要!! 1.       const常量,如const int max = 100;  优点:const常量有数据类型,而宏常量没有数据类型.编译器可以对前者进行类型安全检查,而对后者只进行字符替换,没有类型安全检查,并且在字符替换时可能会产生意料不到的错误(边际效应) 2.       const 修饰类的数据成员.如:class A { const int size; … } const数据成员只在某个对象生存期内是常量,而对于整个类而言却是可变的.因为类可以创建多个对象,

泥鳅般的const(一个小Demo彻底搞清楚)

#include<stdio.h> int main(){ int a = 3; int b = 5; /* C标准库函数中最常见格式, 目的是保护参数, 可读而不可修改参数内容 */ const int *p1 = &a; //含义: 从右到左"p1是一个指针变量, 可指向一个整型常量" //注意1: *p1是只读常量, 不可作为左值再次赋值; //注意2: p1是变量, 可以重新指向新的变量; //注意3: p1指向的变量a的值仍然可以修改; //*p1 = 7

关于 const 的一点小研究

在饱受 var 的折磨之后,ES6 终于推出了新的定义变量的方法:let 和 const 和 var 相比,let 和 const 有了自己的作用域,let 用于定义变量,而 const 用于定义常量 但在实际工作中,它们常常有一些出人意料的表现,今天就先说说 "常量" const 上面的示例十分很常见,也很容易理解.作为一个常量,定义之后是不能再次定义的 但是如果换一种方式... 在定义了常量 name 之后,没有重定义 name,而是增加一个 info 属性 虽然这个 info 属