C++初阶——入门

目录

1、C++发展历史

2、C++版本更新

3、C++参考文档

4、C++书籍推荐

5、C++的程序

6、命名空间

6.1 namespace的作用

6.2 namespace的定义

6.3 namespace的使用

7、C++输入&输出

8、缺省参数

9、函数重载

10、引用

10.1 引用的概念和定义

10.2 引用的特性

10.3 引用的使用

10.4 const引用

10.5 指针和引用的关系(面试高频考查点)

11、inline

12、nullptr


1、C++发展历史

C++的起源可以追溯到1979年,当时Bjarne Stroustrup(本贾尼·斯特劳斯特卢普,这个翻译的名字不同的地方可能有差异)在贝尔实验室从事计算机科学和软件工程的研究工作。面对项目中复杂的软件开发任务,特别是模拟和操作系统的开发工作,他感受到了现有语言(如C语言)在表达能力、可维护性和可扩展性方面的不足。

1983年,Bjarne Stroustrup在C语言的基础上添加了面向对象编程的特性,设计出了C++语言的雏形, 此时的C++已经有了类、封装、继承等核心概念,为后来的面向对象编程奠定了基础。这一年该语言被正式命名为C++。

在随后的几年中,C++在学术界和工业界的应用逐渐增多。一些大学和研究所开始将C++作为教学和研究的选语言,而一些公司也开始在产品开发中尝试使用C++。这一时期,C++的标准库和模板等特性也得到了进一步的完善和发展。

C++的标准化(使得 C++ 代码在不同的编译器下表现一致)工作于1989年开始,并成立了一个ANSI和ISO(International Standards Organization)国际标准化组织的联合标准化委员会。1994年标准化委员会提出了第一个标准化草案。在该草案中,委员会在保持斯特劳斯特卢普最初定义的所有特征的同时,还增加了部分新特征。

在完成C++标准化的第一个草案后不久,STL(Standard Template Library)是惠普实验室开发的系列软件的统称。它是由Alexander Stepanov、Meng Lee和David R Musser在惠普实验室工作时所开发出来的。在通过了标准化第一个草案之后,联合标准化委员会投票并通过了将STL包含到C++标准中的提议。STL对C++的扩展超出C++的最初定义范围。虽然在标准中增加STL是个很重要的决定,但也因此延缓了C++标准化的进程。

1997年11月14日,联合标准化委员会通过了该标准的最终草案。1998年,C++的ANSI/IS0标准被投入使用。

2、C++版本更新

C++98:这是第一个 ANSI/ISO 标准化的 C++ 版本,发布于 1998 年。它基于 Bjarne Stroustrup 于 1985 年创建的原始 C++ 设计,并加入了一些重要特性,如 STL(标准模板库)、异常处理、I/O Streams、命名空间和 RTTI(运行时类型识别)。

C++03:这个版本主要是对 C++98 的一些修正和改进,发布于 2003 年,并未引入新的语言特性,所以一般不把它当做重要版本,存在感也不强。

C++11:这是 C++ 历史上最重大的更新之一,有时被称为 C++0x(因为它原计划在 200x 年发布,一直跳票)。它引入了大量新特性,如自动类型推断(auto 关键字)、基于范围的 for 循环、Lambda 表达式、智能指针、并发支持、移动语义、nullptr 和更强大的模板功能等。

C++14:作为 C++11 的小幅度更新,C++14 引入了一些改进和新特性,包括泛型 Lambda 表达式、返回类型推导、二进制字面量、数字分隔符、弃用属性等。

C++17:这个版本进一步提升了 C++ 的功能和易用性,新功能不是很多,引入了结构化绑定、if constexpr、std::optional、std::variant、std::string_view、并行算法等特性。

C++20 是继 C++11 之后又一个重大更新,引入了概念(concepts)、范围库(ranges)、协程(coroutines)、模块(modules)、三元运算符的改进、constexpr 的增强、std::span 等新特性。

C++23 是 2023 年 7 月份刚确定下的新标准,目前能完整支持 C++23 的编译器基本没有。变化包括引入标准库的模块化支持、扩展 constexpr 、增加并行算法、ranges 扩展、this 推导、引入更多的属性和注解、增加 std::mdspan、std::generator 等新特性。

3、C++参考文档

不是C++官官方文档,标准也只更新到C++11,但是以头文件形式呈现,内容比较易看好懂

Reference - C++ Reference

是C++官方文档的英文版,更新到了最新的C++标准,但相比第一个不那么易看

https://en.cppreference.com/w/

是C++官方文档的中文版

https://zh.cppreference.com/w/cpp

4、C++书籍推荐

C++ Primer:主要讲解语法,经典的语法书籍,可以作为语法字典,非常好用。

