0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

剖析什么是C語言中的隱式函數(shù)聲明

Q4MP_gh_c472c21 ? 來源:路飯網(wǎng) ? 作者:路飯網(wǎng) ? 2021-05-25 09:38 ? 次閱讀

「1、什么是C語言的隱式函數(shù)聲明」

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

pIYBAGCsVSSAHJBCAAAN2ink3eI176.png

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


o4YBAGCsVT-AE4exAAAXtEILXzs993.png

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


pIYBAGCsVViAajJtAAAQPH_wkyI684.png

「2、帶來的問題」「2.1 隱式聲明函數(shù)名稱恰好在鏈接庫中存在,但返回非int類型」

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

#include
intmain(intargc,char**argv)
{
doublex=sqrt(1);
printf("%lf",x);
return0;
}

gcc編譯鏈接:

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

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

1.000000

編譯時(shí)會(huì)給出警告,提示隱式聲明與內(nèi)建函數(shù)’sqrt’不兼容。

gcc編譯器在編譯時(shí)能夠自動(dòng)在常用庫頭文件(內(nèi)建函數(shù))中查找與隱式聲明同名的函數(shù),如果發(fā)現(xiàn)兩者并不相同,則會(huì)按照內(nèi)建函數(shù)的聲明原型去生成調(diào)用代碼。

這往往也是程序員預(yù)期的想法。上面的例子中隱式聲明的函數(shù)原型為:

intsqrt(int);

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

doublesqrt(double);

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

VC++編譯:

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

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

2884223.000000

顯然,VC++編譯器沒有沒有所謂的“內(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ù)名稱恰好在鏈接庫中存在,且返回int類型」

測(cè)試代碼如下:

#include

intmain(intargc,char**argv)
{
intx=abs(-1);
printf("%d",x);
return0;
}

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

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

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

#include

intmain(intargc,char**argv)
{
intx=abs(-1,2,3,4);
printf("%d",x);
return0;
}

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

gcc編譯鏈接:

[smstong@centos192test]$gcc-cmain.c

[smstong@centos192test]$gccmain.o

可見,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ù)并沒有影響到結(jié)果。這種偶然正確是程序中要避免的。

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

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

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

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

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

gcc -std=c99編譯:

[smstong@centos192test]$gcc-cmain.c-std=c99

main.c:在函數(shù)‘main’中:
main.c警告:隱式聲明函數(shù)‘a(chǎn)bs’

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

g++編譯:

[smstong@centos192test]$g++main.c

main.c:Infunction‘intmain(int,char**)’:
main.c錯(cuò)誤:‘a(chǎn)bs’在此作用域中尚未聲明

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

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

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

編輯:jq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4331

    瀏覽量

    62633
  • C++
    C++
    +關(guān)注

    關(guān)注

    22

    文章

    2108

    瀏覽量

    73657
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4788

    瀏覽量

    68628

原文標(biāo)題:什么是C語言中的隱式函數(shù)聲明?

