当前位置: 首页 > news >正文

c++基础语法

文章目录

  • 前言
  • 命名空间
    • 命名空间的使用
  • 缺省参数
    • 缺省参数的使用
  • 函数重载
    • 函数重载的作用
    • 函数重载的使用
    • 函数重载原理
  • 引用
    • 引用的使用
    • 引用的使用场景
    • 引用和指针
  • extern C
  • inline
  • auto
  • 范围for
  • nullptr

前言

大家好我是jiantaoyab,这篇文章给大家带来的是c语言没有的一些特性之一,是c++的基础语法,对后面的学习有帮助,后面我将逐步编写类和对象、STL容器等C++笔记,欢迎大家关注我!

命名空间

当程序达到一个规模后,会不可避免的使用到相同的函数名和标识符导致冲突,可以使用namespace封装到一个命名空间中,一个命名空间就定义了一个新的作用域,命名空间中的所有内容都局限于该命名空间中

同一个工程中能存在相同的命名空间,相同名字的命名空间会合并到一起

作用域:变量的有效范围,从定义的起点开始,到定义变量之前最近的一对括号确定

命名空间的使用

//1.使用using使用命名空间成员之一
using jiantao::n;
//2.using namespace jiantao
//全部展开,失去封装的效果,不建议这样使用
namespace jiantao
{int n = 10;//可以嵌套使用namespace id{int b = 2;}
}int main()
{cout << n << endl; //using jiantao::n; 直接用cout << jiantao::n << endl; //一般的使用方法cout << jiantao::id::b << endl; return 0;
}

缺省参数

缺省参数是声明或定义函数时为函数参数指定的一个默认值,在调用该函数时,如果没用指定实参则采用该默认值

缺省参数的使用

注意点

  • 半缺省参数必须从右往左给,不能间隔给
  • 缺省参数不能在定义和声明中同时出现,建议在声明中写
  • 缺省值必须是全局变量或者是常量
void fun()
{cout << "fun()" << endl;
}//默认这里的参数会给值
void fun(int a ,int b=10,int c=20)
{cout << a << endl;cout << b << endl;cout << c << endl;}
int main()
{fun(1); fun(1,2);//传2的话b的值就是2
}

函数重载

函数重载是指在同一作用域内,声明几个功能类似的同名函数,这些同名函数的形参列表(参数个数 或 类型 或 顺序)必须不同,这组函数被称为重载函数

函数重载的作用

重载函数通常用来命名一组功能相似的函数,这样做减少了函数名的数量,避免了名字空间的污染,对于程序的可读性有很大的好处

函数重载的使用

int add(int a, int b)
{return a + b;
}
double add(double a, double b)
{return a + b;
}
int main()
{add(1, 2);add(1.1, 1.2); //2个add构成重载return 0;
}

函数重载原理

Linux下,我定义了2个函数名字相同的函数,在gcc编译的时候,报错

image-20240309141027391

什么原因?

回忆一下一个程序编译链接的过程,在汇编的时候,会生成.o文件,.o文件是没有函数的地址的(声明和定义分开写),那么在链接的时候,会根据函数的名字去找地址,而在c语言中,不存在函数名字的修饰,根据函数的名字去符号表中找,假设在符号表中有Add(0x33123)和Add(0x332323),这在符号表中都冲突了,人家链接也不知道链接哪个

image-20240309143324511

而在c++中,存在函数名修饰规则,不同平台下不一样,在linux下大概为作用域+返回类型+函数名+参数列表,就能找到的函数的地址能链接上了

image-20240309143924393

重载大概的流程图

image-20240309144941774

我的知识有限,大家可以看看这篇 c++函数重载 里面有更详细的介绍

引用

引用不是新定义一个变量,而是给已经存在的变量起了一个别名,引用的变量和原来的变量共用一块内存空间

引用的使用

