【C++入门】C语言的不足之处

概要

C++入门主要讲的是C语言的一些不足,C++作为补充,来补充C的不足之处

C++的关键字有63个,C语言有32个(作为了解,不需要专门记)

变量的命名规则:

  1. 变量名必须以字母或下划线开头。
  2. 变量名只能包含字母、数字和下划线。不允许使用其他特殊字符。
  3. 不能与关键字重名

C语言中有命名冲突的问题(与库函数,或者工程里的其他变量/函数名发生冲突),因此就引出了命名空间的概念。

ps:前面的文章有专门讲过命名空间,这里就不再重复介绍辣,可以去翻翻之前的文章

输入输出

istream 输入流

ostream 输出流

而C++中常用到的cin和cout就分别是istream和ostream的对象

cin >> 流提取运算符 输入

cout << 流插入运算符 输出

cout 可以自动识别类型(但是是基于函数重载基础上的,后面会说到函数重载)

类库提供的头文件中已经对"<<"和">>"进行了重载,使之作为流插入运算符和流提取运算符,能用来输出和输入C++标准类型的数据。

本质是在补充C语言的不足 对全局的变量,函数,类型进行封装,防止引发冲突

也可以去库函数中查询相关资料

C++官网:en.cppreference.com/w/

C++的输入输出比scanf/ printf要慢一点点 why?

因为:C++会同步C语言,会检查C语言的错误,如果希望快一点,可以选择关闭同步C语言或者使用scanf / printf

缺省参数(默认参数)

注意:

1)传参是从左往右传参, 缺省是从右向左缺省

2)缺省参数应该在声明的时候给缺省值,定义给缺省值没有实际意义,这样在传参的时候会转化到对应参数个数

3)必须是常量或者全局变量

1】全缺省:会用缺省值来初始化

如果给了参数则用参数值来初始化

2】半缺省:从右向左缺省

函数重载

同一作用域 形参(参数个数 类型 类型的顺序)不同只有返回值不同是不行的

无参调用存在歧义 调用0个的时候存在歧义

为什么C++支持函数重载? 答:与函数名修饰规则有关 以及 编译链接过程有关

三个文件:Stack.h Stack.c test.c

预处理 头文件展开 宏替换 去除注释 条件编译 .i 最后生成Stack.i test.i

编译 语法检查 生成对应汇编代码 .s 最后生成 Stack.s test.s

汇编 xor(逻辑异或) cpu是不认识汇编代码的,只识别二进制。需要把汇编代码转换成二进制机器码 .o目标文件 最后生成 Stack.o test.o

链接 可执行程序 .exe /a.out(不指定) 此过程中 才会去看会不会重命名(找到定义 兑现承诺) 头文件中声明代表承诺 链接的过程就是一个兑现承诺的过程

error:无法解析外部符号 链接错误

一般都是缺少定义(有声明无定义) ,声明和定义不能同时赋值(缺省值),只能在声明中给缺省值,相当于将缺省参数补全了

因此C++支持函数重载而C语言不支持的原因在于:

1、C语言同名编译阶段就会报错 因为同名函数编译器不知道找哪一个

2、C++ 编译链接 g++将函数名和参数的数据类型缩写写入

windows下名称修饰规则:

所有函数都要链接吗?

如果在当前文件就有定义的话 编译阶段直接(兑现承诺),不需要进行链接操作,

因为编译的时候给了定义,相当于直接兑现承诺,就不需要进行链接了。

引用

给已存变量取别名,与被引用的变量公用一块命名空间

给引用变量赋值相当于给被引用变量赋值

引用的价值?

一个变量可以有多个引用,但是引用一旦有一个实体就不能引用其他实体了

引用在定义的时候必须初始化

就是取别名,认为没有开辟新的空间

注意:

1.定义时就必须初始化

2.一个变量可以有多个引用

3.但是引用一旦对应一个实体,就不能改变指向了

用引用的好处在于:

1、做参数

a、做输出型参数

输出型参数是指 形参改变会影响实参 就像swap函数用引用类型做形参,会改变实参