STL源码剖析:主要从底层实现的角度结合STL源码,庖丁解牛式剖析STL的实现,是侯捷老师的经典之作。可以很好的帮助我们学习别人用语法是如何实现出高效简洁的数据结构和算法代码,如何使用泛型封装等。让我们不再坐井观天,闭门造车,适合中后期可以看。

Effctive C++:本书也是侯捷老师翻译的,本书有的一句评价,把C++程序员分为看过此书的和没看过此书的。本书主要讲了55个如何正高效使用C++的条款,建议中后期可以看一遍,工作1-2年后再看遍,相信会有不一样的收获。

5、C++的程序

C++兼容C语言绝大多数的语法,所以C语言实现的hello world依旧可以运行,C++中需要把定义文件代码后缀名改为.cpp,vs编译器看到是.cpp就会调用C++编译器编译,linux下要用g++编译,不再是gcc

// 兼容C语言
// test.cpp
#include<stdio.h>
int main()
{printf("hello world\n");return 0;
}

当然C++有一套自己的输入输出,严格说C++版本的hello world应该是这样写的

// test.cpp
// 这里的std cout看不懂,没关系,下面会依次讲解
#include<iostream>
using namespace std;int main()
{cout << "hello world\n" << endl;return 0;
}

6、命名空间

6.1 namespace的作用

在C/C++中,变量、函数和后面要学到的类都是大量存在的,这些变量、函数和类的名称将都存在于全局作用域中,可能会导致很多冲突。

使用命名空间的目的是对标识符的名称进行本地化,以避免命名冲突或名字污染,namespace关键字的出现就是针对这种问题的。

c语言项目类似下面程序这样的命名冲突是普遍存在的问题,如:

#include <stdio.h>
#include <stdlib.h>
int rand = 10;
int main()
{// 编译报错:error C2365: “rand”: 重定义;以前的定义是“函数”printf("%d\n", rand);return 0;
}

在#include <stdlib.h>中,有rand函数(int rand (void);),此时不知道是rand是变量名,函数名

6.2 namespace的定义

• 定义命名空间,需要使用到namespace关键字,后面跟命名空间的名字,然后接一对{}即可,{}中即为命名空间的成员。命名空间中可以定义 变量 / 函数 / 类型 等。

• namespace本质是定义出一个域,这个域跟全局域各自独立,不同的域可以定义同名变量,所以下面的rand不在冲突了。

• C++中域有 函数局部域,全局域,命名空间域,类域。域影响的是编译时语法查找一个 变量 / 函数 / 类型 出处(声明或定义)的逻辑,所以有了域隔离,名字冲突就解决了。局部域和全局域除了会影响编译查找逻辑,还会影响变量的生命周期,命名空间域和类域不影响变量生命周期。

• namespace只能定义在全局,当然他还可以嵌套定义

• 项目工程中多文件中定义的同名namespace会认为是一个namespace,不会冲突。

C++标准库都放在一个叫std(standard)的命名空间中。

#include <stdio.h>
#include <stdlib.h>// 1. 正常的命名空间定义
// Lzc是命名空间的名字,⼀般开发中是用项目名字做命名空间名
// 日常练习可以用自己的名字命名
namespace Lzc
{// 命名空间中可以定义变量/函数/类型int rand = 10;int Add(int left, int right){return left + right;}struct Node{struct Node* next;int val;};
}int main()
{printf("%p\n", rand);// 打印rand函数的地址printf("%d\n", Lzc::rand);// 打印Lzc命名空间里的rand变量return 0;
}//2. 命名空间可以嵌套
namespace S
{namespace Lzc{int rand = 1;int Add(int left, int right){return left + right;}}namespace Ysy{int rand = 2;int Add(int left, int right){return (left + right) * 10;}}
}int main()
{printf("%d\n", S::Lzc::rand);// 1printf("%d\n", S::Ysy::rand);// 2printf("%d\n", S::Lzc::Add(1, 2));// 3printf("%d\n", S::Ysy::Add(1, 2));// 30return 0;
}//3. 多文件中可以定义同名namespace,他们会默认合并到⼀起,就像同一个namespace一样
// Stack.h
#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
#include<assert.h>
namespace Lzc
{typedef int STDataType;typedef struct Stack{STDataType* a;int top;int capacity;}ST;void STInit(ST* ps, int n);void STDestroy(ST* ps);void STPush(ST* ps, STDataType x);void STPop(ST* ps);STDataType STTop(ST* ps);int STSize(ST* ps);bool STEmpty(ST* ps);
}// Stack.cpp
#include"Stack.h"
namespace Lzc
{void STInit(ST* ps, int n){assert(ps);ps->a = (STDataType*)malloc(n * sizeof(STDataType));ps->top = 0;ps->capacity = n;}// 栈顶void STPush(ST* ps, STDataType x){assert(ps);// 满了, 扩容if (ps->top == ps->capacity){printf("扩容\n");int newcapacity = ps->capacity == 0 ? 4 : ps->capacity* 2;STDataType* tmp = (STDataType*)realloc(ps->a,newcapacity * sizeof(STDataType));if (tmp == NULL){perror("realloc fail");return;}ps->a = tmp;ps->capacity = newcapacity;}ps->a[ps->top] = x;ps->top++;}//...
}// test.cpp
#include"Stack.h"
// 全局定义了一份单独的Stack
typedef struct Stack
{int a[10];int top;
}ST;
void STInit(ST* ps) {}
void STPush(ST* ps, int x) {}int main()
{// 调用全局的ST st1;STInit(&st1);STPush(&st1, 1);STPush(&st1, 2);printf("%d\n", sizeof(st1));// 调用S namespace的Lzc::ST st2;printf("%d\n", sizeof(st2));Lzc::STInit(&st2, 4);Lzc::STPush(&st2, 1);Lzc::STPush(&st2, 2);return 0;
}