1.引用的类型和引用的实体必须的同类型的int a=10; int &b=a;2.引用在定义的时候必须初始化int &c=10 //error 引用的必须是一个对象const int a=10; int &b=a;//error a为常量
3.一旦一个引用被初始化指向一个对象,它就不能变为另一个对象的引用但一个对象可以有多个引用int a = 0;int& b = a;int& b = c; //ERRORint& c = a; //RIGHT
4.引用权限只能缩小,不能放大int a=10;const int &b=a;//✔ 权限缩小const int d = 10;int & f = d; //ERROR,这是权限的放大
5 临时变量本来是右值,是常量不能修改的double b = 1.1;int& i = d; //ERROR,存在类型转换,类型转换有临时变量const int& i2 = d //可以 

引用的使用场景

1.引用做参数 :提高效率,形参的改变能影响实参
int swap(int &x,int &y)
{}
使用引用传参,如果不改变参数的值,建议使用const引用2.引用做返回值:提高效率,修改返回变量(临时变量本来是右值,是常量不能修改的)
int& Add(int a,int b)
{int c=a+b; return c;
}
int main()
{int ret=Add(1,2);cout<<ret<<endl;
}

引用做返回值有一个重要的点,引用的这个返回对象在这个函数栈帧销毁的时候,还存在着才能使用引用返回。

int& Sub(int x, int y)
{int ret = x - y;return ret;
}int& f()
{int *p=(int *)malloc(4);return *p;//是可以的,p指向的空间还在return p;//error 指针p已经给销毁
}
int main()
{int& a = Sub(2, 1);
}  

image-20240309153633784

虽然编译通过了,但是上面的代码会存在非法访问的问题,Sub的返回值原来是临时变量,这里用了引用返回,那相当于int & ret = tmp,在调用完Sub后会回去访问ret的空间,假如Sub栈帧给销毁了,a所取到的值就是随机值,这里我们并没有马上使用这块空间才没报错。

image-20240309161456127

引用和指针

  • 引用在定义的时候必须初始化,指针不初始化也不报错
  • 引用一个对象之后就不能再引用别人对象,指针可以随意更改对象
  • 引用的大小是引用实体的大小,指针的大小是固定是4/8个字节
  • 引用不会开辟空间不占用内存
  • 引用加1是引用的实体加1,指针加1是向后偏移一个类型的大小

通过下面的汇编代码看出,引用和指针在使用的汇编指令是一样的

image-20240309164516106

extern C

大家有没有想过,如果我用c++写的程序,想去调用别人c写的库怎么去调用呢?

一会再回答这个问题,我们先来看看extern的用法。

extern是C语言中的一个关键字,一般用在变量名前或函数名前,作用是用来说明此变量/函数是在别处定义的,要在此处引用,在遇到此变量或函数的时候在其他文件中寻找其定义

extern int a;//声明变量a是在别的文件定义的,只是声明,没有分配内存
举个例子:   
比如我在test.cpp中 extern int num; extren void Print();
然后在fun.cpp 中  
int num = 1; //这里是定义变量  
void Print(){printf("Print()");} //这里是定义函数
这样子就能在test.cpp中使用这个num变量了 和 Print 这个函数了

想要在C++中调用c语言写的库,在vs2013下,需要执行下面操作

  1. 调试->属性->链接器 常规 ->附加库目录把库加上(Debug)

  2. 调试->属性->链接器 输入-> 附加依赖项 手动加入库的名字 xxx.lib

extern "C"
{//把头文件所在的目录包含#include "../../DS/DS/Stack.h" 
}

告诉C++编译器,extern “C”{}里面的函数是C编译器编译的,链接的时候用C的函数名规则去找,就可以链接上, 定义为extern的变量也会在外面去找

在C中调用c++写的库,要在cpp库中修改