b、减少拷贝,提高效率

2、做返回值

a、减少拷贝,提高效率

做返回值的时候,因为出作用域变量会销毁,所以会建立一个临时变量;但是当变量不销毁(在静态区)的时候,也会建立临时变量(会不会生成临时变量,看的并不是变量出了函数会不会销毁,而是返回值类型;如果是传值返回,都会进行拷贝生成临时变量,如何不生成临时变量?)可以用引用类型作为返回值不生成临时变量,可以减少拷贝,提高运行效率

如果引用作为返回值,不需要拷贝,但是如果变量销毁了,返回的结果就不能确定了

可能情况:a、函数调用结束,函数栈帧销毁,未清理函数栈帧,结果是对的

b、函数调用结束,函数栈帧销毁,清理了函数栈帧,结果是错的

所以说一定要保证出了函数,变量不会销毁(仍然存在),才可以返回引用

b、获取返回值 修改返回值

可以用引用实现SLAt()函数,同时实现SLGet()和SLModify()的功能,简化代码

引用权限

引用权限不可以放大,但是可以缩小/平移

cpp 复制代码
const int a;
int& b=a;
//权限放大了,改变b a也会改变 但是a本身是不能被改变的
int a=10;
int& b=a;
//权限平移了,可以这么写
int m=10;
int& n=m;
const int& p=m;
//权限缩小了。可以这么写
//不能通过p来改变m,但是m本身是可以修改的
m++;//可以这么写
n++;
//m和n变了之后p也变了 

临时变量创建场景

1】带返回值的普通函数,调用结束的时候会建立栈帧,建立临时变量

2】在发生类型转换的时候会创建临时变量相同类型不会产生临时变量)

cpp 复制代码
double a=1.0;
int b=a;//a发生类型转换,(截断或提升的时候)先创建临时变量

if(a>b)//比较的类型不同,会发生类型提升(一般是小的向大的提升)
{      //先生成临时变量
    swap(&a,&b);
}

引用与指针区别?

【1】从语法层面上看:

引用:不开空间,只是对变量取别名

指针:开空间,开辟空间存储变量地址

【2】从底层汇编指令角度来看,引用是类似于指针方式实现的

auto用法

可以自动推导变量的类型

cpp 复制代码
int a=1;
auto b=1+1.11;//可以根据右边表达式,自动推导出b是double类型的

cout<<typeid(b).name()<<endl;//输出变量的数据类型

以后学到迭代器之后,代码会很长,写起来比较麻烦,所以用aoto较为简便

注意:

1、auto不能做函数参数

cpp 复制代码
void Test(auto a)//这种写法是错误的
{
    ,,,
}

2、不能直接声明数组

cpp 复制代码
void test()
{
    auto b[]={1,2,3,4};//这也是错误的
}

范围for(语法糖)

cpp 复制代码
//自动推导类型,依次取出数组中元素,赋值给e,自动迭代和判断结束 
int arr[10]={1,2,3,4,5,6,7,8,9,0};

for(auto e:arr)
{
    cout<<e<<" ";
}
相关推荐
霁月风1 小时前
设计模式——适配器模式
c++·适配器模式
萧鼎1 小时前
Python并发编程库:Asyncio的异步编程实战
开发语言·数据库·python·异步
学地理的小胖砸1 小时前
【一些关于Python的信息和帮助】
开发语言·python
疯一样的码农1 小时前
Python 继承、多态、封装、抽象
开发语言·python
^velpro^1 小时前
数据库连接池的创建
java·开发语言·数据库
秋の花1 小时前
【JAVA基础】Java集合基础
java·开发语言·windows
jrrz08281 小时前
LeetCode 热题100(七)【链表】(1)
数据结构·c++·算法·leetcode·链表
小松学前端1 小时前
第六章 7.0 LinkList
java·开发语言·网络
可峰科技1 小时前
斗破QT编程入门系列之二:认识Qt:编写一个HelloWorld程序(四星斗师)
开发语言·qt
咖啡里的茶i1 小时前
Vehicle友元Date多态Sedan和Truck
c++