欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語(yǔ)言中的隱式函數(shù)聲明

 更新時(shí)間:2016年01月17日 16:51:31   投稿:hebedich  
在c語(yǔ)言里面開來(lái)還是要學(xué)習(xí)c++的編程習(xí)慣,使用函數(shù)之前一定要聲明。不然,即使編譯能通過(guò),運(yùn)行時(shí)也可能會(huì)出一些莫名其妙的問(wèn)題。

1 什么是C語(yǔ)言的隱式函數(shù)聲明

在C語(yǔ)言中,函數(shù)在調(diào)用前不一定非要聲明。如果沒(méi)有聲明,那么編譯器會(huì)自動(dòng)按照一種隱式聲明的規(guī)則,為調(diào)用函數(shù)的C代碼產(chǎn)生匯編代碼。下面是一個(gè)例子:

int main(int argc, char** argv)
{
  double x = any_name_function();
  return 0;
}

單純的編譯上述源代碼,并沒(méi)有任何報(bào)錯(cuò),只是在鏈接階段因?yàn)檎也坏矫麨閍ny_name_function的函數(shù)體而報(bào)錯(cuò)。

[smstong@centos192 test]$ gcc -c main.c
[smstong@centos192 test]$ gcc main.o
main.o: In function `main':
main.c:(.text+0x15): undefined reference to `any_name_function'
collect2: ld 返回 1

之所以編譯不會(huì)報(bào)錯(cuò),是因?yàn)镃語(yǔ)言規(guī)定,對(duì)于沒(méi)有聲明的函數(shù),自動(dòng)使用隱式聲明。相當(dāng)于變成了如下代碼:

int any_name_function();
int main(int argc, char** argv)
{
  double x = any_name_function();
  return 0;
}

2 帶來(lái)的問(wèn)題

2.1 隱式聲明函數(shù)名稱恰好在鏈接庫(kù)中存在,但返回非int類型

前面給出的例子,并不會(huì)造成太大影響,因?yàn)樵阪溄与A段很容易發(fā)現(xiàn)存在的問(wèn)題。然而下面這個(gè)例子則會(huì)造成莫名的運(yùn)行時(shí)錯(cuò)誤。

#include <stdio.h>
int main(int argc, char** argv)
{
  double x = sqrt(1);
  printf("%lf", x);
  return 0;
}

gcc編譯鏈接

[smstong@centos192 test]$ gcc -c main.c
main.c: 在函數(shù)‘main'中:
main.c:6: 警告:隱式聲明與內(nèi)建函數(shù)‘sqrt'不兼容
[smstong@centos192 test]$ gcc main.o

運(yùn)行結(jié)果

1.000000

編譯時(shí)會(huì)給出警告,提示隱式聲明與內(nèi)建函數(shù)'sqrt'不兼容。gcc編譯器在編譯時(shí)能夠自動(dòng)在常用庫(kù)頭文件(內(nèi)建函數(shù))中查找與隱式聲明同名的函數(shù),如果發(fā)現(xiàn)兩者并不相同,則會(huì)按照內(nèi)建函數(shù)的聲明原型去生成調(diào)用代碼。這往往也是程序員預(yù)期的想法。
上面的例子中隱式聲明的函數(shù)原型為:

int sqrt(int);

而對(duì)應(yīng)的同名內(nèi)建函數(shù)原型為:

double sqrt(double);

最終編譯器按照內(nèi)建函數(shù)原型進(jìn)行了編譯,達(dá)到了預(yù)期效果。然而gcc編譯器的這種行為并不是C語(yǔ)言的規(guī)范,并不是所有的編譯器實(shí)現(xiàn)都有這樣的功能。同樣的源碼在VC++2015下編譯運(yùn)行的結(jié)果卻是:

VC++編譯

warning C4013: “sqrt”未定義;假設(shè)外部返回 int

運(yùn)行結(jié)果

2884223.000000

顯然,VC++編譯器沒(méi)有沒(méi)有所謂的“內(nèi)建函數(shù)”,只是簡(jiǎn)單的按照隱式聲明的原型,生成調(diào)用sqrt函數(shù)的代碼。由于返回類型和參數(shù)類型的不同,導(dǎo)致錯(cuò)誤的函數(shù)調(diào)用方式,產(chǎn)生莫名奇妙的運(yùn)行時(shí)錯(cuò)誤。

對(duì)著這種情況,由于返回類型的不同,兩種編譯器都可以給出警告信息,至少能引起程序員的注意。而下面這種情況,則更加隱蔽。

2.2 隱式聲明函數(shù)名稱恰好在鏈接庫(kù)中存在,且返回int類型

測(cè)試代碼如下:

#include <stdio.h>

int main(int argc, char** argv)
{
  int x = abs(-1);
  printf("%d", x);
  return 0;
}

此時(shí),由于隱式聲明的函數(shù)原型與gcc的內(nèi)建函數(shù)原型完全相同,所以gcc不會(huì)給出任何警告,結(jié)果也是正確的。
而VC++則仍然會(huì)給出警告:warning C4013: “abs”未定義;假設(shè)外部返回 int。

無(wú)論如何,隱式聲明的函數(shù)原型與庫(kù)函數(shù)完全相同,所以鏈接運(yùn)行都是沒(méi)有問(wèn)題的。

下面,稍微改動(dòng)一下代碼:

#include <stdio.h>

int main(int argc, char** argv)
{
  int x = abs(-1,2,3,4);
  printf("%d", x);
  return 0;
}

gcc下編譯鏈接沒(méi)有任何報(bào)錯(cuò)。