//如果有__cplusplus 这个宏就用 extern "C" 替换 EXTRERN_C
#ifdef __cplusplus
#define EXTRERN_C extern "C" // C++静态库extern "C"告诉编译器以下函数按C的函数名修饰规则去处理
#else
#define EXTRERN_C  //这里走的是else 就是EXTRERN_C 什么都没有
#endifEXTRERN_C void StackInit(ST* ps); //运行后相当于 void StackInit(ST* ps)
EXTRERN_C void StackDestroy(ST* ps);
EXTRERN_C void StackPush(ST* ps, STDataType x);
EXTRERN_C void StackPop(ST* ps);
EXTRERN_C STDataType StackTop(ST* ps);
EXTRERN_C int StackSize(ST* ps);
EXTRERN_C bool StackEmpty(ST* ps);

inline

以inline修饰的函数叫做内联函数,编译时C++编译器会在调用内联函数的地方展开,没有函数压栈的开销(使用内联函数在.o文件符号表不会生成函数的地址),内联函数提升程序运行的效率。

一般调用函数在汇编语言中是用call来调用的

image-20240312203723202

当把Add函数设置成内联函数后,再看看汇编,想要显示出效果需要在vs2013Debug下设置

image-20240312203904395

image-20240312204121429

image-20240312204047665

可以看到直接以函数的本体代替就像宏一样,但是会增加目标码的大小,会导致额外的换页行为,降低指令高速缓存装置的击中率。还有要注意的是inline只是对编译器的一个申请,不一定会采用

还有inline不建议声明的定义分离,分离会导致链接错误在小型、频繁调用的函数上,可以使用inline

小练习

A.使用inline关键字的函数会被编译器在调用处展开
B.头文件中可以包含inline函数的声明
C.可以在同一个项目的不同源文件内定义函数名相同但实现不同的inline函数

A.不一定,因为inline只是一种建议,需要看此函数是否能够成为内联函数

B. inline函数不支持声明和定义分离开,因为编译器一旦将一个函数作为内联函数处理,就会在调用位置展开,即该函数是没有地址的,也不能在其他源文件中调用,故一般都是直接在源文件中定义内联函数的

C.inline函数会在调用的地方展开,所以符号表中不会有inline函数的符号名,不存在链接冲突。

auto

自动推导类型:类型声明时的“占位符”,编译器在编译期会将auto替换为变量实际的类型 ,auto会忽略顶层的const,而底层const被保留(顶层const是指针本身是个 常量,底层const是指针指向的对象是一个常量)

int main()
{const int a = 10;auto b = a; // int cout << typeid(b).name() << endl;//不管原来对象是不是const 要想成为const的auto 要在 auto前加上const //例如 const auto b =a; 这样定义出来的b就是带const属性的int x = 1;auto a = x; //intauto&b = x; //int&auto*c = &x;//int *return 0;
}

auto不能使用的场景

1.不能作为函数的参数
void Add(auto a) //error 编译器无法对a的实际类型进行推导
auto b //error
2.不能用来声明数组
auto arr[]={1,2,3}; //error    

范围for

int main()
{int arr[] = { 1, 2, 3, 4, 5, 6, 7 };//范围for这个地方必须是数组名for (auto e: arr) //一个一个取arr的值放到e中{cout <<" "<< e; }cout << endl;//每个值加1for (auto&e : arr){e++;}cout << endl;for (auto e : arr){cout << " " << e;}cout << endl;}void printf_(int arr[]) for(auto e:arr)//error 数组传参的时候会退化为指针,不再是数组名字{cout<<e<<endl;}
}

nullptr

c++中用nullptr中比null好

void f(int)
{cout << "f(int)" << endl;
}
void f(int *)
{cout << "f(int *)" << endl;
}
int main()
{f(NULL);// NULL = 0 调用了f(int),本意是调用f(int*)f(nullptr);//调用f(int *)
}

相关文章:

c++基础语法

文章目录 前言命名空间命名空间的使用 缺省参数缺省参数的使用 函数重载函数重载的作用函数重载的使用函数重载原理 引用引用的使用引用的使用场景引用和指针 extern Cinlineauto范围fornullptr 前言 大家好我是jiantaoyab&#xff0c;这篇文章给大家带来的是c语言没有的一些特…...