6.3 namespace的使用

编译查找一个变量的声明/定义时,默认只会在局部或者全局查找,不会到命名空间里面去查找。所以下面程序会编译报错。我们要使用命名空间中定义的变量/函数,有三种方式:

指定命名空间访问项目中推荐这种方式。

using将命名空间中某个成员展开项目中经常访问的不存在冲突的成员推荐这种方式。

using展开命名空间中全部成员项目不推荐,冲突风险很大,日常小练习程序为了方便推荐使用。

#include<stdio.h>namespace Lzc
{int a = 0;int b = 1;
}int main()
{// 编译报错:error C2065: “a”: 未声明的标识符printf("%d\n", a);return 0;
}//1. 指定命名空间访问
int main()
{printf("%d\n", Lzc::a);return 0;
}//2. using将命名空间中某个成员展开
using Lzc::b;
int main()
{printf("%d\n", Lzc::a);printf("%d\n", b);printf("%d\n", b);printf("%d\n", b);printf("%d\n", b);printf("%d\n", b);return 0;
}//3. 展开命名空间中全部成员
using namespace Lzc;
int main()
{printf("%d\n", a);printf("%d\n", b);return 0;
}

7、C++输入&输出

• 是 Input Output Stream 的缩写,是标准的输入、输出流库,定义了标准的输入、输出对象。

• std::cin 是 istream 类的对象,它主要面向窄字符(narrow characters (of type char))的标准输 入流。

• std::cout 是 ostream 类的对象,它主要面向窄字符的标准输出流。

• std::endl 是一个函数,流插入输出时,相当于插如一个换行字符加刷新缓冲区。

现在的阶段,std::endl可以理解为就是"\n"

• >是流提取运算符。(C语言还用这两个运算符做位运算左移/右移)

• 使用C++输入输出更方便,不需要像printf/scanf输入输出时那样,需要手动指定格式,C++的输入输出可以自动识别变量类型(本质是通过函数重载实现的,这个以后会讲到),其实最重要的是 C++的流能更好的支持自定义类型对象的输入输出

• IO流涉及类和对象,运算符重载、继承等很多面向对象的知识,这些知识我们还没有讲解,所以这里我们只能简单认识一下C++ IO流的用法,后面我们会有专门的一个章节来细节IO流库。

• cout/cin/endl等都属于C++标准库,C++标准库都放在一个叫std(standard)的命名空间中,所以要 通过命名空间的使用方式去用他们。

注意:这里不是说std里有iostream,那为什么还要引入头文件<iostream>,

在C++标准库中,iostream是一个命名空间,包含了处理输入输出操作的主要类和函数,如cincout等。虽然这些功能已经包含在库中,但是通过引入#include <iostream>头文件,程序员可以明确地告诉编译器他们想要使用这部分功能,并使得编译器能在程序中定位到这些相关的声明和定义。

头文件不仅提供类和函数的原型,还可以包含一些预处理器指令(如宏定义),以及可能导致链接阶段错误的其他信息。此外,通过引用头文件,我们能享受到编译器的依赖管理和类型检查,提高代码的可读性和维护性。

所以,即使库内部已经有了这些内容,每使用一次iostream里的功能都需要包含这个头文件,这是一种编程约定和组织方式。

一般日常练习中我们可以using namespace std,实际项目开发中不建议using namespace std。

• VS系列编译器没有包含<stdio.h>,也可以使用printf和scanf,因为在<iostream>里间接包含了。其他编译器可能会报错,就加上<stdio.h>。

