c++入门基础

一前言

1.发展历史

C++的起源可以追溯到1979,当时的Bjarne Stroustrup(本贾尼·斯特劳斯特卢普)面对项目中复杂的软件开发任务,特别是模拟和操作系统的开发工作,他感受到现有语言(C语言)在表达能力、可维护性和可扩展性方面的不足。

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

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

C++的标准化⼯作于1989年开始,并成⽴了⼀个ANSI和ISO(InternationalStandardsOrganization)国际标准化组织的联合标准化委员会。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标准被投⼊使⽤。

C++参考文档

Reference - C++ ReferenceC++ 参考手册 - cppreference.comicon-default.png?t=N7T8https://zh.cppreference.com/w/cpp

2.c++重要性

TIOBE排行榜是根据互联网上有经验的程序员、课程和第三方厂商的数量,并使⽤搜索引擎(如
Google、Bing、Yahoo!)以及Wikipedia、Amazon、YouTube和Baidu(百度)统计出排名数据,只是反映某个编程语⾔的热⻔程度,并不能说明⼀⻔编程语⾔好不好,或者⼀⻔语⾔所编写的代码数量多少。
 2024年6⽉TIOBE发布的编程语⾔排⾏榜

3.C++在生活中的应用

C++的应⽤领域服务器端、游戏(引擎)、机器学习引擎、⾳视频处理、嵌⼊式软件、电信设备、⾦融应⽤、基础库、操作系统、编译器、基础架构、基础⼯具、硬件交互等很多⽅⾯都有。
1. ⼤型系统软件开发。如编译器、数据库、操作系统、浏览器等等
2. ⾳视频处理。常⻅的⾳视频开源库和⽅案有FFmpeg、WebRTC、Mediasoup、ijkplayer,⾳视频开发最主要的技术栈就是C++。
3. PC客⼾端开发。⼀般是开发Windows上的桌⾯软件,⽐如WPS之类的,技术栈的话⼀般是C++和QT,QT 是⼀个跨平台的 C++图形⽤⼾界⾯(Graphical User Interface,GUI)程序。
4. 服务端开发。各种⼤型应⽤⽹络连接的⾼并发后台服务。这块Java也⽐较多,C++主要⽤于⼀些对性能要求⽐较⾼的地⽅。如:游戏服务、流媒体服务、量化⾼频交易服务等
5. 游戏引擎开发。很多游戏引擎就都是使⽤C++开发的,游戏开发要掌握C++基础和数据结构,学习图形学知识,掌握游戏引擎和框架,了解引擎实现,引擎源代码可以学习UE4、Cocos2d-x等开源引擎实现
6. 嵌⼊式开发。嵌⼊式把具有计算能⼒的主控板嵌⼊到机器装置或者电⼦装置的内部,通过软件能够控制这些装置。⽐如:智能⼿环、摄像头、扫地机器⼈、智能⾳响、⻔禁系统、⻋载系统等等,粗略⼀点,嵌⼊式开发主要分为嵌⼊式应⽤和嵌⼊式驱动开发。
7. 机器学习引擎。机器学习底层的很多算法都是⽤C++实现的,上层⽤python封装起来。如果你只想准备数据训练模型,那么学会Python基本上就够了,如果你想做机器学习系统的开发,那么需要学会C++。
8. 测试开发/测试。每个公司研发团队,有研发就有测试,测试主要分为测试开发和功能测试,测试开发⼀般是使⽤⼀些测试⼯具(selenium、Jmeter等),设计测试⽤例,然后写⼀些脚本进⾏⾃动化测试,性能测试等,有些还需要⾃⾏开发⼀些测试⽤具。功能测试主要是根据产品的功能,设计测试⽤例,然后⼿动的⽅式进⾏测试。

4.第一个c++程序

5.namespace命名空间

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

在C/C++中,变量、函数和后⾯要学到的类都是⼤量存在的,这些变量、函数和类的名称将都存在于全 局作⽤域中,可能会导致很多冲突。使⽤命名空间的⽬的是对标识符的名称进⾏本地化,以避免命名 冲突或名字污染,namespace关键字的出现就是针对这种问题的。 c语⾔项⽬类似下⾯程序这样的命名冲突是普遍存在的问题,C++引⼊namespace就是为了更好的解决 这样的问题。

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

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

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

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

• 项⽬⼯程中多⽂件中定义的同名namespace会认为是⼀个namespace,不会冲突。

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

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

编译查找⼀个变量的声明/定义时,默认只会在局部或者全局查找,不会到命名空间⾥⾯去查找。

而“::”域限定符可以在命名空间里查找数据。

Tip:命名空间必须写在全局变量中

假如在一个项目中,已经使用了一次命名空间,但是仍有两个使用相同变量又该怎么办呢?

这就是namespace的嵌套使用

在同一个命名空间里嵌套定义了 a b两个命名空间

namespace的展开——using

前面说到编译查找⼀个变量的声明/定义时,默认只会在局部或者全局查找,不会到命名空间⾥⾯去查找。所以 下⾯程序会编译报错。

// 指定命名空间访问
 
int main()
{
 printf("%d\n", N::a);
 return 0; 
}
 
// using将命名空间中某个成员展开
 
using N::b;
int main()
{
 printf("%d\n", N::a);
 printf("%d\n", b);
 return 0; 
}
 
// 展开命名空间中全部成员
 
using namespce N;
int main()
{
 printf("%d\n", a);
 printf("%d\n", b);
 return 0; 
}

二、C++的输入输出

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

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

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

• std::endl 是⼀个函数,流插⼊输出时,相当于插⼊⼀个换⾏字符加刷新缓冲区。

