6.2 参数传递 | Parameter passing

参数传递

形参初始化的机理与变量初始化相同。形参的类型决定了形参和实参的交互方式。

  • 引用传递 passed by reference,函数被 传引用调用 called by reference 。形参是引用,绑定到对应实参上。引用形参也是其绑定对象的别名,即引用形参是对应实参的别名。
  • 值传递 passed by value ,函数被传值调用 called by value ,将实参的值拷贝赋给形参。

传值参数

初始化非引用类型的变量,传值会拷贝初值,因此对变量的改动不会影响初始值。(对比于python)

指针形参

当执行指针拷贝操作,拷贝的是指针值,即地址值。拷贝后两个指针是不同的指针。通过指针可以修改它所指对象的值,

指针形参的行为类似指针行为,可以改变实参的值,例如下:

void reset(int* ip){
    *ip = 0;  // ip所指对象置 0
    ip = nullptr;  // 改变ip本身的局部拷贝,不改变实参
}

int main(){
    int i = 42;
    reset(&i);         //由于是指针形参,需要取地址传参。
    cout<<i<<endl;     //i=0
}

建议使用引用类型的形参(C++风格)替代指针类型。

传引用参数

对引用的操作实际上是作用在引用所引的对象上。引用形参类似。通过引用形参可以改变一个或多个实参的值。

void reset(int& i){
    i = 0;
}

int main(){
    int i = 42;
    reset(i);         
    cout<<i<<endl;     //i=0
}

使用引用避免拷贝

对较大类型对象和容器对象以及不支持拷贝的对象,为了性能或者可行性,只能通过引用形参访问该类型的对象。

如比较两个string对象的长度。由于string对象可能非常长不便于拷贝,而又不需要在函数中修改时,可以利用对常量的引用定义形参:

bool isShorter(const string& s1, const string& s2){
    return s1.size()<s2.size();
}

使用引用形参返回额外信息

C++函数只能有一个返回值。若要返回多个且有可能不同类型的值,可以构造一个含有这些值类型的新类,或者直接传入一个引用类型的形参并隐式返回该值。

例如该程序同时统计在字符串中某个字符的首次出现位置索引和总次数,可以编写如下:

string::size_type find_char(const string& s, char c, 
                            string::size_type& occurs){
    auto ret = s.size();
    occurs = 0;
    for(decltype(ret)i=0; i!=s.size(); ++i){
        if (s[i]==c){
            if (ret == s.size()) ret = i;
        ++occurs;
        }
    }
    return ret;
}

int main(){
    string::size_type occ;
    string s = "bcdefghahaha";
    string::size_type r;
    r = find_char(s, 'a', occ);
    cout<<"First sub: "<<r<<";\nTotal occurs: "<<occ;
    return 0;
}

输出:

First sub: 7;
Total occurs: 3

const 形参和实参

仅在用实参初始化形参时形参具有的顶层const会被忽略。因此可以传给其常量和非常量。而只能给底层const的形参传递常量。

因此,虽然C++允许定义同名而拥有不同形参列表的函数,但下述两个定义会被认为是同一个,尽管函数内的形参功能不同:

void a(const int i){}
void a(int a){}

指针或引用形参参与const

具体的底层、顶层及可更改对象和初始化规则一致。

尽量使用常量引用

将函数内不会改变其值的形参定义为常量引用,既可以同时接受const和非const类型的实参,又可以避免不需要的问题。

  • 问题1:非底层const引用无法使用字面值初始化。
    void a(string a, int b){}
    void b(const string a, const int b){}
    a("str", 1) //error
    b("str", 1) //correct
    
  • 问题2:不同函数嵌套时,若内层有函数是普通引用而无法接受const类的实参,会出错。

数组形参

数组的两个特殊性质对定义和使用作用在数组的函数有影响:

  • 不允许拷贝数组(如不可以a=b等):无法以值传递的方式使用数组参数
  • 使用数组时可能会被转化为指针:为函数传递数组时实际上是传递了指向该数组首元素的指针。