#define _CRT_SECURE_NO_WARNINGS 1
#include <iostream>
using namespace std;
int main()
{int a = 0;double b = 0.1;char c = 'x';cout << a << " " << b << " " << c << endl;std::cout << a << " " << b << " " << c << std::endl;scanf("%d%lf", &a, &b);printf("%d %lf\n", a, b);// 可以自动识别变量的类型cin >> a;cin >> b >> c;cout << a << endl;cout << b << " " << c << endl;return 0;
}
#include<iostream>
using namespace std;
int main()
{// 在io需求⽐较⾼的地方,如部分大量输入的竞赛题中,加上以下3行代码// 可以提高C++IO效率ios_base::sync_with_stdio(false);cin.tie(nullptr);cout.tie(nullptr);return 0;
}
  1. ios_base::sync_with_stdio(false):取消标准输入输出缓冲区(stdio)与C库的标准输入输出缓冲区之间的同步。这可以避免在处理大量数据时由于等待I/O操作完成而导致的性能瓶颈。

  2. cin.tie(nullptr)cout.tie(nullptr):取消 cin 和 cout 对齐到同一线程的同步。当有多线程同时读写控制台输出时,这可以避免因为线程间的同步而产生的额外开销。

在需要频繁输入输出、并且对速度有较高要求的情况下,例如编程竞赛或大数据处理等场景,加入这些代码有助于提升程序运行效率。然而,在日常开发中,如果不需要处理这种高并发场景,保持默认设置通常是更安全的选择。

8、缺省参数

• 缺省参数是声明或定义函数时为函数的参数指定一个缺省值。在调用该函数时,如果没有指定实参则采用该形参的缺省值,否则使用指定的实参,缺省参数分为全缺省和半缺省参数。(有些地方把缺省参数也叫默认参数)

• 全缺省就是全部形参给缺省值,半缺省就是部分形参给缺省值。C++规定半缺省参数必须从右往左依次连续缺省,不能间隔跳跃给缺省值

• 带缺省参数的函数调用,C++规定必须从左到右依次给实参,不能跳跃给实参。

函数声明和定义分离时,缺省参数不能在函数声明和定义中同时出现,规定函数声明处给缺省 值。(考虑到函数声明和定义处的缺省值不一致等问题)

#include <iostream>
#include <assert.h>using namespace std;
void Func(int a = 0)
{cout << a << endl;
}int main()
{Func(); // 没有传参时,使用参数的默认值 0Func(10); // 传参时,使用指定的实参 10return 0;
}
#include <iostream>using namespace std;// 全缺省
void Func1(int a = 10, int b = 20, int c = 30)
{cout << "a = " << a << endl;cout << "b = " << b << endl;cout << "c = " << c << endl << endl;
}// 半缺省
void Func2(int a, int b = 10, int c = 20)
{cout << "a = " << a << endl;cout << "b = " << b << endl;cout << "c = " << c << endl << endl;
}int main()
{Func1();Func1(1);Func1(1, 2);Func1(1, 2, 3);Func2(100);Func2(100, 200);Func2(100, 200, 300);return 0;
}
// Stack.h
#include <iostream>
#include <assert.h>using namespace std;typedef int STDataType;
typedef struct Stack
{STDataType* a;int top;int capacity;
}ST;void STInit(ST* ps, int n = 4);// Stack.cpp
#include"Stack.h"// 缺省参数不能声明和定义同时给
void STInit(ST* ps, int n)
{assert(ps && n > 0);ps->a = (STDataType*)malloc(n * sizeof(STDataType));ps->top = 0;ps->capacity = n;
}// test.cpp
#include"Stack.h"int main()
{ST s1;STInit(&s1);// 确定要插入1000个数据,初始化时就开好,避免扩容ST s2;STInit(&s2, 1000);return 0;
}

9、函数重载

C++支持在同一作用域出现同名函数,但是要求这些同名函数的形参不同,可以是

参数个数不同,参数类型不同,参数类型顺序不同

注意:返回值类型不同不能作为重载条件,因为调用时也无法区分

C语言是不支持同一作用域中出现同名函数的

#include<iostream>using namespace std;// 1、参数类型不同
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;
}// 2、参数个数不同
void f()
{cout << "f()" << endl;
}void f(int a)
{cout << "f(int a)" << endl;
}// 3、参数类型顺序不同
void f(int a, char b)
{cout << "f(int a,char b)" << endl;
}void f(char b, int a)
{cout << "f(char b, int a)" << endl;
}// 返回值不同不能作为重载条件,因为调用时也无法区分
//void fxx()
//{}
//
//int fxx()
//{
// return 0;
//}// 下面两个函数构成重载
// f1()调用时,会报错,存在歧义,编译器不知道调用谁
void f1()
{cout << "f()" << endl;
}void f1(int a = 10)
{cout << "f(int a)" << endl;
}int main()
{Add(10, 20);Add(10.1, 20.2);f();f(10);f(10, 'a');f('a', 10);return 0;
}

