C/C++语言基础--从C到C++的不同(下),15个部分说明C与C++的不同

本专栏目的

  • 更新C/C++的基础语法,包括C++的一些新特性

前言

  • 1-10在上篇C/C++语言基础–从C到C++的不同(上);
  • 当然C和C++的不同还有很多,本人暂时只总结这些,其他的慢慢更新;
  • 上一篇C/C++语言基础–从C到C++的不同(上)受到了不少人的喜欢,因此,本人抓紧赶出下篇,欢迎大家点赞 + 收藏 + 关注;
  • C语言后面也会继续更新知识点,如内联汇编;
  • 本人现在正在写一个C语言的图书管理系统,1000多行代码,包含之前所学的所有知识点,包括链表和顺序表等数据结构,请大家耐心等待!!

思维导图如下
在这里插入图片描述

文章目录

    • 11. 枚举类型
      • 1,C语言中的enum
      • 2,C++中的enum
      • 3,C++中的 enum class 强枚举类型
      • 小结
    • 12. auto自动类型推导
    • 13. for循环遍历
    • 14. 类型信息
    • 15. 函数
      • 内联函数
      • 函数默认参数
      • 占位参数
      • 函数重载
        • **函数重载的不同**
        • **重载函数的调用匹配规则**
      • 函数重载遇上默认参数

11. 枚举类型

C语言和C++语言都提供了枚举类型,两者是有一定区别。

有如下定义:

enum SHAPE {CIRCLE,RECT,LINE,POINT};
enum WEEK  {MON,TUE,WED,THI,FIR,SAT,SUN};

1,C语言中的enum

  • 允许非枚举值赋值给枚举类型,允许其他枚举类型的值赋值给另一个枚举类型

    enum WEEK today = 3;	//正确
    today = CIRCLE;			//正确
    
  • 枚举具有外层作用域,容易造成名字冲突(在不同作用域不会冲突,但是遵循就近原则,访问不到外层作用域的枚举)

enum OTHER { RECT };//error C2365: “RECT”: 重定义;以前的定义是“枚举数”
int RECT = 12;		//同上
  • 不同类型的枚举值可以直接比较
if (CIRCLE == MON)
{printf("oh.yes");
}

2,C++中的enum

  • 只允许赋值枚举值
enum WEEK today = 3;	//错误	error C2440: “初始化”: 无法从“int”转换为“main::WEEK”
today = CIRCLE;			//错误	error C2440: “=”: 无法从“main::SHAPE”转换为“main::WEEK”
  • 枚举元素会暴露在外部作用域,不同两个枚举类型,若含有相同枚举元素,则会冲突
enum OTHER { RECT };	//错误 	error C2365: “RECT”: 重定义;以前的定义是“枚举数”
int RECT = 12;			//错误同上	但是可以通过枚举名访问指定的枚举属性
OTHER::RECT;			//正确
  • 不同类型的枚举也可以直接比较
if (CIRCLE == MON)
{cout<<"oh.yes";
}

3,C++中的 enum class 强枚举类型

enum class SHAPE {CIRCLE,RECT,LINE,POINT};
enum class WEEK  {MON,TUE,WED,THI,FIR,SAT,SUN};
  • 强枚举类型不会将枚举元素暴露在外部作用域,必须通过枚举名去访问
cout<<SHAPCE::RECT<<endl;	//输出 1 
  • 不相关的两个枚举类型不能直接比较,编译报错
if (SHAPE::CIRCLE == WEEK::MON)	//error C2676: 二进制“==”:“main::SHAPE”不定义该运算符或到预定义运算符可接收的类型的转换
{cout<<"oh.yes";
}

小结

  • C语言中,枚举可以暴露在外面,且可以枚举与非枚举之间进行比较,赋值等操作;
  • C++中,枚举可以暴露外外面,但是不可以枚举与非枚举之间进行赋值,但是可以比较;
  • C++强枚举中,不可以暴露在外面,必须指明作用域才能使用,枚举与非枚举之间不能进行比较,赋值,就算是枚举与枚举之间,不相关的枚举之间也不行

12. auto自动类型推导

