【C++入门必备知识:缺省参数+函数重载+函数名修饰规则】

简介: 当调用该函数时,如何没有没有指定实参则采用改形参的默认值。

fd4f160bd48840b7a18e0d4615e59e2c.png


①.缺省参数


Ⅰ.概念


当声明或定义函数时,为函数的参数指定一个缺省值,也叫做默认值。


规则:

当调用该函数时,如何没有没有指定实参则采用改形参的默认值。

当调用该函数时,指定实参,那就使用传过来的实参。


通俗的说就是:

没有传参时,使用参数的默认值

传参时,使用指定的实参


void fun(int n=10)
{
  cout << n << endl;
}
int main()
{
  fun();//没有指定实参,则使用缺省参数
  fun(20);//指定实参
}

51334105c45249d6917122c2ef274edb.png


1.全缺省参数


全缺省参数,即函数形参都被指定为缺省值


using namespace std;
void fun(int a=1,int b=2,int c=3)
{
  cout << a<< endl;
  cout << b << endl;
  cout << c << endl;
}
int main()
{
  fun();//全缺省参数,形参全部指定为缺省值,不传实参
}

7105d498e02942a8aa4a90c45fb714d6.png


2.半缺省参数


半缺省参数,即函数形参部分被指定为缺省值

void fun(int a,int b=2,int c=3)
{
  cout << a<< endl;
  cout << b << endl;
  cout << c << endl;
}
void fun2(int a,int b, int c = 9)
{
  cout << a << endl;
  cout << b << endl;
  cout << c << endl;
}
int main()
{
  fun(2);//部分缺省,部分形参指定为缺省值
  fun2(5, 6);
}


bb742f000a2542ec9898f28fb0dfd3e9.png


3.使用规则


  • 1.传参是从左向右传,不能隔着传。
  • 2.而缺省参数是从右到左缺省。
  • 3.声明和定义不能同时给缺省

为什么声明和定义不能同时给缺省呢???

因为会出现这样的场景:当声明和定义的缺省参数不一致时,那编译器到底该用哪个缺省值呢?

  • 4.准确的来说,只能在声明函数的时候来给缺省,定义时候不能给。
  • 5.缺省值必须是常量或者全局变量
  • 6.默认参数也叫缺省参数


4.应用场景再现


比如顺序表中有静态顺序表和动态顺序表。我们知道静态顺序表,写死了固定大小,很不好使,但动态顺序表又需要不断的扩容,扩容操作是需要消耗效率的。所以我们可以利用缺省参数,来对顺序表的默认大小进行缺省,当我们知道要插入多少数据时,则指定传相对的大小,那顺序表一开始就会开辟那么大的空间,就不需要从一小块不断扩容了。而当我们不知道要插入的数据时,那么就按照给定的默认值(缺省值)来进行开辟空间。


typedef struct SQList
{
  int* a;
  int size;
  int capacity;
}SQList;
void SQInit(SQList *s,int defalutCapacity=4)
{
  s->a = (int*)malloc(sizeof(SQList) * defalutCapacity);
  if (s->a == NULL)
  {
    perror("malloc");
  }
}
int main()
{
  SQList s;
  SQInit(&s);//如果不知道要插入多少数据,那就按照默认值4来
  SQInit(&s, 100);//如果知道要插入多少数据,那么直接就开辟这么大的空间,不需要不断的扩容到100
}


②.函数重载


C语言不允许存在同名函数,但C++允许存在同名函数,这么做的原因是用来处理实现功能类似数据类型不同的问题。

那C++是如何做到让同名函数同时存在的呢?


Ⅰ.概念


函数重载:是函数的一种特殊情况,C++允许在同一作用域中声明几个功能类似的同名函数,这些同名函数的形参( |参数个数|或类型|或类型顺序|)不同。


1.参数个数不同


//参数个数不同
int Add2(int left, int mid,int right)//3个参数
{
  return left+mid + right;
}
int Add(int left, int right)//2个参数
{
  return left + right;
}
int main()
{
  printf("%d\n", Add(1, 1));//编译器会自动识别使用哪一个函数的
  printf("%d\n", Add2(1,2,3));
}