•<<是流插入运算符,> >是流提取运算符。(C语⾔还⽤这两个运算符做位运算左移/右移)

• 使⽤C++输⼊输出更⽅便,不需要像printf/scanf输⼊输出时那样,需要⼿动指定格式,C++的输⼊输出可以⾃动识别变量类型,其实最重要的是 C++的流能更好的⽀持⾃定义类型对象的输⼊输出。

• cout/cin/endl等都属于C++标准库,C++标准库都放在⼀个叫std(standard)的命名空间中,所以要 通过命名空间的使⽤⽅式去⽤他们。
三、缺省参数与函数重载

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

• 全缺省就是全部形参给缺省值,半缺省就是部分形参给缺省值。

• 函数声明和定义分离时,缺省参数不能在函数声明和定义中同时出现,规定必须函数声明给缺省 值。

// 全缺省
 
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 Func1(int a = 10, int b = 20, int c = 30)
{
 cout << "a = " << a << endl;
 cout << "b = " << b << endl;
 cout << "c = " << c << endl << endl;
}

Tips:C++规定半缺省参数必须从右往左依次连续缺省,不能间隔跳跃给缺省值。

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

3.2函数重载的类型

C++⽀持在同⼀作⽤域中出现同名函数,但是要求这些同名函数的形参不同,可以是参数个数不同或者类型不同。这样C++函数调⽤就表现出了多态⾏为,使⽤更灵活。C语⾔是不⽀持同⼀作⽤域中出现同名函数的。

参数的类型不同

                     相同函数名的函数其形参类型不同则为函数重载

// 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()调用时会报错,因为编译器不知道执行哪一个,有歧义!

// 下⾯两个函数构成重载
 
// f()但是调⽤时,会报错,存在歧义,编译器不知道调⽤谁
 
void f1()
{
 cout << "f()" << endl;
}
 
void f1(int a = 10)
{
 cout << "f(int a)" << endl;
}

四,引用与指针

4.1引用的定义和特性

引⽤不是新定义⼀个变量,⽽是给已存在变量取了⼀个别名,编译器不会为引⽤变量开辟内存空间, 它和它引⽤的变量共⽤同⼀块内存空间。⽐如:⽔浒传中李逵,宋江叫"铁⽜",江湖上⼈称"⿊旋 ⻛";林冲,外号豹⼦头;

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

引⽤的特性:

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

• ⼀个变量可以有多个引⽤

• 引⽤⼀旦引⽤⼀个实体,再不能引⽤其他实体

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

 可以引⽤⼀个const对象,但是必须⽤const引⽤。const引⽤也可以引⽤普通对象,因为对象的访 问权限在引⽤过程中可以缩⼩,但是不能放⼤。

可以看到 a为常量 只能读不能写,但是此处的引用参数可读可写(相当于放大了权限)所以编译器会报错!

所以C++中存在 const引用的语法

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;
}

const引用也可以直接引用常量

const int& a=20;

• 不需要注意的是类似 int& rb = a*3;  double d = 12.34;  int& rd = d; 这样⼀些场景下的结果保存在⼀个临时对象中, int& rd = d 也是类似,在类型转换中会产⽣临时对象存储中间值,也就是时,rb和rd引⽤的都是临时对象,⽽C++规定临时对象具有常性,所以这⾥就触发了权限放⼤,必须要⽤常引⽤才可以。
 

#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;
}

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

//此时b可读可写
 
 
int b=20;
const int& rb=b;//权限缩小 可以实现
 
b++;
rb++;//会报错 因为是常量引用不可写只可读
//此时b可读可写
 
 
int b=20;
const int& rb=b;//权限缩小 可以实现
 
b++;
rb++;//会报错 因为是常量引用不可写只可读
4.3引用与指针的关系

C++中指针和引⽤就像两个性格迥异的亲兄弟,指针是哥哥,引⽤是弟弟,在实践中他们相辅相成,功能有重叠性,但是各有自己的特点,互相不可替代。

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

• 引⽤在定义时必须初始化,指针建议初始化,但是语法上不是必须的。

• 引⽤在初始化时引⽤⼀个对象后,就不能再引⽤其他对象;⽽指针可以在不断地改变指向对象。

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

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

• 指针很容易出现空指针和野指针的问题,引⽤很少出现,引⽤使⽤起来相对更安全⼀些

五、内联与宏

inline函数的使用与意义

• ⽤inline修饰的函数叫做内联函数,编译时C++编译器会在调⽤的地⽅展开内联函数,这样调⽤内联函数就需要建⽴栈帧了,就可以提⾼效率。

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

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

inline如何替代宏呢?

可以用宏写一个相加函数
 

#define ADD(a, b) ((a) + (b))

提出三个问题:

A为什么不能加分号?

B为什么要加里面的括号?

C为什么要加外面的括号?

A.宏函数是展开的,宏函数中加分号,当应用于if else语句时也会带上分号,编译器就会报错。

B.反例如下、

 int ret = ADD(1, 2);
 
 cout << ADD(1, 2) << endl;
 
 cout << ADD(1, 2)*5 << endl;

在宏函数中此处语句会被替换成1+2*5而不是(1+2)*5

C.反例如下

 int x = 1, y = 2;
 
 ADD(x & y, x | y); 
 
 // -> (x&y+x|y)

因为优先级的关系此时会先进行加法而不是按位与和按位或运算,与我们预期相悖。

Tip:

内联函数是建议性地展开相对宏函数的定义规则我们更愿意去实现相对安全的内联函数

  • 11
    点赞
  • 23
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值