網(wǎng)站首頁 編程語言 正文
一、單引號和雙引號
- C語言中的單引號用來表示字符字面量
- C語言中的雙引號用來表示字符串字面量
'a'表示字符字面量,在內(nèi)存中占1個字節(jié),'a'+1表示'a'的ASCII碼加1,結(jié)果為'b'
"a"表示字符串字面量,在內(nèi)存中占2個字節(jié),"a"+1表示指針運算,結(jié)果指向"a"結(jié)束符'\0'
下面看一段單引號和雙引號本質(zhì)的代碼:
#include <stdio.h>
int main()
{
char* p1 = 1 ;
char* p2 = '1';
char* p3 = "1";
printf("%s, %s, %s", p1, p2, p3);
printf('\n');
printf("\n");
return 0;
}
編譯后會有警告,繼續(xù)運行會發(fā)生段錯誤,如下:
究竟問題出現(xiàn)在哪里呢?下面先把打印語句注釋掉并運行:
#include <stdio.h>
int main()
{
char* p1 = 1 ;
char* p2 = '1';
char* p3 = "1";
//printf("%s, %s, %s", p1, p2, p3);
//printf('\n');
//printf("\n");
return 0;
}
編譯運行雖然有警告,但是沒有段錯誤,那么段錯誤肯定是打印語句那里了
把打印語句去掉注釋
#include <stdio.h>
int main()
{
char* p1 = 1 ;
char* p2 = '1';
char* p3 = "1";
printf("%s, %s, %s", p1, p2, p3);
//printf('\n');
//printf("\n");
return 0;
}
編譯運行后果然出現(xiàn)了段錯誤
究竟是怎么回事呢?接著往下看
二、小貼士
- 字符字面量被編譯為對應(yīng)的 ASCII 碼
- 字符串字面量被編譯為對應(yīng)的內(nèi)存地址
- printf 的第一個參數(shù)被當(dāng)成字符串內(nèi)存地址
- 內(nèi)存的低地址空間不能在程序中隨意訪問
三、程序?qū)嵗治?
注意:字符 '1' 所對應(yīng)的 ASCII 碼就是十六進制的 0x00000031
0x08048000這個內(nèi)存地址是非常特別的內(nèi)存地址,所有低于這個地址的內(nèi)存空間是不能隨便訪問的,一旦訪問,就會發(fā)生段錯誤。程序里面的字符串編譯之后得到的地址都是大于0x08048000,所以在程序中可以訪問。
所以說 1 和 '1' 的內(nèi)存地址都不能訪問,訪問了就會出現(xiàn)段錯誤。如果把printf("%s, %s, %s", p1, p2, p3); 注釋打印下面的語句會發(fā)生什么?
#include <stdio.h>
int main()
{
char* p1 = 1 ;
char* p2 = '1';
char* p3 = "1";
//printf("%s, %s, %s", p1, p2, p3);
printf('\n');
printf("\n");
return 0;
}
輸出結(jié)果如下,還是會發(fā)生段錯誤:
四、程序?qū)嵗治?
'\n' 字符字面量還是處于地址值區(qū)間,所以訪問這個地址就會發(fā)生段錯誤。
如果把 printf('\n'); 注釋掉
#include <stdio.h>
int main()
{
char* p1 = 1 ;
char* p2 = '1';
char* p3 = "1";
//printf("%s, %s, %s", p1, p2, p3);
//printf('\n');
printf("\n");
return 0;
}
編譯就不會出錯,而且換了一行:
五、容易混淆的代碼
這段代碼的本意是如果字符 c 是 "\t" 或者 " " 或者 "\n",,則輸入字符。
#include <stdio.h>
int main()
{
char c = " ";
while( (c == "\t") || (c == " ") || (c == "\n") )
{
scanf("%c", &c);
}
return 0;
}
編譯運行后,發(fā)現(xiàn)程序并沒有讓我們輸入:
在進行解釋之前,先看下面這樣會發(fā)生什么:
分析:
1)編譯后字符串"string"的內(nèi)存地址被賦值給變量 c
2)內(nèi)存地址占用 4個字節(jié),而變量 c 只占用1個字節(jié)
3)由于類型不同,賦值后產(chǎn)生截斷
所以說 char c = " "; 賦值后產(chǎn)生截斷,那么 while 里面的語句也就不會執(zhí)行了,這段程序是混淆了字符和字符串的概念。
可以這么改:
#include <stdio.h>
int main()
{
char c = ' ';
while( (c == '\t') || (c == ' ') || (c == '\n') )
{
scanf("%c", &c);
}
return 0;
}
這樣就能正確了:
六、小結(jié)
- 單引號括起來的單個字符代表整數(shù)
- 雙引號括起來的字符代表字符指針
- 編譯器接受字符和字符串的比較,無任何意義
- 編譯器允許字符串對字符變量賦值,只能得到錯誤
原文鏈接:https://blog.csdn.net/weixin_43129713/article/details/123333915
相關(guān)推薦
- 2022-07-21 H5 開發(fā)內(nèi)嵌頁面跨域問題
- 2023-02-25 C++中vector迭代器失效與深淺拷貝問題詳析_C 語言
- 2023-01-17 Pytorch-Geometric中的Message?Passing使用及說明_python
- 2023-05-05 Python?pip更新的兩種方式詳解_python
- 2022-08-29 Python神器之Pampy模式匹配庫的用法詳解_python
- 2022-05-15 ASP.NET?CORE實現(xiàn)跨域_實用技巧
- 2022-06-24 PyTorch中permute的基本用法示例_python
- 2022-05-06 C語言基礎(chǔ)知識點指針的使用_C 語言
- 最近更新
-
- 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被代理目標對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支