www.久久久久|狼友网站av天堂|精品国产无码a片|一级av色欲av|91在线播放视频|亚洲无码主播在线|国产精品草久在线|明星AV网站在线|污污内射久久一区|婷婷综合视频网站

當前位置:首頁 > 工業(yè)控制 > 電子設計自動化
[導讀]C 和 C++ 字符串字面量(String Literal)既有相同之處,又有一些區(qū)別。了解這些內容對于加深字符串字面量以及相關一些概念的理解、澄清一些常見的概念誤區(qū)不無助益。本文以普通字符串字面量 "hello" 為例總結說明如

C 和 C++ 字符串字面量(String Literal)既有相同之處,又有一些區(qū)別。了解這些內容對于加深字符串字面量以及相關一些概念的理解、澄清一些常見的概念誤區(qū)不無助益。本文以普通字符串字面量 "hello" 為例總結說明如下。

如果你發(fā)現了本文中的錯誤,或者對本文有什么感想或建議,可通過 whyglinux AT gmail DOT com 郵箱和作者聯系。

相同點:

[list][*]字符串字面量是對象

C/C++ 中的對象(Object)指的是一塊存儲區(qū)。字符串字面量是不需要創(chuàng)建過程就可使用的對象,所以它既沒有變量那樣的聲明或者定義(字符串字面量是無名對象),也不需要象動態(tài)分配的對象那樣進行動態(tài)分配。由于這個原因,用來限定變量的類型限定符(如 const、volatile)以及存儲類別指示符(如 extern、static、auto、register)不能用在修飾字符串字面量上。

[*] 數組類型

字符串字面量是數組類型的對象,因而具有數組的一切特點。關于這一點在下面還會進一步說明。

[*] 靜態(tài)存儲期

C/C++ 中對象的生存期按照其存儲性質可分為三類:靜態(tài)存儲期(static storage duration)、自動存儲期(automatic storage duration)以及動態(tài)存儲期(dynamic storage duration)。相應地,對象可根據存儲期性質分為靜態(tài)對象、自動對象和動態(tài)對象三種。

字符串字面量是靜態(tài)對象,所以在程序運行期間會一直存在。

[*] 字符串字面量是左值,而且是不可被更改的左值

例如,char s[] = "hello"; 中的 “hello” 是數組類型的左值(lvalue),用于初始化 s 數組;sizeof( "hello" ) 以及 &"hello" 中的 "hello" 也都是左值。在這些情況下,"hello" 處于左值語義上下文環(huán)境中,所以不會產生下面將要提到的數組轉換為指針的現象。

另外,有些運算不但要求其操作數是左值,還要求可變。例如,對對象進行賦值、自加、自減等運算。因為數組是不可被更改的左值,所以不能對數組進行這些操作,也就是說不存在數組類型的賦值、自加、自減等運算。

[*]字符串字面量可以轉換為指向其首第一個字符的指針

處于右值語義環(huán)境中的字符串字面量將被默認轉換為指向第一個字符的指針。例如,char* p = "hello"; 中的 “hello” 在轉換為字符指針后用于初始化指針變量 p;表達式 "hello"[0](相當于 *("hello" + 0) 或者 *"hello")中的 “hello” 也是同樣轉換為指針后參與下標運算,等等。

這種性質也是數組類型的特點。在右值語義環(huán)境下,一般類型的對象表示的值是由其存儲內容決定的;而數組類型的對象與此不同,它代表的值不是來源于其內容,而是來源于數組對象首元素所在的地址。這是數組最為特殊的地方,也是人們容易產生誤解的地方。

[*] 取址運算

字符串字面量是一個可取址的對象。例如:&"hello" 是合法的表達式。

[*] 地址常量

靜態(tài)對象的地址在編譯期間即可被確定,所以其地址(如 &"hello")是常量;而字符串字面量又可以從數組類型自動轉換為指針(如 "hello" 轉換為指針后等同于 &"hello"[0]),所以字符串字面量可以直接作為地址常量表達式來使用。

[*] 修改字符串字面量的行為是無定義的

下面的操作都試圖修改字符串字面量中的第一個字符從而改變字符串字面量,所以其結果是無定義(Undefined)的:

”hello”[0] = ‘A’; /* Undefined */
char* p = “hello”; *p = ‘A’; /* Undefined */

使用了無定義行為的程序是錯誤的;避免在程序中出現無定義行為是程序員的責任。

[/list]
區(qū)別點:
[list][*]在類型限定上的不同

C 中的字符串字面量 "hello" 是數組類型 char[6](相應地,每個字符元素是無 const 限定的 char 型);作為右值使用的時候轉換為指針類型 char*。

在 C++ 中 "hello" 是 char const [6] 類型(相應地,每個字符元素的類型是 char const);轉換為指針使用的時候是 char const*、在特殊情況下也可以是 char*。

之所以在 C 中字符串字面量不是 const 數組(也就是說每個字符元素的類型不是 char const),是因為 C 要照顧或者考慮到標準制定之前已經存在的大量代碼——那時的 C 語言還沒有 const 關鍵字,如果硬性規(guī)定為 const 數組,則 char* p = "hello"; 這樣的初始化或者 char* q; q = "hello"; 這樣的賦值就是非法的了(因為右邊的類型 char const* 不能默認轉換為左邊的類型 char* )。