在 C++11 之前的版本中,定义变量或者声明变量之前都必须指明它的类型,比如 int、char 等,这个其实很不方便,尤其是在复杂项目的时候,有多个命名空间,如:Boost::asio::ip::tcp等等,而在其他语言,如js,就只需要用关键字var或者let定义变量就行,到C++11后,auto诞生了,可以在编译的时候制动推导替换类型。

注意:auto 仅仅是一个占位符,在编译器期间它会被真正的类型所替代。或者说,C++ 中的变量必须是有明确类型的,只是这个类型是由编译器自己推导出来的。

  • 使用 auto 类型推导的变量必须马上初始化
  • auto 不能在函数的参数中使用(但是能作为函数的返回值)
  • auto 不能作用于类的非静态成员变量(也就是没有 static 关键字修饰的成员变量)中
  • auto 关键字不能定义数组
  • auto 不能作用于模板参数

13. for循环遍历

对于一个有范围的集合而言,由程序员来说明循环的范围是多余的,有时候还会容易犯错误,因此C++大叔给我们提供了一个简单的方法,如下:

int arr[]={1,2,3,4,5,6,7};
//一般用法
for(int i=0;i<sizeof(arr)/sizeof(arr[0]);i++)
{cout<<arr[i]<<" ";
}
//新用法
for(int i:arr)
{cout<<i<<" ";
}

特点:

  • 从数组的第一个元素开始,逐个赋值给迭代变量;
  • 不依赖于下标元素,通用;
  • 但是,经常还是喜欢用第一种,哈哈哈哈哈哈哈。

14. 类型信息

typeid 是一个运算符,用来获取一个表达式的类型信息。

typeid 的操作对象既可以是表达式,也可以是数据类型,下面是它的两种使用方法:

typeid( dataType )
typeid( expression )

typeid 会把获取到的类型信息保存到一个 type_info 类型的对象里面,并返回该对象的常引用;当需要具体的类型信息时,可以通过成员函数来提取,本质是一个类,封装了大量的API,如下:

//获取一个普通变量的类型信息
int n = 100;
const type_info& nInfo = typeid(n);
cout << nInfo.name() << " | " << nInfo.raw_name() << " | " << nInfo.hash_code() << endl;//获取一个字面量的类型信息
const type_info& dInfo = typeid(25.65);
cout << dInfo.name() << " | " << dInfo.raw_name() << " | " << dInfo.hash_code() << endl;//获取一个普通类型的类型信息
const type_info& charInfo = typeid(char);
cout << charInfo.name() << " | " << charInfo.raw_name() << " | " << charInfo.hash_code() << endl;//获取一个表达式的类型信息
const type_info& expInfo = typeid(20 * 45 / 4.5);
cout << expInfo.name() << " | " << expInfo.raw_name() << " | " << expInfo.hash_code() << endl;

其中:

  • name() 用来返回类型的名称。
  • raw_name() 用来返回名字编码(Name Mangling)算法产生的新名称。。
  • hash_code() 用来返回当前类型对应的 hash 值。

除此之外,还可以用 == 比较两个类型是否相等,如下:

char *str;
int a = 2;
int b = 10;
float f;

类型判断结果为:

类型比较结果类型比较结果
typeid(int) == typeid(int)truetypeid(int) == typeid(char)false
typeid(char*) == typeid(char)falsetypeid(str) == typeid(char*)true
typeid(a) == typeid(int)truetypeid(b) == typeid(int)true
typeid(a) == typeid(a)truetypeid(a) == typeid(b)true
typeid(a) == typeid(f)falsetypeid(a/b) == typeid(int)true

建议:内容很多,这个其实也并不常用,可以留个映像,到后面用到的时候查询即可。

15. 函数

内联函数

函数调用时,通过哈希表,需要跳转到函数的地址去执行,执行完成后返回到被调用函数,比较费时,因此,C++中提供了一种操作方式,允许编译时直接把函数替换到调用处,即内联函数

// 使用:
inline int add(int a, int b) {return a + b;
}

为什么使用内联函数?
内联函数没有普通函数调用时的额外开销(压栈,跳转,返回)