【工作实践-07】uniapp关于单位rpx坑

问题&#xff1a;在浏览器页面退出登录按钮上“退出登录”字样消失&#xff0c;而在手机端页面正常;通过查看浏览器页面的HTML代码&#xff0c;发现有“退出登录”这几个字&#xff0c;只不过由于样式问题&#xff0c;这几个字被挤到看不见了。 样式代码中有一行为&#xff1a…...

服务层组件

目录 连接层(Connection Pool) SQL接口(SQL Interface) 查询缓存(Caches&Buffers) Management Services&Utilities 查询分析器(Parser) 优化器(Optimizer)...

【学习笔记】VMware vSphere 6.7虚拟化入门

VMware vSphere 6.7虚拟化入门课程介绍 课程内容 1、VMware vSphere 6.7虚拟化入门课程介绍 2、ESXi6.7控制台设置 3、使用vSpkere Host client管理虚拟机 4、VMware EsXi基础操作 5、VMware Esxi存储管理 6、管理ESXi主机网络与虚拟机网络 7、安装配置vCenter Server Applia…...

如何防范企业内部安全威胁?

1 用户行为分析&#xff08;UEBA&#xff09; 现代化的用户行为分析产品具有多种优势功能&#xff0c;使企业能够有效地检测内部威胁。用户行为分析软件通过收集和分析来自各种来源的数据来分析和检测内部人员的可疑行为。这些来源包括网络日志和用户活动日志。通过检查这些数…...

内网渗透-跨域环境渗透-1

目录 smbclient工具 mimikatz工具 Kerbers协议 NTLM认证 hash传递攻击&#xff08;PTH攻击&#xff09; 黄金票据攻击 白银票据 MS14-068 smbclient工具 在linux里面连接远程windows共享目录&#xff0c;可以使用这个工具 ​ 第一种连接方式&#xff1a;smbclient -L 目…...

安信可IDE(AiThinker_IDE)编译ESP8266工程方法

0 工具准备 AiThinker_IDE.exe ESP8266工程源码 1 安信可IDE&#xff08;AiThinker_IDE&#xff09;编译ESP8266工程方法 1.1 解压ESP8266工程文件夹 我们这里使用的是NON-OS_SDK&#xff0c;将NON-OS_SDK中的1_UART文件夹解压到工作目录即可 我这里解压到了桌面&#xff0c…...

【java数据结构】HashMap和HashSet

目录 一.认识哈希表&#xff1a; 1.1什么是哈希表&#xff1f; 1.2哈希表的表示&#xff1a; 1.3常见哈希函数&#xff1a; 二.认识HashMap和HashSet: 2.1关于Map.Entry的说明:,> 2.2Map常用方法说明&#xff1a; 2.3HashMap的使用案例&#xff1a; 2.4Set常见方法…...

基于Springboot的高校汉服租赁网站(有报告)。Javaee项目,springboot项目。

演示视频&#xff1a; 基于Springboot的高校汉服租赁网站&#xff08;有报告&#xff09;。Javaee项目&#xff0c;springboot项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构…...

分布式解决方案

目录 1. 分布式ID1-1. 传统方案1-2. 分布式ID特点1-3. 实现方案1-4. 开源组件 2. 分布式Session2-1. 传统Session2-2. Spring-Session2-3. Token Redis2-4. JWT2-5. 拦截器统一处理Token2-6. Oauth2 3. 分布式锁3-1. redis3-2. Zookeeper 1. 分布式ID 1-1. 传统方案 时间戳U…...

力扣刷题日记——L724. 寻找数组的中心下标

1. 前言 今天是力扣刷题日记的第二天&#xff0c;今天依旧是一道简单题啊&#xff0c;慢慢来&#xff0c;先看看题目是什么吧。 2. 题目描述 给你一个整数数组 nums &#xff0c;请计算数组的 中心下标。 数组 中心下标 是数组的一个下标&#xff0c;其左侧所有元素相加的和…...