因此,下列三个表达式等价,形参类型都是const int*

void a(const int*);
void a(const int[]);     //本意是作用与数组,实际形参是一个指针。
void a(const int[10]);   //10仅仅是期望的元素,实际不一定。

存在的问题:由于无法拷贝且会转为指针,当确实需要数组作为实参时,开始和结束位置难以确定。

使用标记指定数组长度

管理数组实参的第一种方法:要求数组本身包含一个结束标记。典型示例是c风格字符串(以'\0'结尾)的处理:

void print(const char* cp){
    if (cp)          //指向内容不为空
        while (*cp)  //所指字符不是空字符
            cout<<*cp++; 
}

使用标准库规范

管理数组实参的第二种方法:传递指向数组首元素和尾后元素的指针,注意需要在使用函数前使用std::begstd::end提前取得这两个指针:

void print(const char* beg, const char* end){
    while (beg!=end)
        cout<<*beg++;
}

显式传递表示数组大小的形参

管理数组实参的第三种方法:定义一个表示数组大小的形参。在旧版本的 C++ 和 C 中常用这种方法。注意该形参的类型是size_t

数组形参和const

和引用形参一样,尽量将指针形参其定义为const类型避免不必要的问题。

数组引用形参

若将变量定义为数组的引用,可以直接使用。但是注意引用构成的数组(X)-int& arr[10]对数组的引用-int (&arr)[10]

传递多维数组

一般第一个维度使用指针传递,而操作必不可少的第二个维度,只能传入其含有的元素个数:

void print(int (*matrix)[10], int rowSize){}
// 等价于
void print(int matrix[][10], int rowSize){}

int matrix[][10] 看似是个二维数组的声明,实际上声明了一个指向含有10个整型元素的数组的指针。与 int (*matrix)[10] 一致。

main:处理命令行选项

往main()函数中定义参数以设置运行选项,如将这些命令传递给prog文件中的程序:prog -d -o ofile data0

int main(int argc, char* argv[]){}

第二个形参argv是一个数组,数组内每项都是char指针;第一个形参表示字符串数量。第二个形参是数组,所以main函数也可以定义为:

int main(int argc, char** argv){}

其中 argv 指向 char*。其内容(数组)或指向的内容(指针):

argv[0] = "prog"; //程序名
argv[1] = "-d";   //开始参数
argv[2] = "-o";
argv[3] = "ofile";
argv[4] = "data0";
argv[5] = 0;

含有可变形参的函数

常常会无法提前预知向函数传递的实参数量。为了使函数处理未知数量实参,C++11提供两种方法:

  • 所有实参类型相同:传递名为 initializer_list 的标准库类型;
  • 实参类型不同:可变参数模板

initializer_list 形参

expression description
initializer_list<T> lst; 默认初始化空列表
initializer_list<T> lst{a, b, c...}; 列表初始化,内部元素都是const
lst2(lst) 赋值,前后二者共享元素
lst2 = lst 拷贝,前后二者共享元素
lst.size() 列表中元素数量
lst.begin() 返回指向lst首元素的指针
lst.end() 返回指向lst尾后元素的指针
  • initializer_list对象中的元素永远是常量值,无法改变。

  • 使用begin和end成员可以完成类似vector的迭代器操作。

  • 向该类型形参中传递一个值的序列,必须把序列放在一对花括号内:

    initializer_list<string> str = a?  {"A", "B"}:{"C", "D", "E"};
    
  • 含有该类型形参的函数也可以含有其他类型的形参

省略符形参

省略符形参是为了便于C++程序访问某些特殊的C代码设置的,这些代码使用了名为varargs的C标准库功能。通常省略符形参不应用于其他目的

只能出现在形参列表最后一个位置,但注意大多数类类型的对象无法正确拷贝给该形参。

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

推荐阅读更多精彩内容