c編譯器是程序運行的基礎,一款好的c編譯器具備優(yōu)良性能。目前,GCC是三大主流c編譯器之一。對于這款c編譯器,可能大家并非十分了解。本文中,將為大家介紹gcc常見的編譯警告以及解決方法。如果你對這款c編譯器比較感興趣,不妨繼續(xù)往下看哦。
GCC有很多的編譯選項,警告選項;指定頭文件、庫路徑;優(yōu)化選項。本文針整理一下GCC的警告選項以及gcc編譯警告整理和解決方法為中心而展開的討論。
一、GCC編譯警告總概
-w
禁止編譯警告的打印。這個警告不建議使用。大約2012年底,公司代碼進行一次大重構,另外從Codeblock集成開發(fā)環(huán)境轉向Makefile管理,Makefile里面默認使用了-w,因而代碼一直沒有警告,今年個別項目開發(fā)中發(fā)現(xiàn)一些代碼筆誤導致的BUG,而這些問題可以從編譯警告中知道。前幾個月,領導安排我來fix這些警告。為了自己,為了后人,不建議使用-w選項。
-Werror
將所有的警告當成錯誤處理。此選項謹慎建議加上。有的開源庫警告很多(大名鼎鼎的ffmpeg也有很多警告呢),一一改掉耗時耗人力,必要性也不大。最后,公司代碼加入了一個開源庫,里面有很多代碼警告,可能領導又安排我來fix了。
-Wfatal-errors
遇到第一個錯誤就停止,減少查找錯誤時間。建議加上。很多人遇到錯誤,沒有意識到從第一個開始排查。不管是編譯錯誤,還是程序運行出錯,從最開始的錯誤查起,是個好的做法。
-Wall開啟“所有”的警告。強烈建議加上,并推薦該選項成為共識。如case語句沒有default處理,有符號、無符號處理,未使用變量(特別是函數(shù)有大量未使用的數(shù)組,占用??臻g,測試發(fā)現(xiàn),開辟一個未使用的8MB的數(shù)組,程序有coredump),用%d來打印地址,或%s打印int值,等,都可以發(fā)出警告。
-Wextra
除-Wall外其它的警告。建議加上。
在GCC編譯時,加上必要的警告選項,可以避免很多低級錯誤引發(fā)的問題,我就在實際工程代碼中遇到用“==”來賦值,我自己寫的代碼也出現(xiàn)過把“=”當成判斷的。但是,有些錯誤卻不是用GCC選項能解決的。比如一般項目都會自定義調試信息打印函數(shù),但在處理可變參數(shù)類型時,往往不注意??蓞⒖嘉恼隆兑粋€可變參數(shù)類型檢查的示例》。
二、GCC編譯警告細化
上面只是大概講幾個重要的選項。由于GCC的警告選項太多了,下面盡自己能力寫一下。
-Wall選項,顧名思義,就是“所有”的意思,它包括:
[html] view plain copy-Wall包括:
-Waddress
-Warray-bounds=1 (only with -O2)
-Wc++11-compat -Wc++14-compat
-Wchar-subscripts
-Wenum-compare (in C/ObjC; this is on by default in C++)
-Wimplicit-int (C and ObjecTIve-C only)
-Wimplicit-funcTIon-declaraTIon (C and ObjecTIve-C only)
-Wbool-compare
-Wduplicated-cond
-Wcomment
-Wformat
-Wmain (only for C/ObjC and unless -ffreestanding)
-Wmaybe-uninitialized
-Wmissing-braces (only for C/ObjC)
-Wnonnull
-Wopenmp-simd
-Wparentheses
-Wpointer-sign
-Wreorder
-Wreturn-type
-Wsequence-point
-Wsign-compare (only in C++)
-Wstrict-aliasing
-Wstrict-overflow=1
-Wswitch
-Wtautological-compare
-Wtrigraphs
-Wuninitialized
-Wunknown-pragmas
-Wunused-function
-Wunused-label
-Wunused-value
-Wunused-variable
-Wvolatile-register-var
但不要被它的表面意思迷惑,要不,怎么還會有-Wextra呢。-Wextra包括(有幾個選項重復了,不懂原因):
[html] view plain copy-Wclobbered
-Wempty-body
-Wignored-qualifiers
-Wmissing-field-initializers
-Wmissing-parameter-type (C only)
-Wold-style-declaration (C only)
-Woverride-init
-Wsign-compare
-Wtype-limits
-Wuninitialized
-Wshift-negative-value
-Wunused-parameter (only with -Wunused or -Wall)
-Wunused-but-set-parameter (only with -Wunused or -Wall)
-Wchar-subscripts:
使用char類作為數(shù)組下標(因為char可能是有符號數(shù))
-Wcomment:
注釋使用不規(guī)范。如“/* */”注釋中還包括“/*”。我在項目源碼發(fā)現(xiàn)過,不止一處。
-Wmissing-braces
括號不匹配。在多維數(shù)組的初始化或賦值中經(jīng)常出現(xiàn)。下面a沒有完整被初始化,b完整初始化:
int a[2][2] = { 0, 1, 2, 3 };
int b[2][2] = { { 0, 1 }, { 2, 3 } };
-Wparentheses
括號不匹配,在運算符操作或if分支語句中,可能會出現(xiàn)此警告。
如“a&&b||c^d”會出現(xiàn)警告。下面代碼片段也會有警告
{
if (a)
if (b)
foo ();
else
bar (); // 這個else實際是if (b)的分支,不是if (a),因此,要用括號來表明其屬于哪個分支
}
這類bug隱藏得深,建議顯式地加上括號。
-Wsequence-point
如出現(xiàn)i=i++這類代碼,則報警告。-Wall默認有該警告
-Wswitch-defaultcase
沒有default時,報警告
-Wunused-but-set-parameter
設置了但未使用的參數(shù)警告
-Wunused-but-set-variable
設置了但未使用的變量警告
-Wunused-function
聲明但未使用函數(shù)
-Wunused-label
未使用的標簽,比如用goto會使用label,但在刪除goto語句時,忘了刪除label。
-Wunused-variable
未使用的變量
-Wmaybe-uninitialized
變量可能沒有被初始化。特別是在有if語句或switch語句中,最好在聲明變量時加上初始化。
下面代碼片段中,當y不是1、2、3時,x沒有明確的值,是不安全的。
{
int x;
switch (y)
{
case 1: x = 1;
break;
case 2: x = 4;
break;
case 3: x = 5;
}
foo (x);
}
-Wfloat-equal
對浮點數(shù)使用等號,這是不安全的。
{
float d = 2.0;
if (d == i)
{
。。。
}
}
-Wreturn-type
函數(shù)有返回值,但函數(shù)體個別地方?jīng)]有返回值(特別是有if判斷,可能忘記在else添加返回值)。
int foo()
{
if(a==1)
{
return ok;
}
// no return here
}
-Wpointer-sign
指針有符號和無符號的錯誤傳參。如函數(shù)使用unsigned char*,但傳入char*指針。
-Wsign-compare
有符號和無符號比較。
-Wconversion-null
-Wsizeof-pointer-memaccess
在sizeof中經(jīng)常出現(xiàn),下面代碼片段中,this為指針,4字節(jié),無法保證完整初始化類。
memset(this, 0, sizeof(this));
-Wreorder
C++出現(xiàn),構造函數(shù)中成員變量初始化與聲明的順序不一致。
-Woverflow
范圍溢出。
-Wshadow
局部變量覆蓋參數(shù)、全局變量,報警告
三、常見gcc編譯警告整理以及解決方法
1、warning: no newline at end of file
在文件最后一行加上回車鍵
解釋:在《Rationale for the C99 standard》一文中,有C99的相關信息:
A backslash immediately before a newline has long been used to continue string literals, as well as preprocessing command lines. In the interest of easing machine generation of C, and of transporting code to machines with restrictive physical line lengths, the C89 Committee generalized this mechanism to permit any token to be continued by interposing a backslash/newline sequence.
c/c++代碼的每一行后面有一個“結束符”,也就是newline。避免當被include的文件展開后,前一個文件的最后一行與后一個文件的第一行直接被連接成一行從而造成錯誤。
2、warning: comparison between pointer and integer
解釋:integer與pointer比較
3、 warning: assignment discards qualifiers from pointer target type
解釋:賦值時,取消了右值的限定。
4、 warning: passing argument 1 of ‘send’ makes pointer from integer without a cast
解釋:函數(shù)send的第一個integer型參數(shù)沒有強制轉換為pointer型
5、warning: comparison is always true due to limited range of data type
解釋:由于數(shù)據(jù)類型范圍的限制,比較結果一直為真。
6、warning: initialization from incompatible pointer type
解釋:不兼容指針類型的初始化
7、 warning: return makes pointer from integer without a cast
解釋:return使integer轉換為pointer,沒有加強制類型轉換。
8、warning: incompatible implicit declaration of built-in function ‘printf’
解釋:與內置的printf函數(shù)隱士聲明不兼容。
9、warning: initialization discards qualifiers from pointer target type
解釋:initialization取消了指針目標類型的限定。
10、warning: comparison is always false due to limited range of data type
由于類型限制,比較一直是假
11、warning: assignment from incompatible pointer type
不兼容的指針間賦值
12、warning: passing argument 1 of ‘mes_read_time’ discards qualifiers from pointer target type12、
mes_函數(shù)第一個參數(shù)的傳遞,丟棄了指針目標類型限定。
13、warning: “protocol_type” redefined
——type重定義
14、warning: ‘return’ with a value, in function returning void
在void返回類型的函數(shù)中,return返回值。
15、error: expected expression before ‘else’
else之前無表達式。
16、error: lvalue required as left operand of assignment
左值問題。
17、error: invalid storage class for function ‘XXXXXX’
在文件的某個地方,丟失了一個大括號‘}’。
四、Linux編程gcc編譯器禁止所有警告和顯示所有警告
編譯程序的時候,經(jīng)常會出現(xiàn)警告。不過對于很多經(jīng)過,程序員經(jīng)常無視它的存在,甚至覺得警告挺煩人的。
在linux編譯程序時,我們可以很方便的禁止所有警告和顯示所有警告。
gcc編譯器命令選項-Wall 用來顯示所有警告信息,而-w則用來禁止所有警告的顯示。默認是顯示警告的。
警告不是錯誤,所以錯誤會正常的顯示。
下面是命令的使用示例:
顯示所有警告信息
gcc hello.c -o hello -Wall
禁止所有警告信息
gcc hello.c -o hello -w
這是linux編程需要知道的,就在這里記錄一下,以便以后查詢。