文章出處:【微信號(hào):gh_c472c2199c88,微信公眾號(hào):嵌入式微處理器】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    C語言中申請(qǐng)的堆內(nèi)存能不能自動(dòng)釋放

    C語言中申請(qǐng)的堆內(nèi)存能不能自動(dòng)釋放?每次都要手動(dòng) free 太麻煩,也容易忘記。 學(xué)過 C++ 的同學(xué),應(yīng)該首先能想到智能指針。 但是這是C語言
    的頭像 發(fā)表于 11-27 09:33 ?118次閱讀

    使用C語言實(shí)現(xiàn)函數(shù)模板

      用C語言能不能實(shí)現(xiàn)一個(gè)通用的函數(shù),既能完成整數(shù)的相加,又能完成浮點(diǎn)數(shù)的相加?
    的頭像 發(fā)表于 11-09 11:38 ?391次閱讀

    C語言中的socket編程基礎(chǔ)

    數(shù)據(jù) 步驟6:關(guān)閉socket 創(chuàng)建socket 在C語言中,創(chuàng)建socket需要使用socket()函數(shù)。這個(gè)函數(shù)需要兩個(gè)參數(shù):域
    的頭像 發(fā)表于 11-01 16:51 ?325次閱讀

    c語言中從左到右結(jié)合怎么看

    C語言中,操作符的結(jié)合性(Associativity)是指當(dāng)操作符在表達(dá)式中連續(xù)出現(xiàn)時(shí),它們?nèi)绾闻c操作數(shù)結(jié)合的順序。對(duì)于大多數(shù)二元操作符(即需要兩個(gè)操作數(shù)的操作符),C語言遵循兩種基
    的頭像 發(fā)表于 08-20 11:42 ?895次閱讀

    技術(shù)干貨驛站 ▏深入理解C語言:基本數(shù)據(jù)類型和變量

    語言的知識(shí),為后續(xù)的編程學(xué)習(xí)打下堅(jiān)實(shí)的基礎(chǔ)。1基本數(shù)據(jù)類型在C語言中,數(shù)據(jù)類型指的是用于聲明不同類型的變量或函數(shù)的一個(gè)廣泛的系統(tǒng),用于定義變
    的頭像 發(fā)表于 07-26 17:53 ?2138次閱讀
    技術(shù)干貨驛站 ▏深入理解<b class='flag-5'>C</b><b class='flag-5'>語言</b>:基本數(shù)據(jù)類型和變量

    PHP用戶定義函數(shù)詳細(xì)講解

    描述 在所有編程和腳本語言中函數(shù)是可以在程序中重復(fù)使用的語句塊。在 PHP 中,函數(shù)的概念與另一種語言(如“C”)中的概念相同。標(biāo)準(zhǔn) PH
    的頭像 發(fā)表于 03-20 14:27 ?403次閱讀

    C語言數(shù)據(jù)類型有哪些

    C 語言中,數(shù)據(jù)類型指的是用于聲明不同類型的變量或函數(shù)的一個(gè)廣泛的系統(tǒng)。變量的類型決定了變量存儲(chǔ)占用的空間,以及如何解釋存儲(chǔ)的位模式。
    發(fā)表于 03-20 10:56 ?484次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>數(shù)據(jù)類型有哪些

    嵌入系統(tǒng)中C語言結(jié)構(gòu)體的基礎(chǔ)實(shí)現(xiàn)與應(yīng)用

    C語言中的數(shù)組只能允許程序員定義存儲(chǔ)相同類型數(shù)據(jù)。但是結(jié)構(gòu)是C語言編程中允許您存儲(chǔ)不同數(shù)據(jù)類型的數(shù)據(jù)。
    發(fā)表于 03-12 14:29 ?503次閱讀
    嵌入<b class='flag-5'>式</b>系統(tǒng)中<b class='flag-5'>C</b><b class='flag-5'>語言</b>結(jié)構(gòu)體的基礎(chǔ)實(shí)現(xiàn)與應(yīng)用

    C語言中的三種形式變量

    局部變量是在一個(gè)函數(shù)、代碼塊內(nèi)部聲明的變量,只能被該函數(shù)或者代碼塊內(nèi)部應(yīng)用。局部變量在函數(shù)之外不可用。
    發(fā)表于 03-11 17:34 ?720次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言中</b>的三種形式變量

    淺談C語言中函數(shù)定義

    如果函數(shù)要使用參數(shù),則必須聲明接受參數(shù)值的變量。這些變量稱為函數(shù)的形式參數(shù)。 形式參數(shù)就像函數(shù)內(nèi)的其他局部變量,在進(jìn)入函數(shù)時(shí)被創(chuàng)建,退
    發(fā)表于 03-11 10:09 ?390次閱讀

    C語言字符串編譯函數(shù)介紹

    C語言中,字符串實(shí)際上是使用null字符O'終止的一維字符數(shù)組。因此,一個(gè)以null結(jié)尾的字符串,包含了組成字符串的字符。
    的頭像 發(fā)表于 03-07 16:18 ?514次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>字符串編譯<b class='flag-5'>函數(shù)</b>介紹

    介紹C語言中錯(cuò)誤處理和異常處理的一些常用的方法和策略

    C語言是一種低級(jí)的、靜態(tài)的、結(jié)構(gòu)化的編程語言,它沒有提供像C++或Java等高級(jí)語言中的異常處理機(jī)制,例如try-catch-finally
    的頭像 發(fā)表于 02-28 14:25 ?621次閱讀

    C語言中的可變參數(shù)介紹

    C 語言為這種情況提供了一個(gè)解決方案,它允許您定義一個(gè)函數(shù),能根據(jù)具體的需求接受可變數(shù)量的參數(shù)
    發(fā)表于 02-28 14:00 ?311次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言中</b>的可變參數(shù)介紹

    C語言中的動(dòng)態(tài)內(nèi)存管理講解

    本章將講解 C 中的動(dòng)態(tài)內(nèi)存管理。C 語言為內(nèi)存的分配和管理提供了幾個(gè)函數(shù)。這些函數(shù)可以在 頭文件中找到。
    的頭像 發(fā)表于 02-23 14:03 ?397次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言中</b>的動(dòng)態(tài)內(nèi)存管理講解

    如何解決C語言中的“訪問權(quán)限沖突”異常?C語言引發(fā)異常原因分析

    如何解決C語言中的“訪問權(quán)限沖突”異常?C語言引發(fā)異常原因分析? 在C語言中,訪問權(quán)限沖突異常通
    的頭像 發(fā)表于 01-12 16:03 ?5731次閱讀