fbcdbfdaf6094d0fafd21e407cd2f68e.png


参数个数不同:


void f()
{
  cout << "f()" << endl;
}
void f(int a)
{
  cout << "f(int a)" << endl;
}


不过要注意的是下面这个例子


void f()
{
  cout << "f()" << endl;
}
void f(int a=0)
{
  cout << "f(int a)" << endl;
}


你觉得它构成函数重载吗?


从定义的角度来说,它确实是一个没有参数,一个有一个参数,它们参数个数不同,函数名相同,所以是构成函数重载的。但有没有什么问题呢?


你看,第二个函数的参数被缺省了,也就是相当于当函数传参时,传不传都无所谓,传参数,那就使用这个实参,如果不传,那就使用缺省参数。那么问题来了,如果我们不传参数,到底调用的是第一个函数,还是调用第二个函数呢?


第一个函数没有参数,如果我们不传参正常来说就应该调用第一个,但第二个同名函数的参数被缺省了。所以就会出现

调用不明确问题。


a27e59e17c7a460a9e61f4a017cdfd3a.png


2.参数类型不同


//参数类型不同
int Add(int left, int right)//int 类型
{
  return left + right;
}
double Add(double left, double right)//double 类型--注意这里返回值是double ,不需要管返回值类型
{
  return left + right;
}
int main()
{
  printf("%d\n", Add(1, 1));
  printf("%f\n", Add(1.1, 1.1));//编译器会自动识别数据类型,并且使用相应的函数
}

21096031a36a4de09602f0242c225c15.png


3.参数类型顺序不同


void fun1(int a, char c)
{
  cout << a << " " << c << endl;
}
void fun1(char c, int a)
{
  cout << c << " " << a << endl;
}
int main()
{
  fun1(1,'x');
  fun1('w', 6);
}

7a37541fae8042d8875da208d3c9e363.png


要明确注意的是这里是类型顺序,而不是变量顺序。


比如下面这个坑


void fun1(int a, char c)
{
  cout << a << " " << c << endl;
}
void fun1(int c, int a)
{
  cout << c << " " << a << endl;
}


你觉得构成重载吗?


当然不构成了,我们要严格按照定义,是类型顺序不同如果是上面这个例子,那我要传fun1(1,'x');到底调用的是哪一个函数呢,是上面的还是下面的呢?所以是有歧义的。


c++虽然允许同名函数并且会自动识别变量类型

但要严格遵守函数重载的规则才可以。


4.对返回值没有要求


函数重载对函数的返回值是没有要求的。


返回值没有要求,但是当后面的要求不符合是,仍然不能构成重载(函数参数类型不同,函数参数的个数不同,函数类型顺序不同)


返回值没有要求 --不构成重载 —无法使用


int Add(int left, int right)
{
 cout << "int Add(int left, int right)" << endl;
 return left + right;
}
double Add(double left, double right)
{
 cout << "double Add(double left, double right)" << endl;
 return left + right;
}


比如这样,两个同名函数符合参数类型不同,所以构成重载。虽然它们返回值不同,但函数重载对返回值没有要求。


③.函数名修饰规则


为什么C语言不支持函数重载,而C++支持重载呢?

C++又是如何支持的呢?

这其中就要涉及程序的【编译与链接过程】与【函数名修饰规则】


59b260ace12a451cba87a78e8b040b63.png


假设一个工程里有三个文件,一个是专门声明各函数的文件,一个是专门定义各函数的文件,一个是用来测试这些函数功能的文件。


d1ff3c0510aa41bea5113e9ca314b479.png


我们知道一个程序要通过预编译,编译,汇编,链接四个部分才可以生成可执行程序。


而各个阶段的处理也不相同。


大体是就是在预编译阶段,在声明和定义的两个文件里包含的头文件会被展开,宏会被替换,还有注释会被去掉。