10、引用

10.1 引用的概念和定义

引用不是新定义一个变量,而是给已存在变量取了一个别名编译器不会为引用变量开辟内存空间, 它和它引用的变量共用同一块内存空间。

类型& 引用别名 = 引用对象;

#include<iostream>
using namespace std;
int main()
{int a = 0;// 引用:b和c是a的别名int& b = a;int& c = a;// 也可以给别名b取别名,d相当于还是a的别名int& d = b;++d;// 这里取地址我们看到是⼀样的cout << &a << endl;cout << &b << endl;cout << &c << endl;cout << &d << endl;return 0;
}

10.2 引用的特性

• 引用在定义时必须初始化

一个变量可以有多个引用

C++规定引用不能改变指向,引用一旦引用一个实体,再不能引用其他实体

#include<iostream>using namespace std;int main()
{int a = 10;// 编译报错:“ra”: 必须初始化引⽤//int& ra;int& b = a;int c = 20;// 这里并不是让b引用c,因为C++规定引用不能改变指向// 这里是⼀个赋值,即 a = 20;b = c;cout << &a << endl;cout << &b << endl;cout << &c << endl;return 0;
}

10.3 引用的使用

• 引用在实践中主要是于引用传参和引用做返回值中减少拷贝提高效率改变引用别名时同时改变引用对象

引用传参跟指针传参功能是类似的,引用传参相对更方便一些

• 引用返回值的场景相对比较复杂,我们在这里简单讲了一下场景,还有一些内容后续类和对象章节中会继续深入讲解。

引用和指针在实践中相辅相成,功能有重叠性,但是各有特点,互相不可替代(如:在链表里的next,如果用引用,因为不能改变其指向,所以不能改变next,只能用指针)。C++的引用跟其他语言的引用(如Java)是有很大的区别的,除了用法,最大的区别就是是,C++引用定义后不能改变指向, Java的引用可以改变指向。

• 一些主要用C代码实现版本数据结构教材中,使用C++引用替代指针传参,目的是简化程序,避开复杂的指针。