【Kotlin】类和对象

1 前言 Kotlin 是面向对象编程语言&#xff0c;与 Java 语言类似&#xff0c;都有类、对象、属性、构造函数、成员函数&#xff0c;都有封装、继承、多态三大特性&#xff0c;不同点如下。 Java 有静态&#xff08;static&#xff09;代码块&#xff0c;Kotlin 没有&#xff1…...

Docker完整版(一)

Docker完整版&#xff08;一&#xff09; 一、Docker概述1.1、Docker简介1.2、Docker的用途1.3、容器与虚拟机的区别1.4、Docker系统架构1.5、Docker仓库 二、Docker引擎2.1、Docker引擎架构2.2、Docker引擎分类2.3、Docker引擎的安装2.4、Docker镜像加速器 三、Docker镜像3.1、…...

AIOPS:Zabbix结合讯飞星火做自动化告警+邮件通知并基于人工智能提供解决方案

目前Zabbix官方已经提供Zabbix+ChatGPT的解决方案 ChatGPT一周年,你充分利用了吗?Zabbix+ChatGPT,轻松化解告警! 但是由于需要魔法等其他因素,比较不稳定,遂决定使用国内模型,这里我挑选的是讯飞星火,基于我之前的文档,在此基础上通过Zabbix的告警脚本实现调用AI模型…...

AHU 汇编 实验六

一、实验名称&#xff1a;实验6 输入一个16进制数&#xff0c;把它转换为10进制数输出 实验目的&#xff1a; 培养汇编中设计子程序的能力 实验过程&#xff1a; 源代码&#xff1a; data segmentbuff1 db Please input a number(H):$buff2 db 30,?,30 dup(?),13,10buff3 …...

Linux的输出、输入重定向和管道

目录 输出重定向 输入重定向 < << 管道操作 输出重定向 当我输⼊⼀个命令之后&#xff0c;回⻋&#xff0c;命令产⽣了结果&#xff0c;结果默认是输出到屏幕上的。 默认情况&#xff0c;⽆论⼀个命令执⾏正确与否&#xff0c;结果都会默认输出到屏幕上。 在有…...

java-新手笔记(枚举)

枚举&#xff08;Enumeration&#xff09;是一种特殊的类&#xff0c;用于表示固定数量的常量值。 枚举类型使得代码更加清晰&#xff0c;易于维护&#xff0c;同时也增加了类型安全。 这边使用一个枚举封装重要数据 enum Day {SUNDAY,MONDAY,TUESDAY,WEDNESDAY,THURSDAY,FR…...

Centos7 安装postgresql14后无法连接数据库

1、数据库服务器允许外部访问5432端口。 2、postgresql.conf 3、pg_hba.conf a、制定某个IP&#xff08;192.168.0.107&#xff09;访问 b、指定ip段访问 允许10.1.1.0~10.1.1.255网段登录数据库 host all all 10.1.1.0/24 trust c、指定全网访问 host a…...

GaussDB(DWS)运维利刃:TopSQL工具解析

在生产环境中&#xff0c;难免会面临查询语句出现异常中断、阻塞时间长等突发问题&#xff0c;如果没能及时记录信息&#xff0c;事后就需要投入更多的人力及时间成本进行问题的定位和解决&#xff0c;有时还无法定位到错误出现的地方。在本期《GaussDB(DWS)运维利刃&#xff1…...

信息安全、网络安全以及数据安全三者之间的区别

随着信息技术的飞速发展&#xff0c;网络安全、信息安全、数据安全等词汇在平时出现的频率越来越高&#xff0c;尤其是数据安全&#xff0c;是大家都关心的一个重要话题。事实上&#xff0c;有很多人对网络安全、信息安全、数据安全的概念是区分不清的&#xff0c;下面由我帮大…...

RestClient

