当前位置 : 主页 > 编程语言 > c语言 >

C++(2)

来源:互联网 收集:自由互联 发布时间:2023-09-06
引用的使用场景: 做参数 做返回值 传值、传引用效率比较 以至作为参数或返回值,在传参和返回期间, 函数不会直接传递实参或者将变量本身直 接返回,而是传递实参或者返回变量

引用的使用场景:

做参数

做返回值

传值、传引用效率比较

以至作为参数或返回值,在传参和返回期间,函数不会直接传递实参或者将变量本身直

接返回,而是传递实参或者返回变量的一份临时的拷贝,因此用值作为参数或者返回值类型,效

率是非常低下的,尤其是当参数或者返回值类型非常大时,效率就更低。

而传引用的效率要高于传值

引用与指针的区别

语法概念上:引用是一个别名,没有独立的空间,与其引用实体共用同一块空间

(引用在底层实现实际是有空间的,引用底层是按照指针实现的)

指针存储一个变量的地址

2.引用必须初始化,指针没有要求

3.引用在初始化时引用一个实体后,就不能再引用其他实体,而指针可以改变所指向的实体

4.没有NULL引用,但有NULL指针

5.在sizeof中的含义不同:引用时引用的实体的大小,但指针始终是地址空间所占字节个数(32位是4个字节)

6.引用自加即其实体自加,指针则向后偏移一个类型的大小

7.有多极指针,但无多级引用

8.引用相对更安全

内联函数


以inline修饰的函数,编译时C++编译器会在调用内联函数的地方展开,不建立栈帧,提高程序效率

(注意:内联说明只是向编译器发出的一个请求,编译器可以选择忽略)

特性:

1.inline是一种以空间换时间的做法,这样可能会使目标文件变大,但减少了调用开销

2.内联机制用于优化规模较小、流程直接、调用频繁的函数,内联函数体中一般不能含有较为复杂的控制语句,如for、switch等

3.inline不建议声明与定义分离,分离会导致链接错误,因为inline被展开,没有函数地址

4.C++的内联函数与C语言中的带参宏定义#define有些相似,都是直接替换,但是宏定义在编译前对其进行预处理的,不做语法检查

复习:宏的优缺点

优点:增强代码的复用性,提高性能

缺点:1.不方便调试(与编译阶段惊醒了替换)2.导致代码的可读性差,容易误用3.没有类型的检查

C++替代宏:常量:const、enum

短小函数定义:inline函数

auto关键字:

auto作为一个新的类型指示符来指示编译器,auto声明的变量必须由编译器在编译时期推导而得

使用auto定义变量时必须对其进行初始化,在编译阶段编译器需根据初始化表达式来推导其类型,因此,auto不是一种类型的声明,而是一个类型声明时的占位符,编译器在编 译期会将auto替换为变量实际的类型。

1.auto不能作为函数的参数,不能直接用来声明数组

2.C++11只保留了auto作为类型指示符的用法

3. auto在实际中最常见的优势用法就是跟以后会讲到的C++11提供的新式for循环,还有 lambda表达式等进行配合使用。

基于范围的for循环

对于有范围的集合,C++11引入了基于范围的for循环,for循环后的括号由冒号“:”分为两部分,第一部分是范围内用于迭代的变量,第二部分则表示被迭代的范围

注意:与普通循环类似,可以用continue来结束本次循环,也可以用break来跳出整个循环

C++(2)_内联函数

int main()
{
	int arr[10] = { 1,2,3,4,5,6,7,8,9,0 };
	for (int i : arr)//如果使用引用,可更改数组元素
	{
		cout << i << endl;
	}
	return 0;
}
范围for的使用条件

1. for循环迭代的范围必须是确定的

对于数组而言,就是数组中第一个元素和最后一个元素的范围;对于类而言,应该提供 begin和end的方法,begin和end就是for循环迭代的范围。

注意:以下代码就有问题,因为for的范围不确定

void test(int array[])
{
  for(auto & e:array)
    cout<<e<<endl;
}

指针空值nullptr(C++)

NULL实际上是一个宏,在头文件(stddef.h)中

#ifndef NULL
#ifdef __cplusplus
#define NULL   0
#else
#define NULL   ((void *)0)
#endif
#endif

NULL被定义为字面常量0,或者被定义为无类型指针(void *),在使用空值指针时会产生一些麻烦

void f(int)
{
  cout<<"f(int)"<<endl;
}
void f(int *)
{
  cout<<"f(int *)"<<endl;
}
int main()
{
  f(0);
  f(NULL);
  f((int *)NULL);
  return 0;
}

程序本意是想通过f(NULL)调用指针版本的f(int*)函数,但是由于NULL被定义成0,因此与程序的 初衷相悖。

在C++98中,字面常量0既可以是一个整形数字,也可以是无类型的指针(void*)常量,但是编译器 默认情况下将其看成是一个整形常量,如果要将其按照指针方式来使用,必须对其进行强转(void *)0。

注意:1.使用nullptr不需要包含头文件,C++11中nullptr是关键字

2.在C++11中,sizeof(nullptr)与sizeof((void*)0)所占的字节数相同

3.在表示指针空值时最好使用nullptr

上一篇:原码、反码和补码的求法
下一篇:没有了
网友评论