C++开始

简介: C++开始

C++开始

1. C++ 关键字

C++98里共63个,其中32个是C语言的。现在最新版本有68个。

2. 命名空间(namespace)

避免命名冲突和名字污染。在大型工程中变量/函数非常多,每一个模块都有大量的文件,大量的变量/函数/结构体/…,这时非常容易引起命名冲突。

引入命名空间就是解决命名冲突,不同命名空间中的变量可以一样。(docker中的隔离性!!) 一个命名空间就是一个新的作用域。

域作用限定符::: , A:: , B:::: (默认的是全局域)

#include <iostream>
namespace A
{
  int a = 10;
  int b = 20;
}
int main()
{
  int a = 1;
  int b = 2;
  // 这个作用域里的a和b
  std::cout << a + b << std::endl;
  // 命名空间A里的a和b
  std::cout << A::a + A::b << std::endl;
  return 0;
}
// 把 std(标准库) 里的变量全部放出来
using namespace std;
// 只放出来常用的
using std::cout;
using std::endl;

几点注意:

  1. 命名空间可以嵌套定义。
  2. 一个工程中有相同的命名空间时会合并。
  3. 命名空间只能在全局声明,不能在main函数里。
  4. 命名空间中的变量本质上还是全局的变量,只是使用命名空间进行隔离。

3. 输入输出

这两个运算符涉及到==运算符的重载。==正是因为重载,他们才可以自动识别类型。

// << 流插入
std::cout << "hello world" << std::endl;
// >> 流提取
int a, b;
cin >> a >> b;

4. 缺省参数

在函数形参那里可以指定一个缺省值,当实参那里没有传参数时就使用这个缺省参数,实参传了参数就用传过来的那个参数。

// 全缺省
void func(int a = 10, char ch = 's')
{
  // ...
}
// 半缺省
void func2(int a, int b, char ch = 's')
{
  // ...
}
int main()
{
  // 都可以
  func();
  func(1);
  func(1, 'a');
  func2(1, 2);
  func(1, 2, 'a');
}
  1. 可以全缺省,也可以半缺省(缺省一部分形参),但是半缺省时必须从最后一个参数开始顺序向前开始缺省!!就是从右向左依次给缺省值。(因为参数是从左往右依次传的)
  2. 当**一个函数声明和定义分开时,不能同时给缺省值!!**编译器不知道听谁的。这个时候应该是声明的时候写好缺省值,定义的时候不写缺省。(经过检验,在VS和Linux下同时给缺省是可以的,以声明的值为准!)
  3. 给定的缺省值只能是常量或全局变量。

5. 函数重载(重点)

在同一个作用域中可以定义多个同名的函数,但是函数的参数列表(参数个数/参数类型)必须不同,叫做函数重载。

C++支持重载的原因(面试题)

1. 程序的编译链接过程(预编译->编译->汇编->连接)
  1. 预处理(预编译):进行头文件展开,宏替换,条件编译,去掉注释…
  2. 编译:进行语法检查,语义分析,词法检查… 最终生成汇编代码。
  3. 汇编:形成符号表(函数名和对应的地址之间的映射),最终生成二进制文件。
  4. 连接:合并符号表,并进行符号表的重定位,就是将各个文件里只有函数声明,没有具体定义的那些函数,去符号表查找,找到其具体的地址!! 最终生成可执行文件(.o)。
2. C++支持重载具体原因

C++编译器(g++)在编译时会对函数名进行修改:

// 如下
Printf(int a, double b) -> _Z5Printid

具体修改规则(Linux下):_Z + 函数名长度(协议) + 函数名 + 所有参数类型缩写

这样就可以将重载的函数区分开。而C编译器(gcc)在编译时不会对函数名做修改,相同的函数名就无法区分,也就不能支持重载。

Linux下反汇编:
objdump -S[-c -t ....] filename > filename.txt
3. extern “C”