#include<iostream>using namespace std;void Swap(int& rx, int& ry)
{int tmp = rx;rx = ry;ry = tmp;
}int main()
{int x = 0, y = 1;cout << x << " " << y << endl;Swap(x, y);cout << x << " " << y << endl;return 0;
}
#include<iostream>using namespace std;typedef int STDataType;
typedef struct Stack
{STDataType* a;int top;int capacity;
}ST;void STInit(ST& rs, int n = 4)
{rs.a = (STDataType*)malloc(n * sizeof(STDataType));rs.top = 0;rs.capacity = n;
}// 栈顶
void STPush(ST& rs, STDataType x)
{// 满了, 扩容if (rs.top == rs.capacity){printf("扩容\n");int newcapacity = rs.capacity == 0 ? 4 : rs.capacity * 2;STDataType* tmp = (STDataType*)realloc(rs.a, newcapacity *sizeof(STDataType));if (tmp == NULL){perror("realloc fail");return;}rs.a = tmp;rs.capacity = newcapacity;}rs.a[rs.top] = x;rs.top++;
}// int STTop(ST& rs)
int& STTop(ST& rs)
{return rs.a[rs.top - 1];
}int main()
{ST st1;STInit(st1);STPush(st1, 1);STPush(st1, 2);cout << STTop(st1) << endl;// 2STTop(st1) += 10;// 返回了rs.a[rs.top - 1]的别名,可以进行修改cout << STTop(st1) << endl;// 12return 0;
}
#include<iostream>
#include<assert.h>using namespace std;
typedef struct SeqList
{int a[10];int size;
}SLT;void SeqPushBack(SLT& sl, int x)
{}typedef struct ListNode
{int val;struct ListNode* next;
}LTNode, * PNode;// 指针变量也可以取别名,这里LTNode*& phead就是给指针变量取别名
// 这样就不需要用⼆级指针了,相对而言简化了程序
//void ListPushBack(LTNode** phead, int x)
//void ListPushBack(LTNode*& phead, int x)
void ListPushBack(PNode& phead, int x)
{PNode newnode = (PNode)malloc(sizeof(LTNode));assert(newnode);newnode->val = x;newnode->next = NULL;if (phead == NULL){phead = newnode;}else{//...}
}int main()
{PNode plist = NULL;ListPushBack(plist, 1);return 0;
}

10.4 const引用

• 可以引用一个const对象,但是必须用const引用。const引用也可以引用普通对象,因为对象的访
问权限在引用过程中可以缩小,但是不能放大。

• 不需要注意的是类似 int& rb = a * 3; double d = 12.34; int& rd = d; 这样一些场景下a * 3的结果保存在一个临时对象中, int& rd = d 也是类似,在类型转换中会产生 存储中间值的临时对象,也就是,rb和rd引用的都是临时对象,而C++规定临时对象具有常性(只能读,不能改),所以这里就触发了权限放大,必须要用const引用才可以。

• 所谓临时对象就是编译器需要一个空间暂存表达式的求值结果时而临时创建的一个未命名的对象,C++中把这个未命名对象叫做临时对象。

• C++中的临时对象除了上述的 表达式求值,类型转换,还有以下两种:

     1. 函数返回值:当你调用一个函数并且它的返回类型不是引用或指针时,会生成一个临时对象来存储返回值。

int x = getSomeValue(); // getSomeValue() 返回一个新的整数,x 是对该临时对象的引用

    2. 参数传递:在函数参数列表中,如果传递的是非引用类型的值,也会创建临时对象来存储实际传入的数据。

void func(int temp) { // temp 是对一个临时变量的引用 }

func(42); // 内部创建了一个临时变量来作为func的参数

int main()
{const int a = 10;// 编译报错:error C2440: “初始化”: ⽆法从“const int”转换为“int &”// 这里的引用是对a访问权限的放⼤//int& ra = a;// 这样才可以const int& ra = a;// 编译报错:error C3892: “ra”: 不能给常量赋值//ra++;// 这里的引用是对b访问权限的缩⼩int b = 20;const int& rb = b;// 编译报错:error C3892: “rb”: 不能给常量赋值//rb++;return 0;
}
#include<iostream>using namespace std;int main()
{int a = 10;const int& ra = 30;// 编译报错: “初始化”: 无法从“int”转换为“int &”// 临时对象具有常性// int& rb = a * 3;const int& rb = a * 3;double d = 12.34;// 编译报错:“初始化”: ⽆法从“double”转换为“int &”// 临时对象具有常性// int& rd = d;const int& rd = d;return 0;
}

10.5 指针和引用的关系(面试高频考查点)

C++中指针和引用就像两个性格迥异的亲兄弟,指针是哥哥,引用是弟弟,在实践中相辅相成,功能有重叠性,但是各有特点,互相不可替代(如:在链表里的next,如果用引用,因为不能改变其指向,所以不能改变next,只能用指针)。

• 语法概念上引用是一个变量的取别名不开空间,指针是存储一个变量地址,要开空间。

• 引用在定义时必须初始化,指针可以不初始化。

• C++规定引用不能改变指向,引用在初始化时引用一个对象后,就不能再引用其他对象;而指针可以在不断地改变指向对象。

• 引用可以直接访问指向对象,指针需要解引用才是访问指向对象。

sizeof中含义不同,引用结果为引用类型的大小,但指针始终是地址空间所占字节个数(32位平台下 占4个字节,64位下是8byte)

指针很容易出现空指针和野指针的问题,引用很少出现,相对更安全一些。

11、inline

• inline修饰的函数叫做内联函数编译时C++编译器会在调用的地方展开内联函数,这样调用内联 函数就不需要建立栈帧了,就可以提高效率

• inline对于编译器而言只是一个建议,也就是说,你加了inline编译器也可以选择在调用的地方不展开,不同编译器关于inline什么情况展开各不相同,因为C++标准没有规定这个。inline适用于频繁调用的短小函数对于递归函数,代码相对多一些的函数,加上inline也会被编译器忽略

• C语言实现宏函数也会在预处理时替换展开,但是宏函数实现很复杂很容易出错的,且不方便调 试,C++设计了inline目的就是替代C的宏函数。

• vs编译器 debug版本下面默认是不展开inline的,这样方便调试,debug版本想展开需要设置⼀下 以下两个地方。

• inline不建议声明和定义分离到两个文件,分离会导致链接错误。因为inline被展开,就没有函数地址,链接时会出现报错,建议直接声明处定义

#include<iostream>using namespace std;inline int Add(int x, int y)
{int ret = x + y;ret += 1;ret += 1;ret += 1;return ret;
}
int main()
{// 可以通过汇编观察程序是否展开// 有call Add语句就是没有展开,没有就是展开了int ret = Add(1, 2);cout << Add(1, 2) * 5 << endl;return 0;
}
// F.h
#include <iostream>using namespace std;inline void f(int i);// F.cpp
#include "F.h"void f(int i)
{cout << i << endl;
}// main.cpp
#include "F.h"int main()
{// 链接错误:无法解析的外部符号 "void __cdecl f(int)" (?f@@YAXH@Z)f(10);return 0;
}

12、nullptr

NULL实际是一个宏,在传统的C头文件(stddef.h)中,可以看到如下代码:

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

• C++中NULL可能被定义为字面常量0,或者C中被定义为无类型指针(void*)的常量。不论采取何种定义,在使用空值的指针时,都不可避免的会遇到一些麻烦,如:

在C语言中,由于类型检查不严格,类型都可以转换,f(NULL),两个重载函数都可以调用,报错

在C++中,本想通过f(NULL)调用指针版本的 f(int*)函数,但是由于NULL被定义成0,调用了f(int x),因此与程序的初衷相悖。f((void*)NULL); 调用会报错(类型检查严格)。

• C++11中引入了nullptr,nullptr是一个特殊的关键字,nullptr是一种特殊类型的字面量,它可以转换成任意其他类型的指针类型

nullptr只能被隐式地转换为指针类型,而不能被转换为整数类型。

#include<iostream>using namespace std;void f(int x)
{cout << "f(int x)" << endl;
}void f(int* ptr)
{cout << "f(int* ptr)" << endl;
}int main()
{f(0); // f(int x)f(NULL); // f(int x)f((int*)NULL); // 强制转换 f(int* ptr)// 编译报错:error C2665: “f”: 2 个重载函数中没有一个匹配的参数类型// f((void*)NULL);f(nullptr); //f(int* ptr)return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/882341.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

10月9日

肯定是对x求导 刨根问底求导数解析式 区间再现均值不等式 没利用B-E 0 同解方程组 趋于0的时候&#xff0c;看1次项 没有考虑x -1的情况 还要加一&#xff0c;非齐次解

AdaTAD(CVPR 2024)视频动作检测方法详解

前言 论文&#xff1a;End-to-End Temporal Action Detection with 1B Parameters Across 1000 Frames 代码&#xff1a;AdaTAD 从论文标题可以看出&#xff0c;AdaTAD 可以在 1B 参数且输入视频在 1000 帧的情况下实现端到端的训练&#xff0c;核心创新点是引入 Temporal-Inf…

STM32_实验4_控制蜂鸣器

1.设置 PB2 引脚&#xff0c;生成代码。 2.打开蜂鸣器 // 循环反复HAL_GPIO_WritePin(GPIOB, GPIO_PIN_2, GPIO_PIN_SET); // 开启蜂鸣器printf("beep on\n");HAL_Delay(500); // 等待响500msHAL_GPIO_WritePin(…

解锁C++多态的魔力:灵活与高效的编码艺术(下)

文章目录 前言&#x1f3b1;四、多态的原理&#x1f52e;4.1 虚函数表&#xff08;vtable&#xff09;&#x1f52e;4.2 派生类对象中的虚函数表4.2.1 编写程序去访问虚函数表4.2.2 虚表存储位置的验证 &#x1f3b1;五、 多态的静态绑定和动态绑定&#x1f52e;5.1 静态绑定&a…

spring底层原理

本文参考黑马程序员的spring底层讲解&#xff0c;想要更详细的可以去看视频。 另外文章会每日更新&#xff0c;大概持续1个月&#xff01;&#xff01;&#xff01;每天更新一讲 这部分比较抽象&#xff0c;要经常复习&#xff01;&#xff01;&#xff01; 一、BeanFactory与A…

【JPCS独立出版 | 福州大学主办 | 有确定的ISSN号】第三届可再生能源与电气科技国际学术会议(ICREET 2024)

第三届可再生能源与电气科技国际学术会议&#xff08;ICREET 2024&#xff09; 2024 3rd International Conference on Renewable Energy and Electrical Technology ICREET 2024已成功申请JPCS - Journal of Physics: Conference Series (ISSN:1742-6596) 独立出版&#xf…

引领智慧文旅新纪元,开启未来旅游新境界

融合创新科技&#xff0c;重塑旅游体验&#xff0c;智慧文旅项目定义旅游新未来 在全球化的浪潮中&#xff0c;旅游已成为连接世界的重要纽带。智慧文旅项目&#xff0c;不仅仅是一次技术的革新&#xff0c;更是对旅游行业未来发展的一次深刻思考。信鸥科技通过运用云计算、大数…

Vue3动态组件原来是这样

什么是Vue3动态组件 在Vue3中&#xff0c;动态组件简单来说就是根据不同的条件进行不同组件的渲染&#xff0c;可以联想一下在前端中常用到的动态样式 基本使用 在Vue3中&#xff0c;动态组件的使用也是非常简单的&#xff0c;只需要使用<component>标签&#xff0c;并…

WPFDeveloper正式版发布

WPFDeveloper WPFDeveloper一个基于WPF自定义高级控件的WPF开发人员UI库&#xff0c;它提供了众多的自定义控件。 该项目的创建者和主要维护者是现役微软MVP 闫驚鏵: https://github.com/yanjinhuagood 该项目还有众多的维护者&#xff0c;详情可以访问github上的README&…

Redis 高可用:从主从到集群的全面解析

目录 一、主从复制 (基础)1. 同步复制a. 全量数据同步b. 增量数据同步c. 可能带来的数据不一致 2. 环形缓冲区a. 动态调整槽位 3. runid4. 主从复制解决单点故障a. 单点故障b. 可用性问题 5. 注意事项a. Replica 主动向 Master 建立连接b. Replica 主动向 Master 拉取数据 二、…

STM32传感器模块编程实践(八) HX711压力传感器称重模块简介及驱动源码

文章目录 一.概要二.HX711主要技术指标三.HX711模块参考原理图四.模块接线说明五.模块工作原理介绍六.模块通讯协议介绍七.STM32单片机与HX711模块实现重量测量实验1.硬件准备2.软件工程3.软件主要代码4.实验效果 八.小结 一.概要 电子秤是将检测与转换技术、计算机技术、信息…

Python网络爬虫从入门到实战

目录 引言 一、网络爬虫的概念 二、 网络爬虫的基本工作流程 &#xff08;一&#xff09;过程&#xff1a; &#xff08;二&#xff09;安装requests模块和beautifulsoup4模块 &#xff08;三&#xff09;requests库的使用 1、requests库的基本介绍 2、导入requests库的…

一款零依赖、跨平台的流媒体协议处理工具,支持 RTSP、WebRTC、RTMP 等视频流协议的处理

大家好&#xff0c;今天给大家分享一款功能强大的流媒体协议处理工具go2rtc&#xff0c;支持多种协议和操作系统&#xff0c;具有零依赖、零配置、低延迟等特点。 项目介绍 go2rtc可以从各种来源获取流&#xff0c;包括 RTSP、WebRTC、HomeKit、FFmpeg、RTMP 等&#xff0c;并…

学习文档10/16

MySQL 字符集&#xff1a; MySQL 支持很多种字符集的方式&#xff0c;比如 GB2312、GBK、BIG5、多种 Unicode 字符集&#xff08;UTF-8 编码、UTF-16 编码、UCS-2 编码、UTF-32 编码等等&#xff09;。 查看支持的字符集 你可以通过 SHOW CHARSET 命令来查看&#xff0c;支持…

ARINC 429总线协议

一、概述 ARINC 是美国航空无线电公司英文字头的缩写&#xff0c; 该公司1977年7月21日出版了“ARINC 429规范”一书&#xff0c;429规范就是飞机电子系统之间数字式数据传输的标准格式&#xff0c;在飞机上使用429总线的电子设备均应遵守这个规范&#xff0c;这样才能保证电子…

Redis应用高频面试题

Redis 作为一个高性能的分布式缓存系统,广泛应用于后端开发中,因此在后端研发面试中,关于 Redis 的问题十分常见。 本文整理了30个常见的 Redis 面试题目,涵盖了 Redis 的源码、数据结构、原理、集群模式等方面的知识,并附上简要的回答,帮助大家更好地准备相关的面试。 …

2024年【N2观光车和观光列车司机】及N2观光车和观光列车司机模拟考试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 N2观光车和观光列车司机考前必练&#xff01;安全生产模拟考试一点通每个月更新N2观光车和观光列车司机模拟考试题题目及答案&#xff01;多做几遍&#xff0c;其实通过N2观光车和观光列车司机操作证考试很简单。 1、…

LabVIEW提高开发效率技巧----用户权限控制

在LabVIEW开发中&#xff0c;用户权限控制是一个重要的设计模块&#xff0c;尤其在多用户系统中&#xff0c;它可以确保数据安全并控制不同用户的操作权限。为了实现用户权限控制&#xff0c;可以通过角色与权限管理模块来进行设计和实施。以下将从多个角度详细说明如何在LabVI…

Sentinel 快速入门

前置推荐阅读:Sentinel 介绍-CSDN博客 前置推荐阅读&#xff1a;Nacos快速入门-CSDN博客 快速开始 欢迎来到 Sentinel 的世界&#xff01;这篇新手指南将指引您快速入门 Sentinel。 Sentinel 的使用可以分为两个部分: 核心库&#xff08;Java 客户端&#xff09;&#xff1a…

新版vs code + Vue高亮、语法自动补全插件

vs code 版本或及以上 安装以下三个插件插件 Vetur Vue语法支持。包括语法高亮、语法代码提示、语法lint检测 ESLint语法纠错 Prettier 2.左下角设置 3.进行配置 配置内容&#xff1a; {"editor.fontSize": 20,"window.zoomLevel": 1,"workben…