同樣,為了使上述代碼能順利通過編譯過程,C++ 采取了另外一種策略:它規(guī)定了字符串字面量的類型是 const 數組,同時又特別規(guī)定了字符串字面量也可以有限制地轉換為指向非常量的指針(對于 "hello" 來說就是 char*),從而解決了上述代碼中存在的問題。不過,轉換到 char* 主要是為了兼容以前的代碼,這種轉換被 C++ 標準標記為“Deprecated”,所以在寫程序的時候不應該依賴于這種轉換。

[*]C++ 中的字符串字面量是常量,而在 C 中不是常量。

正是由于標準在類型上的不同規(guī)定造成了在 C 和 C++ 中字符串字面量常量性質上的差別。

在 C 中,除了 string literals 和 compound literals(C99 only)之外,其它的 literals 都是常量;而在 C++ 中,包括 string literals 在內的所有 literals 都是常量(注意:C++ 中不存在 compound literals。)

在現實中,經??梢钥吹接谩白址A俊眮碇复白址置媪俊钡那闆r,其實對于 C 來說這是不正確的,因為在 C 中字符串字面量不屬于常量;而對于 C++ 來說,“字符串常量”和“字符串字面量”實際上是一回事,只不過看問題的角度不同罷了。

順便提一下:C++ 中的常量可以有對象常量(如字符串字面量、const 限定的對象)和非對象常量之分,而 C 中的常量不包含對象,它們最明顯的特征就是不能進行取址運算,因此常量只能作為非左值(即右值)來使用。

[*]語法及語義上的區(qū)別

C 中的字符串字面量不是常量,它的每個字符元素也不是常量,所以字符元素的不可變性僅僅表現在語義層面,但在語法和約束條件上沒有要求。而 C++ 中字符串字面量是常量,每個字符元素也是常量,因此在語義和約束條件兩方面都要求不能改變其中的每個字符元素;另外,出于兼容性考慮 C++ 還存在著特殊情況下的向非 const 指針的轉換。

下面用具體的代碼來對以上內容進行說明。

*"hello" = 'A';

表達式 *"hello" 代表字符串字面量的第一個字符元素對象。上述語句試圖通過賦值操作改變第一個元素,當然這樣的行為在 C 和 C++ 中都是無定義的。除了這個相同點外,還有如下的一些細微的區(qū)別:

在 C++ 中,*"hello" 是一個 const 對象(其類型是 const char。注意:這里的 "hello" 不會轉換為 char* 指針、從而 *"hello" 不會是 char 類型),所以上述賦值違反了賦值號左操作數必須是一個可被改變的左值的約束條件。在此情況下,標準要求給出診斷信息。

在 C 中,*"hello" 是一個非 const 對象(其類型是 char),是一個可被改變的左值,所以不違背賦值的約束條件。在此情況下,盡管這個賦值操作是未定義的,標準對診斷信息沒有要求。

char* p = "hello";
char* q; q = "hello";
void foo( char* s ); foo( "hello" );

上面的初始化和賦值語句中 "hello" 都能轉換為 char* 指針類型,所以都是合法的。在 C++ 中,盡管 "hello" 作為指針使用時是 char const * 類型,在此情況下(如果不是 char* 類型則初始化或者賦值不能成立)基于對字符串字面量的特殊規(guī)定使它可以轉換為 char * 使用。

要注意 C++ 中字符串字面量轉換為指向非常量的指針是有限制的,僅僅在有明確的目標類型要求的情況下才能進行這樣的轉換,否則是非法的。比如下面的情況:

char* p = "hello" + 1;
char* q; q = "hello" + 1;
void foo( char* s ); foo( "hello" + 1 );

上述是合法的 C 代碼,但是作為 C++ 代碼是非法的。非法的原因在于:"hello" 轉換為 char const * 指針類型,而不能轉換為 char *,因為 + 運算符對其操作數的類型沒有轉換為 char* 這樣直接的要求(因為無論是 char const * 還是 char* 都能進行指針加法運算),所以指針加法表達式的結果仍然是 char const * 類型。這樣,上面指針的初始化或賦值操作就違反了在類型上的約束條件,需要給出診斷信息。



來源:ks990次

本站聲明: 本文章由作者或相關機構授權發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內容真實性等。需要轉載請聯系該專欄作者,如若文章內容侵犯您的權益,請及時聯系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或將催生出更大的獨角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數字化轉型技術解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關鍵字: AWS AN BSP 數字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術公司SODA.Auto推出其旗艦產品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關鍵字: 汽車 人工智能 智能驅動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務中斷的風險,如企業(yè)系統復雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務連續(xù)性,提升韌性,成...

關鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據媒體報道,騰訊和網易近期正在縮減他們對日本游戲市場的投資。

關鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數據產業(yè)博覽會開幕式在貴陽舉行,華為董事、質量流程IT總裁陶景文發(fā)表了演講。

關鍵字: 華為 12nm EDA 半導體

8月28日消息,在2024中國國際大數據產業(yè)博覽會上,華為常務董事、華為云CEO張平安發(fā)表演講稱,數字世界的話語權最終是由生態(tài)的繁榮決定的。

關鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應對環(huán)境變化,經營業(yè)績穩(wěn)中有升 落實提質增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務引領增長 以科技創(chuàng)新為引領,提升企業(yè)核心競爭力 堅持高質量發(fā)展策略,塑強核心競爭優(yōu)勢...

關鍵字: 通信 BSP 電信運營商 數字經濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術學會聯合牽頭組建的NVI技術創(chuàng)新聯盟在BIRTV2024超高清全產業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現場 NVI技術創(chuàng)新聯...

關鍵字: VI 傳輸協議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯合招商會上,軟通動力信息技術(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關鍵字: BSP 信息技術
關閉
關閉