注意:

  • 内联函数声明时inline关键字必须和函数定义结合在一起,否则编译器会直接忽略内联请求。
  • C++编译器不一定准许函数的内联请求!(只是对编译器的请求,因此编译器可以拒绝)
  • 现代C++编译器能够进行编译优化,因此一些函数即使没有inline声明,也可能被编译器内联编译
    C++中内联函数的限制:
    • 不能存在任何形式的循环语句
    • 不能存在过多的条件判断语句
    • 函数体不能过于庞大
    • 不能对函数进行取址操作
    • 编译器对于内联函数的限制并不是绝对的,内联函数相对于普通函数的优势只是省去了函数调用时压栈,跳转和返回的开销。因此,当函数体的执行开销远大于压栈,跳转和返回所用的开销时,那么内联将无意义。

总结:上面内容过多,很难记住,我也记不住,我只记住了:内联函数可以直接跳转到函数地址中执行,省去入栈出栈等开销,还有就是内联函数之间不能写太多东西

函数默认参数

定义函数时可以给形参指定一个默认的值,这样调用函数时如果没有给这个形参赋值(没有对应的实参),那么就使用这个默认的值。

void showX(int x = 666)
{cout<<"x:"<<x<<endl;
}
showX();
showX(6);

小结:

  • 有函数声明时,默认参数可以放在声明或定义中,但不能同时存在
int add(int a,int b = 5);
int add(int a,int b)
{return a+b;
}
  • 注意:函数声明时,必须按照从右向左的顺序,依次给与默认值。
int foo(int a, int b = 2, int c = 3);     // 正确
int foo1(int a, int b = 2, int c);         // 错误, i3未指定默认值
int foo2(int a = 1, int b, int c = 3);     // 错误, i2未指定默认值

占位参数

定义函数时,还可以给函数提供占位参数

  • 占位参数只有参数类型,而没有参数名
  • 在函数体内部无法使用占位参数
  • 占位参数也可以指定默认参数
void func(int a,int = 0)
{cout<<a<<endl;
}
func(2);

函数重载

不知道在C语言写函数的时候会不会遇到这种情况,想函数名难想(不想直接拼音),而如果想实现以下功能:

int add(int a, int b) {return a + b;
}int add(double a, double b) {return a + b;
}

想实现以上功能的时候,好像名字都可以用add,故在C++的时候,引入了一个新的概念,函数重载。(当然上面的实现还可以更加简介,他可用模板,这个后面会更新)。

函数重载是指:在同一作用域内,可以有一组具有相同函数名不同参数列表的函数,这组函数被称为重载函数。

函数重载的不同
  • 参数个数不同
  • 参数类型不同
  • 参数顺序不同
  • 函数重载与返回值类型无关

来个例子体会一下,比较不同类型的两个变量的大小

int maxmum(int a, int b)
{return a > b?a:b;
}
long maxmum(long int a, long int b)
{return a > b ? a : b;
}
char maxmum(char a, char b)
{return a > b ? a : b;
}
double maxmum(double a, double b)
{return a > b ? a : b;
}
const char* maxmum(const char* str1,const char* str2)
{return strcmp(str1, str2)==1?str1:str2;
}
char* maxmum(char* str1, char* str2)
{return strcmp(str1, str2) == 1 ? str1 : str2;
}int main()
{cout << maxmum(2, 6) << endl;cout << maxmum(2L, 6L) << endl;cout << maxmum('A', 'C') << endl;cout << maxmum("maye", "MAYE") << endl;char str1[] = "hello";char str2[] = "hello";cout << maxmum(str1, str2) << endl;return 0;
}

函数重载可以根据具体的参数去决定调用哪一个函数。

重载函数的调用匹配规则
  • 精确匹配:参数匹配而不做转换,或者只是做微不足道的转换,如数组名到指针、函数名到指向函数的指针;
  • 提升匹配:即整数提升(如bool 到 int、char到int、short 到int),float到double
  • 使用标准转换匹配:如int 到double、double到int、double到long double、Derived到Base、T到void、int到unsigned int;

函数重载遇上默认参数

如果在给重载函数的时候指定默认参数时,这个时候很容易造成函数冲突,如下:

