C++:引用语法

本文是笔者在C++学习过程中,对引用语法的总结记录,如有不队的地方,欢迎大家指正!


一、 什么是引用

  引用(reference),是C++对C语言的一项扩充,作用是给变量起别名,换句话说,就是将多个变量名指向同一个地址,从而使对其中的任意一个变量名的操作,都是对同一地址的操作。而在这种情况下,被声明为引用类型的变量名,就是实际变量名的一个别名。


二、引用的基本语法与注意事项

1、 声明语法

  引用的声明语法:数据类型 &引用变量名 = 原变量名;

eg:

#include <iostream>
using namespace std;

int main() {
    //语法:数据类型 &引用变量名 = 原变量名;
    int a=10;
    int &reference_a=a;

    cout<<"a="<<a<<endl;                     //a=10
    cout<<"reference_a="<<reference_a<<endl; //reference_a=10

    a+=10;
    cout<<"a="<<a<<endl;                     //a=20
    cout<<"reference_a="<<reference_a<<endl; //reference_a=20

    reference_a+=10;
    cout<<"a="<<a<<endl;                     //a=30
    cout<<"reference_a="<<reference_a<<endl; //reference_a=20
       
    return 0;
}

2、 注意事项

  1. 对引用进行操作,实际上就是对被引用的变量进行操作。

  2. 声明引用时,必须对其进行初始化。

  3. 引用在初始化之后,不可以再进行修改。

  4. 引用变量在编译器的底层实现是指针常量,即一个指针指向不可以发生更改的指针,所以引用占用内存空间的大小,和指针其实是一样的。

    但是,和指针不同的是,引用实际上不能算是一个变量,所以对引用变量求地址,与对实际变量求地址,结果是相同的。

#include <iostream>
using namespace std;

int main() {
    int a=10;
    int &reference_a=a;

    //对引用变量求地址,与对实际变量求地址,结果是相同的
    cout<<"对引用变量求地址"<<endl;
    cout<<"a的地址:"<<&a<<endl;
    cout<<"reference_a的地址:"<<&reference_a<<endl;
    
    return 0;
}

运行结果:


对引用变量求地址 运行结果
  1. 不能建立引用的数组,即数组中的元素不能是引用。因为引用使劲上并不能算是一个变量,所以在声明引用的数组时,并不能分配内存,也就无法声明和定义引用数组。

    但是,可以建立数组的引用,这是没有问题的。

#include <iostream>
using namespace std;

int main() {
    //不能建立引用的数组,但是,可以建立数组的引用
    int arr[]={0,1,2,3,4};
    // int &re_arr=&arr; 
    // 编译错误
    // error: invalid conversion from ‘int (*)[5]’ to ‘int’

    //数组的引用
    int (&re_arr)[5]=arr;

    return 0;
}

三、引用作为函数参数

  引用在C++中的一个重要作用,就是可以作为函数的参数使用。

  在C语言之中,函数参数传递使用的方法是值传递,而在有大量数据作为参数传递时,往往采用的是地址传递,即使用指针,来避免在函数调用时有大量数据压入栈中。现在,C++中可以使用引用作为函数参数,以此来代替指针作为函数参数,这种方法可以使代码更易阅读和维护。

eg:

#include <iostream>
using namespace std;

//引用作为函数参数
void fun1(int &a,int &b);

int main() {
    int a=10;
    int b=20;
    cout<<"a="<<a<<",b="<<b<<endl;
    fun1(a,b);
    cout<<"a="<<a<<",b="<<b<<endl;
    return 0;
}

//引用作为函数参数
void fun1(int &a,int &b){
    int temp=a;
    a=b;
    b=temp;
}

运行结果:


引用作为函数参数 运行结果

  可以看出,在使用引用作为函数参数时,与使用指针作为函数参数的效果一致,也就是在调用方法的时候,可以用形参修饰实参。


四、引用作为函数返回值

  C++支持将引用作为函数的返回值返回。

eg:

#include <iostream>
using namespace std;

//引用作为函数返回值
int& fun2(int &a);

int main() {
    int b=10;
    cout<<"fun2()前,b="<<b<<endl;
    cout<<"fun2()后,b="<<fun2(b)<<endl;
    return 0;
}

int& fun2(int &a){
    a*=a;
    return a;
}

运行结果:


引用作为函数返回值 运行结果

注意事项

1. 不要返回局部变量的引用

  因为局部变量是存放在栈区的,在函数返回后,局部变量就会被销毁,所以返回的引用就会像悬空指针一样,指向一个未知的空间。

eg:

#include <iostream>
using namespace std;

int& func2_local();

int main() {
    int &ref_locol=func2_local();
    
//    cout<<"ref_locol="<<ref_locol<<endl;          //编译错误
    cout<<"ref_locol的地址为:"<<&ref_locol<<endl;   //输出地址为0
    
    return 0;
}

// 不要返回局部变量的引用
int& func2_local(){
    int a=10;   //栈区,函数调用结束后释放
    return a;   //Reference to stack memory associated 
                // with local variable 'a' returned
}

运行结果:


不要返回局部变量的引用 运行结果

2. 函数调用可以作为左值

eg:

#include <iostream>
using namespace std;

int& func2_left();

int main() {
    int left=func2_left();
    cout<<"left="<<left<<endl;
    cout<<"func2_left()作为左值,修改为100"<<endl;
    func2_left()=100;
    cout<<"left="<<left<<endl;
    return 0;
}

//函数调用可以作为左值
int& func2_left(){
    static int a=10; //全局区,程序结束后由系统释放
    return a;
}

运行结果:


函数调用可以作为左值 运行结果

3. 不要返回函数内部使用 new 分配的内存的引用

  这种编程习惯是非常不好的,因为每个用 new 产生的指针都要调用 delete 释放,否则就会造成内存泄漏。而在被函数返回的引用作为一个临时变量时,我们很容易未对其进行释放,从而造成内存泄漏。

eg:

#include <iostream>
#include <string>
using namespace std;

string& func2_new();

int main() {
    string &str=func2_new();    //如果调用后没有用 delete 释放,就会造成内存泄漏
    cout<<"str="<<str<<endl;
    delete &str;                //释放
    return 0;
}
//不要返回函数内部使用 new 分配的内存的引用
string& func2_new(){
    string* str=new string("hello world");
    return *str;
}

  由代码可以看出,返回函数内部使用 new 分配的内存的引用时,极易引起内存泄漏,尤其是在形如以下代码的情况下:

string str="hello world"+func2_new();

五、常量引用/常引用

  声明方式:const 数据类型 &引用名 = 目标变量名;

eg:

#include <iostream>
using namespace std;

string& func3();

int main() {
    func3(a);
    return 0;
}

//常量引用/常引用
void func3(const int &a){
    cout<<"a="<<a<<endl;
}

  常引用主要用于修饰形参,防止函数中误操作。另外,在引用作为函数参数时应该尽量定义为 const 。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,053评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,527评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,779评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,685评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,699评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,609评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,989评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,654评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,890评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,634评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,716评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,394评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,976评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,950评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,191评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,849评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,458评论 2 342

推荐阅读更多精彩内容