gcc編譯鏈接

[smstong@centos192 test]$ gcc -c main.c
[smstong@centos192 test]$ gcc main.o

可見(jiàn),gcc的內(nèi)建函數(shù)機(jī)制并不關(guān)心函數(shù)的參數(shù),只是關(guān)心函數(shù)的返回值。

vc++編譯鏈接

warning C4013: “abs”未定義;假設(shè)外部返回 int

雖然這個(gè)例子的運(yùn)行結(jié)果都是正確的,但是這種正確是“碰巧”的,因?yàn)轭~外的函數(shù)參數(shù)并沒(méi)有影響到結(jié)果。這種偶然正確是程序中要避免的。

3 編程中注意事項(xiàng)

C語(yǔ)言的隱式函數(shù)聲明,給程序員帶來(lái)了各種困惑,給程序的穩(wěn)定性帶來(lái)了非常壞的影響。不知道當(dāng)初C語(yǔ)言設(shè)計(jì)者是如何考慮這個(gè)問(wèn)題的?

* 為了避免這種影響,強(qiáng)烈建議程序員重視編譯器給出的關(guān)于隱式聲明的警告,及時(shí)通過(guò)包含必要的頭文件來(lái)消除這種警告。*

對(duì)于gcc來(lái)說(shuō),前面給出的那個(gè)abs(-1,2,3,4)的特殊例子,編譯器根本不會(huì)產(chǎn)生任何警告,只能靠程序員熟悉自己調(diào)用的每一個(gè)庫(kù)函數(shù)了。

為了避免這種問(wèn)題,在C語(yǔ)言的C99版本中,無(wú)論如何都會(huì)給出警告。如gcc使用C99編譯上述代碼。

gcc -std=c99編譯

[smstong@centos192 test]$ gcc -c main.c -std=c99
main.c: 在函數(shù)‘main'中:
main.c:5: 警告:隱式聲明函數(shù)‘a(chǎn)bs'

而C++則更嚴(yán)格,直接拋棄了隱式函數(shù)聲明,對(duì)于未聲明函數(shù)的調(diào)用,將直接無(wú)法通過(guò)編譯。

g++編譯

[smstong@centos192 test]$ g++ main.c
main.c: In function ‘int main(int, char**)':
main.c:5: 錯(cuò)誤:‘a(chǎn)bs'在此作用域中尚未聲明

vc++編譯(作為C++)

error C3861: “abs”: 找不到標(biāo)識(shí)符

在函數(shù)強(qiáng)類型這一點(diǎn)上,C++確實(shí)比C更嚴(yán)格,更嚴(yán)謹(jǐn)。

相關(guān)文章

  • C++11-20 常量表達(dá)式的使用

    C++11-20 常量表達(dá)式的使用

    本文主要介紹了C++11-20常量表達(dá)式,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-10-10
  • C++?電話號(hào)碼的字母組合功能實(shí)現(xiàn)

    C++?電話號(hào)碼的字母組合功能實(shí)現(xiàn)

    這篇文章主要介紹了C++?電話號(hào)碼的字母組合,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-08-08
  • Cocos2d-x學(xué)習(xí)筆記之Hello World!

    Cocos2d-x學(xué)習(xí)筆記之Hello World!

    這篇文章主要介紹了Cocos2d-x學(xué)習(xí)筆記之Hello World!本文基于vs2010和C++語(yǔ)言開發(fā),需要的朋友可以參考下
    2014-09-09
  • C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)易學(xué)生成績(jī)管理系統(tǒng)

    C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)易學(xué)生成績(jī)管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)易學(xué)生成績(jī)管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-12-12
  • C++函數(shù)中return語(yǔ)句的使用方法

    C++函數(shù)中return語(yǔ)句的使用方法

    C++中的return語(yǔ)句是函數(shù)中一個(gè)重要的語(yǔ)句,return語(yǔ)句用于結(jié)束當(dāng)前正在執(zhí)行的函數(shù),并將控制權(quán)返回給調(diào)用此函數(shù)的函數(shù),需要的朋友可以了解下
    2012-12-12
  • C語(yǔ)言學(xué)生成績(jī)管理系統(tǒng)小設(shè)計(jì)

    C語(yǔ)言學(xué)生成績(jī)管理系統(tǒng)小設(shè)計(jì)

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言學(xué)生成績(jī)管理系統(tǒng)小設(shè)計(jì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • C語(yǔ)言模擬實(shí)現(xiàn)掃雷游戲

    C語(yǔ)言模擬實(shí)現(xiàn)掃雷游戲

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言模擬實(shí)現(xiàn)掃雷游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-01-01
  • C++實(shí)現(xiàn)神經(jīng)BP神經(jīng)網(wǎng)絡(luò)

    C++實(shí)現(xiàn)神經(jīng)BP神經(jīng)網(wǎng)絡(luò)

    這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)神經(jīng)BP神經(jīng)網(wǎng)絡(luò),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-05-05
  • C語(yǔ)言模擬實(shí)現(xiàn)C++的繼承與多態(tài)示例

    C語(yǔ)言模擬實(shí)現(xiàn)C++的繼承與多態(tài)示例

    本篇文章主要介紹了C語(yǔ)言模擬實(shí)現(xiàn)C++的繼承與多態(tài)示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-05-05
  • C++命名空間namespace的介紹與使用

    C++命名空間namespace的介紹與使用

    今天小編就為大家分享一篇關(guān)于C++命名空間namespace的介紹與使用,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧
    2018-12-12

最新評(píng)論