在实际中,可以使用C语言调用C++的动静态库(多个 .o 打包),也可以使用C++调用C语言的动静态库。但是:==C动静态库中编译后的函数名是没有经过更改的原始的函数名,而C++的动静态库中编译后的函数名是经过编译器修改后的函数名!==这和C/C++项目本身的编译规则是不一样的,这时就会导致链接错误----在链接时找不到符号表中对应的函数名!

这时就可以使用 extern "C" 来解决。

C++调用C

修改这个C++文件/项目

// 在C++文件中 加这个extern "C"后{}内的内容就会以C语言的规则编译(不修改函数名)
// 这样就可以避免C++编译时修改函数名而导致链接时在符号表中找不到对应的函数
extern "C"
{
  // 这个库里文件就会以C的方式编译,不修改函数名
  #include "/home/c_lib/.... .h"
}
C调用C++

C++的动静态库进行修改!!都是对C++的那一方进行修改。

利用条件编译!条件编译在跨平台方面非常有用!!

// 在C++库文件中(头文件)
#ifdef __cplusplus  // 在C++文件中默认有这个定义
#define EXTERN_C extern "C"
#else
#define EXTERN_C
#endif
// 这些函数声明在C++库文件中就会以C的方式编译,不会对函数名做修改!
// 而在C的项目中包含这个头文件,展开后 EXTERN_C 是空!!也不会有任何问题!
EXTERN_C 函数声明
EXTERN_C 函数声明
EXTERN_C 函数声明
......

==本质就是为了做到在C++中以C的方式编译,同时不影响C包含C++的头文件而出错!!==或者这样也可以:

// 本质就是为了做到在C++中以C的方式编译,同时不影响C包含C++的头文件而出错!!
#ifdef __cplusplus
extern "C"
{
#endif
  函数声明
  函数声明
  函数声明
  // .....
#ifdef __cplusplus 
}
#endif

这时C++库中就不能使用函数重载了!!因为以C的方式编译不修改函数名,不支持重载!

6. 引用

1. 基本概念

就是给一个变量起别名。常用作函数形参函数返回值减少参数拷贝带来的消耗(函数调用会将实参拷贝到形参)。

做函数返回值时要特别注意一下,如果返回的变量在出了函数作用域后就销毁了,那就会有类似野指针的问题!(引用的对象销毁了!)

函数返回值问题:函数的返回值并不是直接返回的,而是先将返回值拷贝到一个临时变量(一般临时变量有常属性)。当返回值比较小时就可以拷贝到寄存器中;当返回值很大时会在 调用函数 建立栈帧的时候就提前开辟好这个空间!然后出了函数作用域这个返会的变量可能被销毁,所以再将这个临时变量拷贝到接收返回值的变量中!!

这就导致中间多了一步拷贝,当返回值比较大时效率很低!传引用返回就可以提高效率,直接就返回这个变量的别名(就是这个变量)。但是要求函数返回之后这个变量还在,没有被销毁!

**常引用:**这个还是很常用的,可以接收变量,常量,隐式类型转换等类型。

void TestConstRef()
{
  const int a = 10;
  // int& ra = a; // 该语句编译时会出错,a为常量
  const int& ra = a; // 可以
  // int& b = 10; // 该语句编译时会出错,10为常量
  const int& b = 10;  // 可以
  double d = 12.34;
  // int& rd = d; // 该语句编译时会出错,类型不同
  // 这个可以!!但是 rd 已经不是d这个变量的引用了!而是会产生一个int类型的临时     变量,rd是那个临时变量的引用。
  const int& rd = d;  // 这个留意一下!!
}

引用的几点注意:

  1. 引用在定义的时候必须初始化。
  2. 一旦引用了某一个实体,就不能改变,一直作为他的引用!
  3. 引用在语法层面就是一个别名,没有独立的空间,但是底层还是指针!!
2. 指针和引用的区别:
  1. 没有NULL引用,但是有NULL指针。
  2. sizeof(引用)大小是所引用类型的大小,sizeof(指针)在32位平台下是4字节。
  3. 引用进行加减操作是对本体进行加减,指针进行加减操作是向后/前 偏移一个类型大小。

