C++跳轉語句之Goto對變量定義的影響詳解
前言
goto語句也稱為無條件轉移語句,其基本形式如下 :
語句標號由一個有效地標識符和符號";"組成,其中,標識符的命名規(guī)則與變量名稱相同,即由字母、數(shù)字和下劃線組成,且第一個字符必須是字母或下劃線。執(zhí)行goto語句后,程序就會跳轉到語句標號處,并執(zhí)行其后的語句。
通常goto語句與if條件語句連用,但是,goto語句在給程序帶來靈活性的同時,也會使得使程序結構層次不清,而且不易讀,所以要合理運用該語句。
發(fā)現(xiàn)問題
我們經常碰到有在goto后面定義變量,linux下編譯不通過的問題(報錯信息:crosses initialization of)。其實,只要注意一下就好了,今天問了一下公司前輩之后,也翻了些資料,記錄一下,加深記憶,也希望能對一些人有些許幫助。
錯誤示例代碼:
#include <iostream>
using namespace std;
int main()
{
goto Exit;
int a = 0;
Exit:
return 0;
}
報錯:
[root@localhost c-c++]# g++ goto_study.cpp goto_study.cpp: In function 'int main()': goto_study.cpp:31: error: jump to label 'Exit' goto_study.cpp:29: error: from here goto_study.cpp:30: error: crosses initialization of 'int a'
正確寫法
也不能說是正確的寫法,只能說是編譯OK的寫法。
直接上代碼:
寫法一:
改變域,變成局部變量:
int main()
{
goto Exit;
{
int a = 0;
}
Exit:
return 0;
}
寫法二
神奇的寫法:
int main()
{
goto Exit;
int a;
a = 1;
Exit:
cout << "a = " << a << endl;
return 0;
}
關鍵是還可以訪問!結果:
[root@localhost c-c++]# g++ goto_study.cpp [root@localhost c-c++]# ./a.out a = 1259648
研究
神奇的寫法
看到兩個可以編譯通過的寫法之后,最納悶的是寫法二為毛可以編譯通過,而且還能使用???
C++規(guī)定
參考[1][2]中提到了C++標準中的規(guī)定: > It is possible to transfer into a block, but not in a way that bypasses declarations with initialization. A program that jumps from a point where a local variable with automatic storage duration is not in scope to a point where it is in scope is ill-formed unless the variable has POD type (3.9) and is declared without an initializer.
意思是說:如果一個程序的執(zhí)行路徑從代碼中的點A(某個局部變量x還未定義)跳到代碼中另一點B(該局部變量x已定義,并且定義的時候有初始化),那么編譯器會報錯。這樣的跳躍可以是由于執(zhí)行goto語句,或者是switch-case造成的。所以,在寫法二中a是int類型,是一個POD類型,并且沒有初始化,所以編譯通過。但是,很明顯:如果去使用這個變量a的時候,結果是未知的,就像前輩說的,沒有意義,還不如不支持!那如果只在局部使用,完全可以用花括號括起來!網(wǎng)上也有人說到,C++規(guī)范雖然沒有明確說明這樣是錯誤的,但是變量的域的規(guī)定其實是隱性說這種做法是不可取的,見參考[4]。
隱性說明
Goto can't skip over definitions of variables, because those variables would not exist after the jump, since lifetime of variable starts at the point of definition. The specification does not seem to explicitly mention goto must not do that, but it is implied in what is said about variable lifetime.
-fpermissive標記
參考[4]中提到,g++編譯器默認是檢查的,自己可以設置編譯器的這個標記變成警告,未實踐!??!
查了下資料-fpermissive標記的作用是: 把代碼的語法錯誤作為警告,并繼續(xù)編譯進程,所以就安全起見,這個角度就不要想了,還是老老實實碼磚!
POD類型
參考[3],按照上面C++規(guī)定的說法,只要是POD類型,并且沒有初始化都是可以編譯通過的。
看一段代碼:
#include <iostream>
using namespace std;
class A{
public:
// 注意:和B不同的是有構造和析構函數(shù), 所以編譯報錯
A(){}
~A(){}
void testA(){
cout << "A::test." << endl;
}
};
class B{
public:
void testB(){
cout << "B::test." << endl;
}
};
int main()
{
goto Exit;
// int a = 1; // windows ok.linux failed!
//A classA; // failed:
B classB; // success:
classB.testB();
Exit:
classB.testB();
return 0;
}
結果:
[root@localhost c-c++]# g++ goto_study.cpp [root@localhost c-c++]# ./a.out a = 1259648 B::test.
小結:
1、以上代碼在windows和linux下均編譯通過和執(zhí)行;
2、A classA一句在windows和linux均編譯不通過!因為A有構造和析構函數(shù),不滿足條件;
3、至于int a = 1;這樣的寫法在windows(msvc)下面能夠通過就與C++規(guī)范不符了,求解釋!??!
以下是POD類型(還是看英文吧):
1、int, char, wchar_t, bool, float, double是POD類型,這些類型的long/short and signed/unsigned版本也是;
2、 指針(包括函數(shù)指針和成員指針)都是POD類型;
3、enums枚舉類型;
4、POD的const和普通變量也都是;
5、POD類型的class,struct和union也是。但要求所有的成員是public,并且沒有基類,沒有構造、析構函數(shù)和虛函數(shù)。靜態(tài)成員在這些規(guī)則下也是。
總結
1、最好不要用goto;
2、goto后面不要跳過定義和初始化的變量,如果是POD類型可以先申明再定義,是不會編譯報錯的。但是不建議這么使用,可以看到如果執(zhí)行語句跳過了賦值語句,那么變量的值是未知的,存在危險性;
3、goto后面如果是局部的變量,可以用花括號括起來構成一個局部域,就安全了。
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。
參考
[1]Getting a bunch of crosses initialization error
[2]>switch case、goto對變量定義的影響
[3]>“POD type” in C++
[4]>Statement goto can not cross pointer definition?
[5]>error: jump to label ‘foo' crosses initialization of ‘bar'
您可能感興趣的文章
- 04-02c語言沒有round函數(shù) round c語言
- 01-10深入理解C++中常見的關鍵字含義
- 01-10使用C++實現(xiàn)全排列算法的方法詳解
- 01-10c++中inline的用法分析
- 01-10用C++實現(xiàn)DBSCAN聚類算法
- 01-10全排列算法的非遞歸實現(xiàn)與遞歸實現(xiàn)的方法(C++)
- 01-10C++大數(shù)模板(推薦)
- 01-10淺談C/C++中的static與extern關鍵字的使用詳解
- 01-10深入C/C++浮點數(shù)在內存中的存儲方式詳解
- 01-10深入理解C/C++混合編程


閱讀排行
本欄相關
- 04-02c語言函數(shù)調用后清空內存 c語言調用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調用函數(shù)求fibo C語言調用函數(shù)求
隨機閱讀
- 01-10delphi制作wav文件的方法
- 01-10C#中split用法實例總結
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 04-02jquery與jsp,用jquery
- 01-10SublimeText編譯C開發(fā)環(huán)境設置
- 01-11ajax實現(xiàn)頁面的局部加載
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 08-05DEDE織夢data目錄下的sessions文件夾有什


