青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

zhgw01

C++的重載與虛函數

其實真正要說的是虛函數,不過其中要扯倒重載,所以順便也說了下重載

1. 重載
1.1 簡單重載
      在C++中,是允許同名函數的存在
int add(int i,int j);
float add(float i,float);
      而在c中,函數名是唯一的,所以為了區分int和float版本的add,你需要給它們起不同的名字,比如將int的命名為add_int,將float的命名為add_float,這樣做的壞處就是程序員要記住很多的函數名,雖然這些函數的功能是一樣的,而且也不直觀。
      重載函數的存在,使得這種情況不在存在,它可以根據參數的類型,自動調用合適的重載函數,程序員只要記住要使用加法是調用add函數就可以,不同再像C中那樣猜測int版的add函數函數名是怎樣的,float版的函數名又是怎樣的。實際上,這兩個版本的add函數名在編譯后是不一樣的,編譯器自動為它們進行了修飾,比如int的修飾成add_int_int,float的修飾成add_float_float,不過這都只是編譯生成后的結果,對程序員來說,他只需知道add這個函數,但他要調用的時候,比如使用了int參數,編譯器根據參數推出該調用的版本,這里就是add_int,所有這一切程序員都是看不見的,也不需要關心,從而減輕了程序員的工作。
      從上面的說法也可以看出,函數重載也不是能亂重載的,重載的要求是:
      1. 函數的參數類型不一樣,像上面的int和float的
      2. 函數的參數個數不一樣
      這是因為編譯器在修飾生成的函數名時,一般用所有的參數類型來進行修飾,比如void add(int,float) 修飾成add_int_float,void add(int,int,int) 修飾成add_int_int_int,這樣符合上面2條要求的重載函數最后生成的函數名是不一樣的。
      可能有人會認為,為什么不用返回值來區分,如果編譯器能推測出函數調用該返回什么值那自然沒什么問題,但很多時候,往往只是調用函數,使用函數的副作用,而不要求返回值,這個時候編譯器就推測不出了,比如
 
void f();
int f();

int main()
{
  f();
}

這個時候編譯器怎么知道調用哪個函數
1.2 類中的重載函數
不僅僅是全局函數可以重載,類中的函數也可以重載
class Base
{
public:
   
int f() const{
      cout
<<"Base:f()"<<endl;
      
return 1;
   }

   
   
int f(stringconst{
      
return 1;
   }

}
;

這看起來跟全局的沒什么區別,但是當涉及到繼承的時候,事情就變得麻煩起來
class Derived1: public Base
{
public:
  
//Redefinition
  int f() const{
    cout
<<"Derived1:f()"<<endl;
    
return 1;
  }

}
;

class Derived2: public Base
{
public:
  
//change Return type
  void f()const
  
{
    cout
<<"Derived2:f()"<<endl;
   }

}


class Derived3: public Base
{
public:
  
//change argument list
  int f(intconst
  
{
    cout
<<"Dervide3:f()"<<endl;
    
return 1;
   }

}

子類中定義了跟父類同名的函數,這個時候該如何辦?其實說起來也很簡單,只要子類定義了跟父類同名的函數,不管是重寫了函數內容(Dervied1),改變了返回類型(Derived2),還是改變了參數列表(Derived3),結果都一樣,子類中的同名函數將父類中的同名函數給隱藏了,只要子類中的函數是可見的,通過子類的對象調用父類的同名函數是不合法的,只能調用子類自身的同名函數。這就是所謂的名字隱藏。

2. 重寫與虛函數
2.1 基本知識
虛函數在多態中經常用到。你只要有一個基類的指針或引用,編譯器會為你調用該指針真正對應的函數
class Base
{
public
  
virtual void f()
  
{
   cout
<<"Base:f()"<<endl;
   }

}
;

class Derived:public Base
{
public:
   
//You can also ingore virtual here
   virtual void f()
   