7. 内联函数(inline)

使用关键字inline修饰的函数叫做内联函数,在编译时会直接在调用内联函数的地方把函数体展开,就不需要建立栈帧,进行传统的函数调用了!!提高运行效率(空间换取时间)

  1. 适用于频繁调用的短小函数(非递归的)。
  2. inline也只是一个建议,具体编译器会不会采用是不确定的!
  3. inline函数不能声明和定义写到不同的文件中!会有链接错误:定义内联函数的文件在生成 .o 目标文件时在符号表里不会生成该函数的地址(因为是内联函数),这样在链接的时候就找不到该函数了。
inline Add(int a, int b)
{
  return a + b;
}

在C语言中使用宏来做到这一点。宏的缺点:

  1. 不支持调试。
  2. 没有类型安全检查。

8. auto关键字

自动进行类型推导。

1. auto不能推导的场景

  1. 函数形参的类型不能用auto (C++20 以后也可以了)
  2. 函数返回值不能是auto
  3. auto不能直接用来声明数组。
// 获取变量类型
int a = 10;
cout << typeid(a).name() << endl;

9. 范围 for

// 自动遍历
for(auto& e : arr)
  cout << e << endl;

10. nullptr

空指针。更推荐!

#ifndef NULL
#ifdef __cplusplus
#define NULL 0         // 在C++中,NULL就是0
#else
#define NULL ((void *)0)
#endif
#endif

NULL是一个宏,C++中NULL就是 0,是一个整型,这就可能会有问题。比如:

void f(int)
{
  cout<<"f(int)"<<endl;
}
void f(int*)
{
  cout<<"f(int*)"<<endl;
}
int main()
{
  f(0);               // f(int)
  f(NULL);            // f(int)  但是直观来看,这里应该是 f(int*)
  f((int*)NULL);      // f(int*)
  return 0;
}

nullptr 是一个关键字!使用更安全。

NULL是一个宏,C++中NULL就是 0,是一个整型,这就可能会有问题。比如:

void f(int)
{
  cout<<"f(int)"<<endl;
}
void f(int*)
{
  cout<<"f(int*)"<<endl;
}
int main()
{
  f(0);               // f(int)
  f(NULL);            // f(int)  但是直观来看,这里应该是 f(int*)
  f((int*)NULL);      // f(int*)
  return 0;
}

nullptr 是一个关键字!使用更安全。

最后挂个链接,欢迎一起学习,一起进步!https://xxetb.xet.tech/s/4G6TWG

