重載是編譯期
A. C語言可不可以重載
1.從C語言的語法設計來說是不支持的,早期的C編譯器未考慮過函數重載這一功能,所以就會有那麼多類似的函數abs,labs,fabs等等(每種類型都要考慮一個不同的函數名)。
2.C語言標准就規定不允許同一作用域中兩個函數重名。和C語言相關的域有兩種,一個是給編譯器看得,還有一種是給鏈接器看的,而給編譯器看的這種域主要是關於
變數的可見范圍,而鏈接器看到的那個是叫文件域,而全局域可能要跨越好幾個文件域,這樣只要兩個同名函數用static修飾,並且在不同的文件中,就不會
沖突。不過這個實際上已經超越了編譯器的范圍,表面看起來兩個函數同名,但是實際上編譯器為鏈接器產生是指向的同一個函數地址。
3.函數重載是一個編譯期行為,主要是通過name mangling來產生不同的匯編符號,讓linker可以正確的link代碼。
4.但是,C語言作為一個具有超強功能的底層語言,是有辦法進行模擬函數重載的。那就是函數指針。最簡單的例子就是qsort函數。這個函數可以傳遞一個函數指針變數,通過不同的函數指針,可以對不同的數據類型就行相同的qsort操作,從某種層面來說相當於函數重載。
B. c++函數的重載如何被編譯器識別的
C++將會對重載的函數進行名稱修飾或者叫名稱矯正
比如
intfun(inta)
intfun(floatb)
這樣的重載函數 在編譯器下就可能是?fun@@YXX 和?fun@@XXY這樣的進行了貌似無意義的修飾 用於編譯器的識別
C. java重載和重寫在編譯期和運行期的問題
對你的問題比較好奇,所以親測了一下,證實了之前的想法
輸出如下:
g(Super)
Sub.f()
這也是重寫和重載的含義推導的正確結果。
不會出現你說的現象,再確認一下吧。
D. 重載的編程語言中的重載
編程中重載的定義:函數名相同,函數的參數列表不同(包括參數個數和參數類型),至於返回類型可同可不同。
重載是可使函數、運算符等處理不同類型數據或接受不同個數的參數的一種方法,關於重載一詞在詞義上有兩種不同的說法: 重載是一種多態(如C++,Java),有四種形式的多態:
1.虛函數多態
2模板多態
3重載
4轉換
所謂的動態和靜態區分是另一種基於綁定時間的多態分類,嚴格來說,重載是編譯時多態,即靜態多態,根據不同類型函數編譯時會產生不同的名字如int_foo和char_foo等等,以此來區別調用。故重載仍符合多態定義——通過單一標識支持不同特定行為的能力,只是重載屬於靜態多態,而不是通過繼承和虛函數實現的動態多態。 重載(overloaded)和多態無關,真正和多態相關的是覆蓋(inheritance)。
當派生類重新定義了基類的虛擬方法後,基類根據賦給它的不同的派生類引用,動態地調用屬於派生類的對應方法,這樣的方法調用在編譯期間是無法確定的。因此,這樣的方法地址是在運行期綁定的(動態綁定)。
重載只是一種語言特性,是一種語法規則,與多態無關,與面向對象也無關。
不過針對所謂的第二種重載,有一個專門的名詞--重寫或重定義。重載與重寫的區別就在於是否覆蓋,重寫一般多發生在不同的類且存在繼承關系之間,而重載多是在一個類里或者一塊代碼段里。
特點:
由於重載可以在同一個類中定義功能類似的函數,這給程序員管理類的相似函數提供了極大的方便。例如,在一個定義圓的類中,需要設定圓心和半徑來確定一個圓對象,程序員不需要設定setRadius(float r)和SetPoint(float x,float y)兩個不同名函數,而只需要設定一個CSetCicle函數名就夠了。在這個簡單的例子中重載並沒有明顯的優勢,可是當一個類中相似功能函數有幾十、上百個的時候,重載的優勢就顯現出來了,這時程序員不需要去記這么繁多的函數名,可以把更多的精力放在程序本身上。重載的方法只屬於子類。
函數:
1.函數名必須相同,返回值可以相同,也可以不同,但是特徵標必須不同。是函數名來確定函數的不同,是特徵標是函數可以重載。編譯器首先選在函數名,然後再根據特徵標在眾多重載的函數中找到合適的。
2.匹配函數時,編譯器將不區分類型引用和類型本身,也不區分const和非const變數。(小註:因為這些在定義和聲明時可能不同,但是在調用時都是一樣的,編譯器將無法區分)。但是值得注意的是,形參與const形參的等價性僅適於非引用形參。有const引用形參的函數與有非const引用形參的函數是不同的。類似的,如果函數帶有指向const類型的指針形參,則與帶有指向相同類型的非const對象的指針形參的函數不相同。
3.名稱修飾(name decoration)。編譯器將根據原型中指定的形參對每個函數名進行加密。
重定義:
被重載的函數有不同版本,這些函數地位是一樣的,可以根據特徵標的不同選擇不同的函數。被重定義的函數也有不同的版本,但是你不能隨意選擇,你只能選擇最新的版本,被重定義多發生在類之間的繼承里。
4.函數會有那麼多版本,那麼編譯將選哪一個呢。當然,理想情況是,實參與形參的數據類型完全匹配,但是當不完全匹配時會怎樣呢?這就要牽扯到c++里復雜的類型轉換了。
在重載及函數模板重載里,編譯器選擇函數,要經過以下三步,這個過程稱為重載解析。
第一步:創建候選函數列表,其中包含有與被調函數名稱相同的函數與模板函數。
第二步:使用候選函數列表創建可行函數列表。這些都是參數數目正確的函數。
第三步:確定是否有最佳可行的函數。如果有,則使用。
確定最佳函數,只考慮其特徵標,而不考慮返回類型(也無從考慮,但是要是硬想辦法的話,也有,不過沒有必要為了不必要的性能而浪費資源)。確定最佳函數,匹配特徵標要依次經過以下判斷:(1)完全匹配(常規函數優於模板;允許無關緊要的轉換)(2)提升匹配(如char和short自動轉換為int)(3)標准轉換(int轉換為char,long轉換為double)(4)用戶自定義的轉換(如類聲明中定義的轉換函數)。
完全允許無關緊要的轉換,這些轉換包括引用,指針與實體之間,數組與指針之間,函數與函數指針之間,const與非const等等。
其次還要注意匹配的優先順序。1,指向非const數據的指針和引用優先於const的指針和引用參數匹配(這種優先順序只有當指針或引用出現時產生)。2,非模板函數,優於模板函數,顯示具體化的模板將優於隱式具體化的模板,總之較具體的優先(注意,具體並不是由於顯隱決定的,術語「最具體」是指編譯器推斷使用哪種類型時執行的轉換最少)。 通常,派生類繼承基類的方法,因此,在調用對象繼承方法的時候,調用和執行的是基類的實現.但是,有時需要對派生類中的繼承方法有不同的實現.
例如,假設動物類存在跑的方法,從中派生出馬和狗,馬和狗的跑得形態是各不相同的,因此同樣方法需要兩種不同的實現,這就需要重新編寫基類中的方法.
重寫基類方法就是修改它的實現或者說在派生類中重新編寫 //java代碼//方法重寫publicclassFather{publicvoidovel(inti){/*dosomething...*/}publicStringovef(){/*dosomething...*/return***;}}publicclassSonextendsFather{publicvoidovel(inti){/*doothersomething...*/}publicStringovef(){/*doothersomething...*/returnXXX;}}
E. 什麼是重載
你在街上一個咖啡廳偶遇了她,一場轟轟烈烈的感情在你們兩個身上發生。如今十年過去了,你在那個咖啡廳再次偶遇上了她,你突然發現原來現在你對她已經沒啥感覺。
雖然是同樣的人,同樣的地方!但你的感情卻變了,你的感情在這里被重載了。
明白了把,重載就是對於同一個概念,同一個操作,因為條件不同而生成了不同的結果,或者多了一些東西,或者少了一些東西,或者根本就完全不同
1+1=2
a+b=ab
一堆沙+一堆沙=一堆沙。
同樣都是「+」,結果是不一樣的
ps:對於ls講重載效率低下,個人是不同意的,泛型編程的基礎就是重載,而泛型編程通常效率是很高的。當然這和泛型的機制有關,不過至少重側面說明效率低,並不是以為使用了重載,而是程序本身的運行機制決定了效率
F. 重載意思是什麼
重載
什麼是覆蓋和重載?
這里有一個初學者經常混淆的概念。覆蓋(override)和重載(overload)。上面說了,覆蓋是指子類重新定義父類的虛函數的做法。而重載,是指允許存在多個同名函數,而這些函數的參數表不同(或許參數個數不同,或許參數類型不同,或許兩者都不同)。其實,重載的概念並不屬於「面向對象編程」,重載的實現是:編譯器根據函數不同的參數表,對同名函數的名稱做修飾,然後這些同名函數就成了不同的函數(至少對於編譯器來說是這樣的)。如,有兩個同名函數:function func(p:integer):integer;和function func(p:string):integer;。那麼編譯器做過修飾後的函數名稱可能是這樣的:int_func、str_func。對於這兩個函數的調用,在編譯器間就已經確定了,是靜態的(記住:是靜態)。也就是說,它們的地址在編譯期就綁定了(早綁定),因此,重載和多態無關!真正和多態相關的是「覆蓋」。當子類重新定義了父類的虛函數後,父類指針根據賦給它的不同的子類指針,動態(記住:是動態!)的調用屬於子類的該函數,這樣的函數調用在編譯期間是無法確定的(調用的子類的虛函數的地址無法給出)。
因此,這樣的函數地址是在運行期綁定的(晚邦定)。結論就是:重載只是一種語言特性,與多態無關,與面向對象也無關!
G. Java里,重載的方法為何是靜態編譯,而沒有重載的方法卻是動態編譯這么設計有什麼原因嗎
java允許在一個類中,多個方法擁有相同的名字,但在名字相同的同時,必須有不同的參數,這就是重載,編譯器會根據實際情況挑選出正確的方法,如果編譯器找不到匹配的參數或者找出多個可能的匹配就會產生編譯時錯誤,這個過程被稱為重載的解析
1 普通方法的重載
普通方法的重載是Java實現多態技術的重要手段,為編程帶來了很多便利
當方法同名時,為了讓編譯器區別他們,至少需要下面之一不同
1 參數個數不同
2 對應位置上的參數類型不同
不允許參數完全相同而只是返回值不同的情況出現。無法進行編譯,程序在eclips中顯示錯誤
2 構造方法的重載
見文章構造方法的繼承
重載的解析
當類的設計者提供了重載方法之後,類的使用者在使用這些方法時編譯器需要確定調用哪一個方法,確定的唯一依據是參數列表,確定的過程被稱為重載的解析。
以下舉些例子說明:
show(int a ,int b,int c) //1
show(int a ,int b,double c) //2
show(int a ,double b,double c)//3
show(double a,double b,int c) //4
下面是調用
show(1,2,3);//1,2,3,4都是可行方法所有參數完全匹配1
show(1.0,2.0,3.0);//沒有一個可行方法
show(1.0,2,3);//4是最佳可行方法
show(1,2.0,3);//3,4都是可行方法,沒有最佳可行方法,報錯
重載和覆蓋都是多態的表現,他們在某些地方很相似,很容易引起初學者的疑惑,這里將它們之間的區別總結如下
1 重載和覆蓋的方法名稱都相同,但重載要求參數列表不同,而覆蓋要求參數列表完全相同。
2 重載對於方法前面的修飾符沒有限制,而覆蓋則對這些修飾符的使用有限制
3 重載時編譯器在編譯期間就可以確定調用那一個方法,而覆蓋則有可能在運行期間才能確定。
H. Java中類載入出現在哪個階段,編譯期和運行期 類載入和類裝載是一樣的嗎
使用的類,編譯/運行時都會被載入。
載入/裝載沒有區別,翻譯的詞彙選擇因人而異了
運行 javac /java 時 加上 -verbose 選項就可以看到了。
I. JAVA 的重載是運行時決定還是編譯的時候決
重載屬於編譯時多態。運行時會根據參數的個數和類型調用相應的方法。