     cout
<<"Derived:f()"<<endl;
   }

}
;

int main()
{
  Base
* p=new Derived();
  p
->f();
  delete p;
}


程序輸出Derived::f(),這就是虛函數的作用。你可以不用關心基類指針到底指向那個子類,編譯器會為你調用正確的函數。
這是因為編譯器使用了晚捆綁的緣故。
當一個類中有一個虛函數時(可以是因為在類中聲明了一個虛函數,也可以是因為基類中有虛函數,通過繼承得到)。編譯器就為這個類創造一個虛表(VTABLE),它當中的虛函數位置是固定的,即使被繼承到子類中也一樣。當定義了一個這個類的對象時,編譯器會在這個對象中放入一個虛指針(VPTR)指向這個表。當調用虛函數時,編譯器在匯編代碼中插入
一段代碼,這個代碼首先找到虛表,然后在通過偏移調用正確的函數。
當一個帶有虛函數的基類被繼承時,這個VTABLE會被完整賦值,當然對應的函數地址會改成子類中的函數地址。如果子類另外聲明了虛函數,就會在原來的虛函數后面添加上新的條目。
重寫其實就是在子類中對父類的虛函數進行重定義,因為一般子類有自己的特性。
2.2 虛函數與重載
如果子類中只是改寫了父類中虛函數的內容,這就只是重寫(overriding),函數前面的virtual可以忽略掉
但如果子類中改變了父類中虛函數的參數類型或個數,那么父類中的同名函數就會被隱藏掉,這同普通的重載一樣,有一點不一樣的是,不可以通過改變返回類型來隱藏父類中的同名函數。
2.3 切片
當用子類對象來初始化父類時(如函數中的call by value),新生成的父類對象會正確初始化它自身的vtable,而不會使用子類的vtable。

注:
雖然通過基類指針調用虛函數,最后調用的是子類的函數,但是如果使用的確實基類的默認參數
class Base
{
public:
  
virtual void f(int i=0)
  
{
    cout
<<i<<endl;
   }

}
;

class Derived: public Base
{
public:
  
virtual void f(int i=1)
  
{
    cout
<<i<<endl;
  }

}
;


int main()
{
  Base
* p=new Derived();
  p
->f(); //輸出的是0
}

  

 
注2:
發生在private繼承時的問題,父類中的虛函數是private的,當它被private繼承時,子類是無法訪問到這個函數的,不過子類仍然可以override這個函數
class Base
{
public:
  
void nvi()
  
{
     vfun();
  }

private:
  
virtual void vfun()
  
{
    cout
<<"Base::vfun()"<<endl;
   }

}
;

class Derived1:private Base
{
 
public:
   
void df()
   
{
     nvi(); 
//調用base的nvi,由于這里沒有override vfun,所以輸出的是Base:vfun()
   }


//事實上,這里不能直接調用base中的vfun,因為它是private繼承來的
}


class Derived2:private Base
{
public:
  
void df()
  
{
    nvi();
//調用了Dervied2的vfun
   }

private:
  
void vfun();//要想override,必須重新聲明
}
;

void Derived2::vfun()
{
  cout
<<"Derived2::vfun()"<<endl;
}

posted on 2008-10-17 14:45 apacs 閱讀(5536) 評論(1)  編輯 收藏 引用 所屬分類: c++

Feedback

# re: C++的重載與虛函數 2012-07-26 11:54 曾經半夏

您這里說的 名字隱藏:
子類中定義了跟父類同名的函數,這個時候該如何辦?其實說起來也很簡單,只要子類定義了跟父類同名的函數,不管是重寫了函數內容(Dervied1),改變了返回類型(Derived2),還是改變了參數列表(Derived3),結果都一樣,子類中的同名函數將父類中的同名函數給隱藏了,只要子類中的函數是可見的,通過子類的對象調用父類的同名函數是不合法的,只能調用子類自身的同名函數。這就是所謂的名字隱藏。
和后面虛函數與重載中所說的:
但如果子類中改變了父類中虛函數的參數類型或個數,那么父類中的同名函數就會被隱藏掉,這同普通的重載一樣,有一點不一樣的是,不可以通過改變返回類型來隱藏父類中的同名函數。
應該不是一個概念吧?
因為我這里可以通過直接引用父類的虛函數f()來調用,沒有在子類隱藏掉父類的函數。
class Base
{
public:
virtual void f()
{
cout<<"Base:f()"<<endl;
}
};

class Derived:public Base
{
public:

void f(int i=0)
{
cout<<"Derived:f()"<<endl;
}

};
int _tmain(int argc, _TCHAR* argv[])
{
Base* p=new Derived();
p->f();
delete p;
int i;
cin>>i;


return 0;
}
  回復  更多評論   


My Links

Blog Stats

常用鏈接

留言簿(1)

隨筆分類

隨筆檔案

搜索

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            久久香蕉精品| 伊人精品成人久久综合软件| 亚洲丶国产丶欧美一区二区三区| 国产精品国产三级欧美二区| 午夜老司机精品| 亚洲欧美日韩成人| 欧美中文日韩| 亚洲理论在线| 久久久综合香蕉尹人综合网| 一区福利视频| 亚洲国产精品一区二区久 | 国产亚洲欧美一区二区| 狠狠色狠狠色综合系列| 欧美美女视频| 欧美丝袜一区二区三区| 久久精品人人爽| 麻豆国产精品一区二区三区 | 国产精品乱人伦中文| 久久精品夜夜夜夜久久| 久久香蕉国产线看观看网| 一区二区精品| 久久久亚洲高清| 欧美日本国产在线| 欧美sm重口味系列视频在线观看| 欧美精品亚洲二区| 国产真实乱子伦精品视频| 亚洲精品中文字幕有码专区| 一区二区亚洲精品国产| 亚洲伦理久久| 久久久精品国产99久久精品芒果| 亚洲深夜av| 亚洲精品少妇| 久久成人免费日本黄色| 午夜精彩国产免费不卡不顿大片| 久久精品九九| 久久综合精品国产一区二区三区| 亚洲精品一区二| 亚洲国产精品一区二区www在线 | 欧美国产激情二区三区| 一本色道**综合亚洲精品蜜桃冫| 久久综合导航| 狂野欧美一区| 久久色中文字幕| 国产精品一区二区视频| 99re6热只有精品免费观看| 久久久五月婷婷| 亚洲免费影视| 午夜激情一区| 亚洲欧美日韩天堂| 欧美日韩免费一区二区三区| 欧美午夜在线观看| 欧美日韩亚洲视频| 欧美性淫爽ww久久久久无| 欧美极品在线播放| 欧美日本国产精品| 亚洲黄色影片| 亚洲一区二区三区免费观看| 亚洲视频在线看| 亚洲欧美色婷婷| 亚洲精品精选| 亚洲免费视频在线观看| 欧美日韩综合精品| 国产欧美 在线欧美| 国产精品综合| 新67194成人永久网站| 在线一区二区三区做爰视频网站| 亚洲欧美www| 久久亚洲视频| 尤物网精品视频| 一本色道久久综合狠狠躁篇的优点| 一区二区日韩欧美| 久久高清一区| 亚洲激情自拍| 欧美另类videos死尸| 国产精品揄拍500视频| 亚洲欧洲av一区二区三区久久| 久久乐国产精品| 久久久久久久久久码影片| 伊人天天综合| 亚洲美女黄网| 国产色产综合色产在线视频| 亚洲国产专区| 亚洲激情在线观看| 亚洲欧美国产一区二区三区| 久久精品日产第一区二区| 欧美精品日本| 精品va天堂亚洲国产| 一区二区三区国产精华| 噜噜噜躁狠狠躁狠狠精品视频| 久久久久网址| 在线天堂一区av电影| 91久久极品少妇xxxxⅹ软件| 欧美激情按摩在线| 亚洲欧美精品中文字幕在线| 蜜臀久久99精品久久久久久9| 国产精品盗摄久久久| 久久精品视频免费| 欧美二区不卡| 久久国产黑丝| 欧美久久一区| 亚洲美女黄色片| 午夜欧美大尺度福利影院在线看| 欧美日本三级| 久久国产精品久久精品国产| 一本大道久久精品懂色aⅴ| 国产欧美日韩另类一区| 亚洲国产aⅴ天堂久久| 久久gogo国模裸体人体| 国产精品揄拍500视频| 亚洲第一精品福利| 久久久在线视频| 国产性天天综合网| 亚洲人在线视频| 欧美日韩国产高清| 久久男人av资源网站| 国产精品日韩一区| 91久久久国产精品| 伊人色综合久久天天五月婷| 久久亚洲电影| 国产精品久久7| 亚洲日本理论电影| 欧美三区美女| 亚洲视频免费看| 免费观看日韩| 久久综合久色欧美综合狠狠 | 亚洲精品一线二线三线无人区| 国产一区二区三区不卡在线观看 | 免费日韩av| 免费久久99精品国产自在现线| 国产午夜精品一区二区三区视频 | 玖玖综合伊人| 国语自产精品视频在线看8查询8| 久久久xxx| 久久精品国产精品亚洲精品| 中文精品视频| 香蕉久久a毛片| 在线国产精品播放| 亚洲欧美日韩国产综合| 国精产品99永久一区一区| 欧美在线在线| 国产精品―色哟哟| 久久艳片www.17c.com| 国产日韩精品一区二区三区在线| 一本色道久久综合亚洲精品高清| 日韩视频免费观看高清完整版| 免费成人高清视频| 亚洲午夜av电影| 亚洲永久字幕| 亚洲免费小视频| 久久天天狠狠| 欧美激情一区二区三区在线视频观看| 欧美国产日产韩国视频| 亚洲午夜三级在线| 欧美在线播放视频| 久久精品国产精品亚洲| 女人香蕉久久**毛片精品| 欧美成人乱码一区二区三区| 欧美另类69精品久久久久9999| 亚洲精品少妇| 欧美资源在线| 欧美日韩一二三四五区| 亚洲裸体俱乐部裸体舞表演av| 国产日韩亚洲欧美精品| 欧美激情影音先锋| 国产欧美日韩一区二区三区在线观看| 久热综合在线亚洲精品| 欧美性色综合| 久久爱www.| 性做久久久久久久免费看| 国产在线精品自拍| 欧美激情国产精品| 午夜日韩在线| 亚洲日本中文字幕免费在线不卡| 国产综合色在线视频区| 亚洲美女精品成人在线视频| 小黄鸭精品密入口导航| 欧美日韩一区二区视频在线 | 国产一区三区三区| 国产精品99久久久久久白浆小说| 性做久久久久久免费观看欧美| 精品成人在线观看| 国产精品久久久久久妇女6080| 久久免费国产| 亚洲欧美综合精品久久成人| 亚洲一区二区三区高清| 欧美精品国产| 久久福利资源站| 久久精品99无色码中文字幕| 欧美日韩一区在线播放| 久久精品亚洲精品| 一区二区三区视频观看| 亚洲第一成人在线| 久久久久久久久久久一区| 国产日韩综合一区二区性色av| 夜夜躁日日躁狠狠久久88av| 99亚洲精品| 欧美激情按摩在线| 久久久久久亚洲精品杨幂换脸| 宅男噜噜噜66一区二区 | 久久精品国产99精品国产亚洲性色|