相关文章
|
11天前
|
弹性计算 运维 自动驾驶
首个云超算国标正式发布!
近日,我国首个云超算国家标准GB/T 45400-2025正式发布,将于今年10月实施。该标准由阿里云联合多家机构起草,为云超算在高性能计算领域的应用提供规范。云超算结合传统HPC与云计算优势,解决传统HPC复杂、昂贵等问题。阿里云E-HPC V2.0是国内首批通过该标准认证的产品,支持大规模弹性计算,显著降低成本。新标准将推动算力基础设施迈向标准化、智能化新时代。
|
10天前
|
人工智能 Cloud Native Serverless
解决方案评测|告别复杂配置!基于阿里云云原生应用开发平台CAP快速部署Bolt.diy
本文介绍了基于阿里云CAP平台快速部署Bolt.diy的全流程。Bolt.new是一款强大的无代码对话网站构建工具,其开源版本Bolt.diy虽功能强大但部署复杂。阿里云的新解决方案简化了这一过程,通过函数计算(FC)和通义千问大模型能力,将数十步部署流程精简至“一键启动”。文章详细描述了从访问Web界面、提交请求到生成代码的步骤,并提供了系统提示词优化方案,以提升代码生成效率。此外,还对比了不同版本的Bolt.new,帮助开发者选择适合的方案。此方法适合希望快速落地项目的开发者,尤其在需要整合云服务时表现出色。
|
12天前
|
传感器 自然语言处理 监控
快速部署实现Bolt.diy
Bolt.diy 是 Bolt.new 的开源版本,提供灵活的自然语言交互与全栈开发支持。基于阿里云函数计算 FC 和百炼模型服务,最快5分钟完成部署。新手注册阿里云账号后可领取免费额度,按指引开通相关服务并授权。通过项目模板一键部署,配置 API-KEY 后即可使用。Bolt.diy 支持多种场景,如物联网原型开发、久坐提醒、语音控制灯光等,助力快速实现创意应用。
2278 22
|
11天前
|
人工智能 运维 数据可视化
阿里云百炼 MCP服务使用教程合集
阿里云百炼推出首个全生命周期MCP服务,支持一键部署、无需运维,具备高可用与低成本特点。该服务提供多类型供给、低成本托管及全链路工具兼容,帮助企业快速构建专属智能体。MCP(模型上下文协议)作为标准化开源协议,助力大模型与外部工具高效交互。教程涵盖简单部署、GitHub运营、数据分析可视化及文档自动化等场景,助您快速上手。欢迎加入阿里云百炼生态,共同推动AI技术发展!
阿里云百炼 MCP服务使用教程合集
|
10天前
|
人工智能 弹性计算 运维
阿里云 MCP Server 开箱即用!
本文介绍了如何通过alibaba-cloud-ops-mcp-server和MCP(Model Context Protocol)实现AI助手对阿里云资源的复杂任务操作。内容涵盖背景、准备步骤(如使用VS Code与Cline配置MCP Server)、示例场景(包括创建实例、监控实例、运行命令、启停实例等),以及支持的工具列表和参考文档。借助这些工具,用户可通过自然语言与AI助手交互,完成ECS实例管理、VPC查询、云监控数据获取等运维任务,实现高效“掌上运维”。
|
13天前
|
Serverless API
【MCP教程系列】在阿里云百炼,实现超级简单的MCP服务部署
阿里云百炼推出业界首个全生命周期MCP服务,支持一键在线注册托管。企业可将自研或外部MCP服务部署于阿里云百炼平台,借助FC函数计算能力,免去资源购买与服务部署的复杂流程,快速实现开发。创建MCP服务仅需四步,平台提供预置服务与自定义部署选项,如通过npx安装代码配置Flomo等服务。还可直接在控制台开通预置服务,体验高效便捷的企业级解决方案。
【MCP教程系列】在阿里云百炼,实现超级简单的MCP服务部署
|
6天前
|
人工智能 JavaScript Serverless
从零开始开发 MCP Server
本文介绍如何使用Serverless Devs CLI工具从零开发并一键部署MCP Server到阿里云函数计算(FC)。首先通过初始化MCP Server项目,完成本地代码编写,利用Node.js实现一个简单的Hello World工具。接着对代码进行打包,并通过Serverless Devs工具将项目部署至云端。部署完成后,提供三种客户端接入方式:官方Client、其他本地Client及在FC上部署的Client。最后可通过内置大模型的inspector测试部署效果。Serverless Devs简化了开发流程,提升了MCP Server的构建效率。
529 109
|
13天前
|
云安全 人工智能 安全
|
19天前
|
存储 人工智能 监控
一键部署 Dify + MCP Server,高效开发 AI 智能体应用
本文将着重介绍如何通过 SAE 快速搭建 Dify AI 研发平台,依托 Serverless 架构提供全托管、免运维的解决方案,高效开发 AI 智能体应用。
2383 64
|
6天前
|
人工智能 API
MCP协议的局限性
5年前,我把 AI 比喻为一种智能化的 API 网关,提出一种分治的思想,将一个大问题转换为若干可解的小问题,如今,这种思想正在 mcp 这种协议沿用。但目前来看,它的实现方式还是有点丑陋的,并且有一些问题。