然后各生成一个带 .i的文件。在编译阶段会进行检查语法,生成汇编代码,并生成一个带.s的文件。


汇编阶段会进行将汇编代码转化为机器指令,生成一个符号表,并生成一个带.o的文件,最后在链接阶段,会将符号表合并和重定位,将两个带.o文件链接成一个可执行程序。


6ec5ee38152044ffb4a7eb6c1e3064e4.png


根据函数栈帧的创建和销毁,我们知道当调用一个函数时,会执行一个call的汇编代码。call后面就是函数的地址。然后就会使用jump跳进调用的函数里。


而在实际的项目中通常是由多个头文件和多个源文件。


比如在test.c文件里调用了定义.c文件里的函数A,在链接之前,test.o文件里是没有函数A的地址的,因为函数A的地址在定义.c文件里。


那么test.c程序如何执行呢?

在链接阶段会解决这个问题,链接器看到test.o调用函数A,但是又没有函数A的地址,它就会到定义.o符号表中去找函数A的地址,然后重定位到一起。


我们可以这样比喻:将在.h头文件里声明的看成一种承诺


test.c想要买房,还差钱,向好兄弟.h文件借钱,好兄弟.h满口答应。给定了test.c承诺。


test.c有了承诺,它就敢买房了,这个过程是合法的。所以不会报错的。


但要真正的买到房,还需要.h兑现承诺,而如果.h找到定义的文件就可以兑现承诺了。


所以test.c要执行起来就需要找到定义来兑现承诺。


链接的作用就是:找到定义(兑现承诺)。


那还有一个问题,链接器是如何找定义的呢?是找哪个定义的呢?


是使用哪个名字定义的呢?


在不同的编译器有不同的函数名修饰规则。


Ⅰ.C/C++的不同


我们举例gcc和g++的例子。


  • 1.C语言编译器编译结果:


我们发现gcc的函数修饰后名字是不改变的。


f82789bcdf82430db607d50d3df4dae3.png


  • 2.C++编译器编译器结果:


而在g++的函数修饰后变成了【_Z+函数长度+函数名+类型首字母】


4f318da2c5f7482db788b89a92d7996d.png


  • 3.结论:采用C语言编译器编译后,函数名没有改变,而采用C++编译器编译后,函数名经过函数修饰发生改变,并且跟函数的参数类型,参数个数等有关。


  • 4.通过这里我们就可以理解C语言没有办法支持重载了,因为C编译器,在编译后,同名函数没有办法区分,而C++经过函数修饰规则,根据函数参数的类型,参数的个数不同而改变函数的名字进行区分,只要参数不同,修饰后的函数名就不一样,这样就支持重载了。


  • 5.如果两个函数函数名和参数都是一样的,但是函数返回值不一样,这是不构成重载的,因为重载的定义没不包括返回值。编译器没有办法区分。
相关文章
|
2月前
|
程序员 C++
C++中的函数重载有什么作用
【10月更文挑战第19天】C++中的函数重载有什么作用
25 3
|
2月前
|
编译器 程序员 C++
C++中的函数重载是什么
【10月更文挑战第19天】C++中的函数重载是什么
35 0
|
2月前
|
存储 安全 编译器
【C++打怪之路Lv1】-- 入门二级
【C++打怪之路Lv1】-- 入门二级
26 0
|
2月前
|
自然语言处理 编译器 C语言
【C++打怪之路Lv1】-- C++开篇(入门)
【C++打怪之路Lv1】-- C++开篇(入门)
37 0
|
2月前
|
分布式计算 Java 编译器
【C++入门(下)】—— 我与C++的不解之缘(二)
【C++入门(下)】—— 我与C++的不解之缘(二)
|
2月前
|
编译器 Linux C语言
【C++入门(上)】—— 我与C++的不解之缘(一)
【C++入门(上)】—— 我与C++的不解之缘(一)
|
23天前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
38 2
|
29天前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
83 5
|
1月前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
80 4
|
1月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
86 4