什么是RestClient RestClient 是 Elasticsearch 官方提供的 Java 低级 REST 客户端&#xff0c;它允许HTTP与Elasticsearch 集群通信&#xff0c;而无需处理 JSON 序列化/反序列化等底层细节。它是 Elasticsearch Java API 客户端的基础。 RestClient 主要特点 轻量级&#xff…...

【快手拥抱开源】通过快手团队开源的 KwaiCoder-AutoThink-preview 解锁大语言模型的潜力

引言&#xff1a; 在人工智能快速发展的浪潮中&#xff0c;快手Kwaipilot团队推出的 KwaiCoder-AutoThink-preview 具有里程碑意义——这是首个公开的AutoThink大语言模型&#xff08;LLM&#xff09;。该模型代表着该领域的重大突破&#xff0c;通过独特方式融合思考与非思考…...

【单片机期末】单片机系统设计

主要内容&#xff1a;系统状态机&#xff0c;系统时基&#xff0c;系统需求分析&#xff0c;系统构建&#xff0c;系统状态流图 一、题目要求 二、绘制系统状态流图 题目&#xff1a;根据上述描述绘制系统状态流图&#xff0c;注明状态转移条件及方向。 三、利用定时器产生时…...

Copilot for Xcode (iOS的 AI辅助编程)

Copilot for Xcode 简介Copilot下载与安装 体验环境要求下载最新的安装包安装登录系统权限设置 AI辅助编程生成注释代码补全简单需求代码生成辅助编程行间代码生成注释联想 代码生成 总结 简介 尝试使用了Copilot&#xff0c;它能根据上下文补全代码&#xff0c;快速生成常用…...

高保真组件库:开关

一:制作关状态 拖入一个矩形作为关闭的底色:44 x 22,填充灰色CCCCCC,圆角23,边框宽度0,文本为”关“,右对齐,边距2,2,6,2,文本颜色白色FFFFFF。 拖拽一个椭圆,尺寸18 x 18,边框为0。3. 全选转为动态面板状态1命名为”关“。 二:制作开状态 复制关状态并命名为”开…...

20250609在荣品的PRO-RK3566开发板的Android13下解决串口可以执行命令但是脚本执行命令异常的问题

20250609在荣品的PRO-RK3566开发板的Android13下解决串口可以执行命令但是脚本执行命令异常的问题 2025/6/9 20:54 缘起&#xff0c;为了跨网段推流&#xff0c;千辛万苦配置好了网络参数。 但是命令iptables -t filter -F tetherctrl_FORWARD可以在调试串口/DEBUG口正确执行。…...

LeetCode 0386.字典序排数:细心总结条件

【LetMeFly】386.字典序排数&#xff1a;细心总结条件 力扣题目链接&#xff1a;https://leetcode.cn/problems/lexicographical-numbers/ 给你一个整数 n &#xff0c;按字典序返回范围 [1, n] 内所有整数。 你必须设计一个时间复杂度为 O(n) 且使用 O(1) 额外空间的算法。…...

第14节 Node.js 全局对象

JavaScript 中有一个特殊的对象&#xff0c;称为全局对象&#xff08;Global Object&#xff09;&#xff0c;它及其所有属性都可以在程序的任何地方访问&#xff0c;即全局变量。 在浏览器 JavaScript 中&#xff0c;通常 window 是全局对象&#xff0c; 而 Node.js 中的全局…...

代理服务器-LVS的3种模式与调度算法

作者介绍&#xff1a;简历上没有一个精通的运维工程师。请点击上方的蓝色《运维小路》关注我&#xff0c;下面的思维导图也是预计更新的内容和当前进度(不定时更新)。 我们上一章介绍了Web服务器&#xff0c;其中以Nginx为主&#xff0c;本章我们来讲解几个代理软件&#xff1a…...

python打卡day47

昨天代码中注意力热图的部分顺移至今天 知识点回顾&#xff1a; 热力图 作业&#xff1a;对比不同卷积层热图可视化的结果 import torch import torch.nn as nn import torch.optim as optim from torchvision import datasets, transforms from torch.utils.data import D…...