void fun(int a)
{cout << "fun(int a) " << a << endl;
}
void fun(int a, int b = 8)
{cout << "fun(int,int =8) " << a <<" "<< b << endl;
}
int main()
{//fun(5);	//error C2668: “fun”: 对重载函数的调用不明确fun(5, 6);//正确return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

node.js 中的进程和线程工作原理

本文所有的代码均基于 node.js 14 LTS 版本分析 概念 进程是对正在运行中的程序的一个抽象&#xff0c;是系统进行资源分配和调度的基本单位&#xff0c;操作系统的其他所有内容都是围绕着进程展开的 线程是操作系统能够进行运算调度的最小单位&#xff0c;其是进程中的一个执…

康养小站:长者舒缓疼痛的港湾

【导语】在老龄化日益加剧的当下&#xff0c;如何关爱和照顾好长者&#xff0c;成为社会关注的焦点。近日&#xff0c;笔者走进深圳宝安区一家专注于长者康养的社区小站&#xff0c;探访它如何帮助长者缓解疼痛&#xff0c;提高生活质量。 随着我国人口老龄化问题日益显著&…

算法:30.串联所有单词的子串

题目 链接&#xff1a;leetcode链接 思路分析&#xff08;滑动窗口&#xff09; 这道题目类似寻找异位词的题目&#xff0c;我认为是寻找异位词的升级版 传送门:寻找异位词 为什么说像呢&#xff1f; 注意&#xff1a;这道题目中words数组里面的字符串长度都是相同的&…

[JAVA]介绍怎样在Java中通过字节字符流实现文件读取与写入

一&#xff0c;初识File类及其常用方法 File类是java.io包下代表与平台无关的文件和目录&#xff0c;程序中操作文件和目录&#xff0c;都可以通过File类来完成。 通过这个File对象&#xff0c;可以进行一系列与文件相关的操作&#xff0c;比如判断文件是否存在&#xff0c;获…

Java毕业设计 基于SpringBoot和Vue药店管理系统

Java毕业设计 基于SpringBoot和Vue药店管理系统 这篇博文将介绍一个基于SpringBoot框架和Vue开发的药店管理系统&#xff0c;适合用于Java毕业设计。 功能介绍 首页 图片轮播 登录 注册 药品信息 药品详情 评论 收藏 购买 添加到购物车 用药指南 公告资讯 购物车 …

在深圳停车场我居然能看到很漂亮的瓦房

石岩街道在宝安确实是小透明哈&#xff0c;从市区搬到石岩快4年了&#xff0c;确实这里的建筑特别像老家的感觉&#xff0c;马路很狭窄。如果是开车的话&#xff0c;我是不会进入罗租大道来着&#xff0c;人车太复杂。由于上屋社区适合儿童的室内场所太少了&#xff0c;石岩这块…

python之模块和包的导入与使用,pip的使用(13)

文章目录 1、模块1.1 模块的分类1.1.1 内置模块1.1.2 第三方模块&#xff08;比较重要&#xff09;1.1.3 自定义模块 1.2 模块的导入1.2.1 单个模块的导入1.2.2 同时导入多个模块1.2.3 模块导入规范1.2.4 给导入的模块取别名1.2.5 同时导入模块和名字1.2.6 给导入的名字取别名扩…

【Python机器学习】序列到序列建模——使用序列到序列网络构建一个聊天机器人

为了寻聊天机器人&#xff0c;下面使用康奈尔电影对话语料库训练一个序列到序列的网络来“适当的”湖大问题或语句。以下聊天机器人示例采用的是Keras blog中的序列到序列的示例。 为训练准备语料库 首先&#xff0c;需要加载语料库并从中生成训练集&#xff0c;训练数据将决…

【刷题】Day5--数字在升序数组中出现的次数

Hi! 今日份刷题~ 数字在升序数组中出现的次数_牛客题霸_牛客网 我感觉题目简单&#xff0c;我的解答也很简单&#xff0c;二分法遗忘&#xff0c;有时间复习一下尝试新的解法。 /*** 代码中的类名、方法名、参数名已经指定&#xff0c;请勿修改&#xff0c;直接返回方法规定的…

electron-updater实现electron全量版本更新

在 Electron 应用中使用 electron-updater 来实现自动更新功能时&#xff0c;通常你会在一个专门的模块或文件中管理更新逻辑。如果你想要使用 ES6 的 import 语法来引入 electron-updater&#xff0c;你需要确保你的项目已经配置好了支持 ES6 模块的构建工具&#xff08;如 We…

系统指标优化 Stream流 API使用及源码分析

java 8 Stream流 api 使用 &#xff08;1&#xff09;集合筛选&#xff0c;为什么此处要使用 Collectors.toMap 加 Function.identity() 的方式&#xff0c;而不使用简单的filter过滤呢&#xff1f; Stream流 API 代码: collect(Collectors.toMap(GoldTagResponse::getValue, …

C语言深入理解指针(一)

目录 内存和地址内存编址的理解 指针变量和地址取地址操作符&#xff08;&&#xff09;指针变量和解引用操作符&#xff08;*&#xff09;指针变量如何拆解指针类型解引用操作符 指针变量的大小 指针变量类型的意义指针的解引用指针-整数 const修饰指针const修饰变量const修…

多态的相关知识

一.多态的概念 1.多态&#xff1a;多态是⼀个继承关系的下的类对象&#xff0c;去调⽤同⼀函数&#xff0c;产⽣了不同的⾏为。 2.多态分为编译时多态(静态多态)和运⾏时多态(动态多态)。 1>编译时多态(静态多态)主要就是函数重载和函数模板&#xff0c;他们传不同类型的…

Datawhale------Tiny-universe学习笔记——Qwen

1. Qwen整体介绍 对于一个完全没接触过大模型的小白来说&#xff0c;猛一听这个名字首先会一懵&#xff1a;Qwen是啥。这里首先解答一下这个问题。下面是官网给出介绍&#xff1a;Qwen是阿里巴巴集团Qwen团队研发的大语言模型和大型多模态模型系列。其实随着大模型领域的发展&a…

庆祝中华人民共和国成立75周年答题活

为庆祝中华人民共和国成立75周年&#xff0c;弘扬爱国主义精神&#xff0c;激发广大党员干部和人民群众奋进新征程、建功新时代&#xff0c;奋力推进中国式现代化建设的爱国热情&#xff0c;“学习强国”学习平台采用“线上答题线下竞赛”的形式&#xff0c;举办“学习强国 强国…

STM32外设之LTDC/DMA2D—液晶显示(野火)

文章目录 显示屏有几种?基本参数控制?显存 LTDC 液晶控制器LTDC 结构框图LTDC 初始化结构体 LTDC_InitTypeDefLTDC 层级初始化结构体 DMA2D 图形加速器DMA2D 初始化结构体 要了解什么 屏幕是什么&#xff0c;有几种屏&#xff0c;有什么组成。 怎么控制&#xff0c;不同屏幕控…

go多线程

1、简单使用&#xff08;这个执行完成&#xff0c;如果进程执行比较久&#xff0c;这里不会等待它们结束&#xff09; package mainimport "time"func main() {go func() {println("Hello, World!")}()time.Sleep(1 * time.Second) }2、wg.Add(数量)使用&…

借老系统重构我准备写个OpenAPI3.1版的API管理工具(附录屏演示)

前段时间一直在忙公司老系统重构的方案设计&#xff0c;其中最大的重构点就是前后端分离。为了加快前后端协同开发和对接的工作效率&#xff0c;我决定写一个公司内部使用的OpenAPI3.1版的API管理工具。 文章目录 有现成的工具为啥不用现有成熟方案初步成果展示录屏演示下一步计…

基于springboot酒店管理系统

酒店管理系统 552酒店管理系统 摘 要 21世纪的今天&#xff0c;随着社会的不断发展与进步&#xff0c;人们对于信息科学化的认识&#xff0c;已由低层次向高层次发展&#xff0c;由原来的感性认识向理性认识提高&#xff0c;管理工作的重要性已逐渐被人们所认识&#xff0c;…

医学数据分析实训 项目三 关联规则分析预备项目---购物车分析

文章目录 1 预备项目关联规则分析实践———购物车分析1 产生频繁集2 产生关联规则 1 预备项目 关联规则分析实践———购物车分析 import warnings import numpy as np import pandas as pd from mlxtend.frequent_patterns import apriori from mlxtend.frequent_patterns …