关于const应用与const中的c++陷阱

目录

const在不同位置的作用

[int *const pt 的含义](#int *const pt 的含义)

[const int *pt 的含义:](#const int *pt 的含义:)

[int *const pt与const int *pt 区别](#int *const pt与const int *pt 区别)

[const int *pt(pt指向一个const int)](#const int *pt(pt指向一个const int))

[int *const pt](#int *const pt)

C++陷阱:多级指针的const匹配

[疑问:为什么在一级间接关系 非const指针可以赋值给const指针](#疑问:为什么在一级间接关系 非const指针可以赋值给const指针)


const在不同位置的作用

int *const pt 的含义

pt 是一个指向常量整数的指针

const 修饰的是 pt 本身,不是 pt 指向的数据

这意味着:pt 不能改变指向 (不能指向其他地址)但可以通过 pt 修改指向的数据

const int *pt 的含义:

pt 是一个指向常量整数的指针

const 修饰的是pt指向的数据

这意味着:通过 pt 不能修改 它指向的数据 但 pt 本身可以改变指向(指向其他地址)

int *const pt与const int *pt 区别

const int *pt(pt指向一个const int)

cpp 复制代码
#include <iostream>
int main()
{
    using namespace std;
    int age =20;
    int *pd =&age;
    const int *pt =pd;
    cout<<"pt的值"<<*pt<<endl;
    cout<<"age的地址"<<&age<<endl;
    cout<<"pt指向的地址"<<pt<<endl;
    age=19;//通过更改age的值
    cout<<"更改后pt的值"<<*pt<<endl;
    cout<<"更改后age的地址"<<&age<<endl;
    cout<<"更改后pt指向的地址"<<pt<<endl;
    int agge=18;//更改pt的地址
    pt =&agge;
    cout<<"更改地址后pt的值"<<*pt<<endl;
    cout<<"agge的地址"<<&agge<<endl;
    cout<<"更改地址后pt指向的地址"<<pt<<endl;
    return 0;
    
}

输出的结果如下

通过观察输出的结果可以发现 在const int* pt中 pt可以通过修改其指向的地址来更改数据 同时也可以借助更改开始指向age的值来更改数据

int *const pt

cpp 复制代码
#include <iostream>
int main()
{
    using namespace std;
    int age =20;
    int *pd =&age;
    int *const pt =pd;
    cout<<"pt的值"<<*pt<<endl;
    cout<<"age的地址"<<&age<<endl;
    cout<<"pt指向的地址"<<pt<<endl;
    age =21;
    cout<<"更改后pt的值"<<*pt<<endl;
    cout<<"更改后age的地址"<<&age<<endl;
    cout<<"更改后pt指向的地址"<<pt<<endl;
    /*
    pt =&agge;
    cout<<"更改地址后pt的值"<<*pt<<endl;
    cout<<"agge的地址"<<&agge<<endl;
    cout<<"更改地址后pt指向的地址"<<pt<<endl;
    */
    //注意/**/中的代码会报错
    return 0;
}

注意上述代码后面/**/中的代码会报错 其报错的原因即可以证明出在int *const pt中 const修饰是pt本身 故其不可以更改其指向的地址 上述代码通过其开始修改age(其开始指向的地址)从而更改其值

C++陷阱:多级指针的const匹配

cpp 复制代码
#include <iostream>
int main()
{
    using namespace std;
    const int **pp2;
    int *pl;
    const int n =13;
    pp2= &pl;
    *pp2=&n;
    *pl=10;
    cout<<**pp2;
    return 0;
}

上述代码输出结果:error: assigning to 'const int **' from incompatible type 'int **'

pp2= &pl;

^~~

1 error generated.

为什么上述代码会发生报错 其根本原因在于其犯了"非const指针赋值给const指针"

如果允许上述代码运行 观察下述

cpp 复制代码
const int **pp2;
int *pl;
const int n = 13;

pp2 = &pl;      // 如果允许这个
*pp2 = &n;      // 那么*pp2(即pl)现在指向const的n
*pl = 10;       // 通过非const指针pl修改了const变量n!违反const原则

pp2指向pl的地址(即pp2存储的是pl变量的地址)

->通过pp2解引用访问pl,让pl指向n的地址

->通过pl解引用访问n,将n的值修改为10

**其行为违反const的原则:**const指针承诺不修改所指向的对象,如果我们允许将非const指针(即可以修改所指向对象的指针)赋值给const指针,那么通过这个const指针我们虽然不会修改对象,但是原来的非const指针仍然可以修改对象,这实际上破坏了const指针所做的承诺(即指向的对象不会被修改)。

疑问:为什么在一级间接关系 非const指针可以赋值给const指针

int* p1;

const int* p2 = p1;

观察上述发现 其实其是通过p2来进行访问p1的

其根本原因是因为其权限不同

源指针 (int*):具有读写权限

目标指针 (const int*):只有读权限

即p1将地址传给p2 由于const int*的限制 其只有读权限 无法对地址的值进行修改

感谢阅读

相关推荐
端平入洛6 小时前
delete又未完全delete
c++
端平入洛1 天前
auto有时不auto
c++
哇哈哈20212 天前
信号量和信号
linux·c++
多恩Stone2 天前
【C++入门扫盲1】C++ 与 Python:类型、编译器/解释器与 CPU 的关系
开发语言·c++·人工智能·python·算法·3d·aigc
蜡笔小马2 天前
21.Boost.Geometry disjoint、distance、envelope、equals、expand和for_each算法接口详解
c++·算法·boost
超级大福宝2 天前
N皇后问题:经典回溯算法的一些分析
数据结构·c++·算法·leetcode
weiabc2 天前
printf(“%lf“, ys) 和 cout << ys 输出的浮点数格式存在细微差异
数据结构·c++·算法
问好眼2 天前
《算法竞赛进阶指南》0x01 位运算-3.64位整数乘法
c++·算法·位运算·信息学奥赛
yyjtx2 天前
DHU上机打卡D31
开发语言·c++·算法
czxyvX2 天前
020-C++之unordered容器
数据结构·c++