網(wǎng)站首頁 編程語言 正文
在C++中,const作用于指針時,可以看做是對指針權(quán)限的限制。這里我們先把指針的權(quán)限歸為兩種,分別為指向權(quán)限和修改權(quán)限。(ps:以上是為了理解方便,實際并沒有如此規(guī)定)
int a = 10, b = 20; int* p = &a; p = &b; // 改變指向的權(quán)限 ? *p = 30; // 修改內(nèi)存的權(quán)限 ? const int* cp = &a; // 限制修改權(quán)限 //*cp = 100; // error:表達式必須是可修改的左值 修改 ? cp = &b; // ok. 指向 ? int* const pa = &a; // 限制指向權(quán)限 *pa = 100; // ok. 修改 ? //pa = &b; // error:表達式必須是可修改的左值 指向 ?
指針的賦值一般遵守權(quán)限縮小式的賦值。例如,我有一本書,我有使用權(quán)限(我可以看,可以做筆記),借給你后你只有閱讀權(quán)限(只能看,不能做筆記)。當(dāng)然,如果我們關(guān)系好,我可以賦予你使用權(quán)(你擁有讀寫的權(quán)利)。同樣的,指針的賦值也是如此。
int a = 10; int* p = &a; // int* <== int * int* q = p; // int* <== int* const int* cp = p; // const int* <== int* 權(quán)限縮小,? int* const pa = p; // int* <== int* 注意:int* const pa;是“int*”類型 //int* p1 = cp; // error:int* <== const int * 權(quán)限放大,? int* p2 = pa; // ok. int* <== int*
我們可以得出一級指針賦值的公式
int * <== int * // int* 包含 int* 和 int* const類型 int const * <== int * // int const * <=等同=> const int * // 以上公式反過來賦值就是錯誤的..
練習(xí)一:一級指針指向練習(xí)題
題目一:下列表達式語句賦值錯誤的是?
int a = 10; const int* p = &a; int* q = p; int a = 10; int* const p = &a; int* q = p; int a = 10; int* const p = &a; int* const q = p; int a = 10; int* const p = &a; const int* q = p;
答案:(鼠標(biāo)選中查看)
??錯誤:A,正確:B、C、D??
解析:
int a = 10; const int* p1 = &a; int* q1 = p1; // error:無法從const int * 轉(zhuǎn)為 int * /* 分析: int* <= cosnt int* */ int* const p2 = &a; int* q2 = p2; /* 分析: int* <= int* */ int* const p3 = &a; int* const q3 = p3; /* 分析: int* <= int* */ int* const p4 = &a; const int* q4 = p4; /* 分析: cosnt int* <= int* */
練習(xí)二:二級指針指向練習(xí)題
題目二:下列表達式語句錯誤的有。
// 選項A int a = 10; int* p1 = &a; const int** q1 = &p1; // 選項B int a = 10; int* p2 = &a; int* const* q2 = &p2; // 選項C int a = 10; int* p3 = &a; int** const q3 = &p3; // 選項D int a = 10; int* const p4 = &a; int** q4 = &p4; // 選項E int a = 10; const int* p5 = &a; int* const* q5 = &p5;
答案:(鼠標(biāo)選中查看)
??錯誤:A、D、E,正確:B、C??
A選項;
錯誤; 注:如果const修飾的是二級指針,我們需要對二級指針的逐層解引用進行分析。
int* p1 = &a; const int** q1 = &p1; //error 無法從“int * *”轉(zhuǎn)換為“const int** ”
int* p1 = &a;
?p1的類型為int*
?取地址為?int **
const int** q1 = &p1;
?q1的類型為?const int **
則指針賦值過程為?const int ** <= int* *
分析:
- const作用于(**q1),修飾二級指針。表示不可通過q1對?
a
?的值進行修改。 - *q1?解引用一次后,為一級指針,即?p1?。但是?p1?存在對?
a
?修改的風(fēng)險,因此無法直接賦值。
修改方案:
- 方案一:直接限定一級指針p1。保證p1不會修改a的值,即
const int * p1= &a;
?const int** q1 = &p1;
- 方案二:間接限定q1,使其指向時縮小權(quán)限,對解引用后的(*q1)修改的權(quán)限做出限制,如:
const int * const * q1;
// 方案一 const int* p12 = &a; const int** q12 = &p12; // 方案二 int* p11 = &a; const int* const* q11 = &p1;
B選項;
正確; 注:如果const修飾的是一級指針,我們可以拋開二級指針的表象,但看一級指針的賦值操作是否正確。如本例。
int* p2 = &a; int* const* q2 = &p2; /* 分析: int* const* <== int* * const修飾 *q2,即cosnt修飾一級指針 cosnt* <== * // 去掉前面的 int* <= int* int const * <== int * // 添加一個任意類型,如int 如??所示,這是一個權(quán)限縮小的一級指針賦值,? */
C選項;
正確; 注:如果兩邊類型相同,則無需進行判斷。如本例。
int* p3 = &a; int** const q3 = &p3; cout << typeid(q3).name() << endl; //輸出 q3 類型 int * * /* 分析: int**const <== int* * 即 int** <== int* * 兩邊類型相同,無需進行特殊判斷,? */
ps:如果const修飾的參數(shù)右邊無“*”號,則該cosnt不作用于類型。如:
int n = 10; // 使用typeid(valtypr).name() 輸出變量類型 int const* p1 = &n; // int const * int* const p2 = &n; // int * // 忽略const int* p = &n; int** q = &p; //int const** q1 = &p; int const* const* q1 = &p; // int const* const* int* const* q2 = &p; // int* const* int** const q3 = &p; // int** // 忽略cosnt
D選項;
錯誤; 同B選項相同,對于const修飾的一級指針進行判即可。
int* const p4 = &a; int** q4 = &p4; //error "int *const *"類型的值不能用于初始化"int **類型的實體
int* const p4 = &a;
?類型為?int*
,因為const的存在,取地址后類型為?int * const *
int** q4 = &p4;
?類型為?int**
則指針賦值過程為?int** <== int* const*
分析:
- const作用于(p4),修飾一級指針。則我們忽略沒有const修飾的部分。即
* <== const*
?,//忽略?int?部分,該部分賦值時權(quán)限沒有發(fā)生變化。int* <== int const *
?,給指針確定一個類型,如“int” 類型 - 如??,我們可以看到,該表達式語句想從int const* 類型處,獲得一個?int* 類型的賦值,也就是說這是權(quán)限放大式的賦值。錯誤原因:該賦值會使得 int* 類型指針對常量(int const* 所指向的值)產(chǎn)生修改的風(fēng)險。
修改方案:
- 把
int* <== int const *
型賦值改成?int const* <== int const *
類型賦值即,int*const* q4 = &p4;
?
E選項;
錯誤; 注:如果賦值兩邊都有const時,各論各的分析,如下。
const int* p5 = &a; int* const* q5 = &p5; //error 無法從"const int **"轉(zhuǎn)換為"int *const *"
分析:
- 省略分析過程等賦值類型為?
int* const* <== const int* *
- 分情況分析:
- 提取指針左邊部分:
int* <== const int*
?,錯誤 ? - 提取指針右邊部分:
cosnt * <== *
?即?int const * <== int *
,正確 ?
- 提取指針左邊部分:
- 綜上,錯誤 ?。
修改方案:
1.修改指針左邊類型:int* <== const int*
???int* <= int*
2.修改指針左邊類型:int* <== const int*
???const int* <= const int*
//1 int* <= int* int* p51 = &a; // const int* p5 ? int* p51 int* const* q51 = &p51; //2 const int* <= const int* const int* p52 = &a; const int* const* q52 = &p52; // int* const* q5 ? const int* const* q52
方法總結(jié):
對于二級指針的賦值操作判斷,看const位置、主要有以下兩種情況:
- 如果?const修飾的是二級指針?如:
-
int const **
?,如選項A。我們需要考慮其解引用情況。cosnt修飾二級指針?biāo)赶虻闹禐槌A浚怯捎谝淮谓庖煤蟮闹羔槙嬖谛薷某A康娘L(fēng)險,因此我們需要限制該指針與常量之間進行過度的一級指針 - 針對此類二級指針,我們只需記住合法的賦值為等式兩邊需同時有const?,
const int* cosnt* <== int**
或?左邊等式有兩個cosnt?,const int* cosnt* <== int**
?。
-
- 如果?const修飾的是一級指針或其他?如:
1.const修飾的是一級指針,int * const *
,如選項B。單獨剝離出含cosnt類型的一級指針類型進行分析
2.即修飾一級指針又修飾二級指針?如,?int cosnt * cosnt *
3.無const修飾?如,?int**
?或?int ** cosnt
,如選項C、選項D
- 針對此類二級指針,通過一級指針的比較進行比較即可。
總結(jié)
原文鏈接:https://blog.csdn.net/weixin_43919932/article/details/108632601
相關(guān)推薦
- 2022-07-01 Keras實現(xiàn)Vision?Transformer?VIT模型示例詳解_python
- 2022-09-20 C#先判斷是否存在再創(chuàng)建文件夾或文件與遞歸計算文件夾大小_C#教程
- 2021-12-18 死鎖的處理基本策略和常用方法
- 2022-07-06 python繪制子圖技巧之plt.subplot、plt.subplots及坐標(biāo)軸修改_python
- 2022-09-24 C#中的引用類型以及特殊引用類型詳解_C#教程
- 2023-03-20 Redis腦裂導(dǎo)致數(shù)據(jù)丟失的解決_Redis
- 2022-07-10 使用Docker安裝RabbitMQ
- 2022-03-24 .NET?6開發(fā)TodoList應(yīng)用之實現(xiàn)API版本控制_實用技巧
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支