国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > C++ > 正文

舉例剖析C++中引用的本質及引用作函數參數的使用

2020-05-23 14:07:01
字體:
來源:轉載
供稿:網友
這篇文章主要介紹了C++中引用的本質及引用作函數參數的使用,講解了函數返回值是引用的情況等一些難點,需要的朋友可以參考下
 

引用的意義與本質
1)引用作為其它變量的別名而存在,因此在一些場合可以代替指針
2)引用相對于指針來說具有更好的可讀性和實用性

C++,引用

引用本質思考:
思考、C++編譯器背后做了什么工作?

#include <iostream> using namespace std;  int main() {   int a = 10;   // 單獨定義的引用時,必須初始化;說明很像一個常量   int &b = a;   // b是a的別名   b = 11;   cout << "b--->" << a << endl;   printf("a:%d/n", a);   printf("b:%d/n", b);    printf("&a:%d/n", &a);   printf("&b:%d/n", &b);   system("pause");   return 0; } 

引用是一個有地址,引用是常量。

char *const p

引用的本質:
1)引用在C++中的內部實現是一個常指針
Type& name <--> Type*const name
2)C++編譯器在編譯過程中使用常指針作為引用的內部實現,因此引用所占用的空間大小與指針相同。
3)從使用的角度,引用會讓人誤會其只是一個別名,沒有自己的存儲空間。這是C++為了實用性而做出的細節隱藏

間接賦值成立的三個條件:
1定義兩個變量(一個實參一個形參)
2建立關聯實參取地址傳給形參
3*p形參去間接的修改實參的值

引用在實現上,只不過是把:間接賦值成立的三個條件的后兩步和二為一。
當實參傳給形參引用的時候,只不過是c++編譯器幫我們程序員手工取了一個實參地址,傳給了形參引用(常量指針)。

引用做函數參數

普通引用在聲明時必須用其它的變量進行初始化,
引用作為函數參數聲明時不進行初始化

//復雜數據類型的引用 #include <iostream> using namespace std;  struct Teacher {   char name[64];   int age; };  void printfT(Teacher *pT) {   cout << pT->age << endl; }  //pT是t1的別名 ,相當于修改了t1 void printfT2(Teacher &pT) {   //cout<<pT.age<<endl;   pT.age = 33; }  //pT和t1的是兩個不同的變量 void printfT3(Teacher pT) {   cout << pT.age << endl;   pT.age = 45; //只會修改pT變量 ,不會修改t1變量 } void main() {   Teacher t1;   t1.age = 35;    printfT(&t1);    printfT2(t1); //pT是t1的別名   printf("t1.age:%d /n", t1.age); //33    printfT3(t1);// pT是形參 ,t1 copy一份數據 給pT   //---> pT = t1   printf("t1.age:%d /n", t1.age); //35    cout << "hello..." << endl;   system("pause");   return; } 


引用的難點:函數返回值是引用(引用當左值)
當函數返回值為引用時,若返回棧變量,不能成為其它引用的初始值,不能作為左值使用;
若返回靜態變量或全局變量,可以成為其他引用的初始值,即可作為右值使用,也可作為左值使用。
C++鏈式編程中,經常用到引用。

#include <iostream> using namespace std; //返回值是基礎類型,當引用 int getAA1() {   int a;   a = 10;   return a; }  //基礎類型a返回的時候,也會有一個副本 int& getAA2() {   int a; // 如果返回棧上的引用,有可能會有問題   a = 10;   return a; }  int* getAA3() {   int a;   a = 10;   return &a; }  int main() {   int a1 = 0;   int a2 = 0;    a1 = getAA1();   a2 = getAA2(); // a是10   int &a3 = getAA2(); // 若返回棧變量,不能成為其他引用的初始值   cout << a1 << endl;   cout << a2 << endl;   cout << a3 << endl; // a3是亂碼,這里出現了問題   // 編譯器看到a3是個引用,自動進行對a3的地址進行取值   // 但是函數getAA2退出的時候已經釋放了這個地址的內存,所以這里是亂碼    return 0; } 

返回值是static變量,當引用

//static修飾變量的時候,變量是一個狀態變量 int j() {   static int a = 10;   a++;   printf("a:%d /n", a);   return a;  }  int& j1() {   static int a = 10;   a++;   printf("a:%d /n", a);   return a; }  int *j2() {   static int a = 10;   a++;   printf("a:%d /n", a);   return &a; }   void main() {   // j()的運算結果是一個數值,沒有內存地址,不能當左值   //11 = 100;   //*(a>b?&a:&b) = 111;   //當被調用的函數當左值的時候,必須返回一個引用   j1() = 100; //編譯器幫我們打造了環境   j1();   *(j2()) = 200; //相當于手工的打造,做左值的條件   j2();   system("pause"); } 

返回值是形參,當引用

int g1(int *p) {   *p = 100;   return *p; }  int& g2(int *p) // {   *p = 100;   return *p; } //當使用引用語法的時候 ,不去關心編譯器引用是怎么做的 //當分析亂碼這種現象的時候,才去考慮c++編譯器是怎么做的。。。。 void main() {   int a1 = 10;   a1 = g2(&a1);    int &a2 = g2(&a1); //用引用去接受函數的返回值,是不是亂碼,關鍵是看返回的內存空間是不是被編譯器回收了。。。。   printf("a1:%d /n", a1);   printf("a2:%d /n", a2);    system("pause"); } 
 


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 阳曲县| 团风县| 奇台县| 平塘县| 靖宇县| 谢通门县| 宁晋县| 横山县| 北京市| 正安县| 田阳县| 县级市| 夏河县| 天柱县| 临颍县| 滁州市| 磐石市| 广安市| 德惠市| 桂阳县| 冀州市| 南平市| 南岸区| 广汉市| 惠安县| 岗巴县| 外汇| 波密县| 启东市| 广安市| 西昌市| 十堰市| 额尔古纳市| 略阳县| 大悟县| 和平区| 新郑市| 莱芜市| 甘南县| 五常市| 厦门市|