【C++】C++入门详解 I【C++入门 这一篇文章就够了】

C++入门

  • 前言
  • 一、C++关键字(C++98)
  • 二、命名空间 namespace
    • (一)namespace的出现
    • (二)namespace的定义
      • (1)namespace 的正常定义
      • (2)namespace的功能特性
        • 1. 命名空间 可嵌套
        • 2. 同一工程中允许存在多个相同名称的命名空间,编译器最后会合成同一个命名空间中。
    • 命名空间的使用
      • (1)加 命名空间名称 及 `::` 作用域限定符 [ 每次指定命名空间 ]
      • ☆(2) using 将 命名空间中某个成员 引入 [ 部分展开 ]
      • (3) using namespace 命名空间名称 引入 [ 全展开 ]
  • 三、using namespace std
  • 四、C++ 输入&输出 [ IO流 ]
    • (一)代码实现
      • (1)包含头文件 `# include<iostream>`
      • (2)cout cin endl
      • (3)<< _ 流插入运算符,>> _ 流提取运算符
    • (二)C++输入输出 的特性:可 自动识别变量类型
    • (三)cout,cin更复杂的用法
    • (三).h 头文件 以及 std命名空间
  • 五、缺省参数
    • (一) 缺省参数 概念
    • (二) 缺省参数分类
  • 六、函数重载
    • (一) 函数重载概念
    • (二)函数重载
      • (1)参数类型不同
      • (2)参数个数不同
      • (3)参数类型顺序不同
  • (三)函数重载 底层原理_名字修饰(name Mangling)



前言

C++ 就是在对 C语言使用中遇到的缺陷与不足的改进。

C++是在C的基础之上,容纳进去了面向对象编程思想,并增加了许多有用的库,以及编程范式等。熟悉C语言之后,对C++学习有一定的帮助。

本章节主要目标:

  1. 补充C语言语法的不足,以及C++是如何对C语言设计不合理的地方进行优化的,比如:作用域方面、IO方面、函数方面、指针方面、宏方面等。

  2. 为后续类和对象学习打基础


一、C++关键字(C++98)

C++总计63个关键字(涵盖 C语言32个关键字)

在这里插入图片描述


二、命名空间 namespace

(一)namespace的出现

在C/C++中出现的问题:

  1. 变量、函数和后面要学到的类都是大量存在的,这些变量、函数和类的名称将都存在于全局作用域中,可能会导致很多冲突
  2. 写的变量名与库里面的名冲突
  3. 在项目异步进行时,最终项目合并时,也容易导致很多命名上的冲突(重命名)

如下面这个例子

#include <stdio.h>

#include <stdlib.h>        //包含rand()函数
int rand = 10;             //全局变量中也有 命名为rand的变量

// C语言没办法解决类似这样的命名冲突问题,所以C++提出了namespace来解决
int main()
{
 printf("%d\n", rand);
return 0;
}
// 编译后后报错:error C2365: “rand”: 重定义;以前的定义是“函数”
  • C++ namespace [ 关键字 ] 对此进行了优化:
    平时程序运行中 默认访问全局变量里的变量namespace相当于一堵围墙平时都会绕开,默认不会进入墙里找域访问 。只有 展开命名空间,才有权限进入命名空间内访问存放的变量。

使用 namespace 后

#include <stdio.h>

#include <stdlib.h>        //包含rand()函数

namespace nini{
int rand = 10;             
}
// C语言没办法解决类似这样的命名冲突问题,所以C++提出了namespace来解决
int main()
{
 printf("%d\n",nini::rand);   // :: 域作用限定符 
return 0;
}

使用命名空间的目的是对标识符的名称进行本地化,以避免命名冲突或名字污染
namespace 关键字的出现就是针对这种问题的。



(二)namespace的定义

(1)namespace 的正常定义

namespace[关键字] + 命名空间的名字 + { } +( { } 里面 )命名空间的成员

  1. 一般开发中是用 项目名字命名空间名
  2. 命名空间中可以定义 变量 / 函数 / 类型(如结构体等)

[ 注意:一个命名空间就定义了一个新的作用域,命名空间中的所有内容都局限于该命名空间中 ]

namespace nini
{
 // 命名空间中可以定义变量/函数/类型
 int rand = 10;                        //变量
 
 int Add(int left, int right)          //函数
 {
 return left + right;
 }

struct Node                            //类型
 {
 struct Node* next;
 int val;
 };
 
}


(2)namespace的功能特性

1. 命名空间 可嵌套
// test.cpp
namespace N1
{
int a;
int b;
int Add(int left, int right)
 {
     return left + right;
 }
 
namespace N2
 {
     int c;
     int d;
     int Sub(int left, int right)
     {
         return left - right;
     }
 }
 
}
2. 同一工程中允许存在多个相同名称的命名空间,编译器最后会合成同一个命名空间中。

ps:一个工程中的test.h和上面test.cpp中两个N1会被合并成一个

// test.h
namespace N1
{
int Mul(int left, int right)
 {
     return left * right;
 }
}


命名空间的使用

(1)加 命名空间名称 及 :: 作用域限定符 [ 每次指定命名空间 ]

int main()
{
    printf("%d\n", N::a);    // :: 作用域限定符 【指定命名空间】
    return 0;    
}

☆(2) using 将 命名空间中某个成员 引入 [ 部分展开 ]

using N::b;
int main()
{
    printf("%d\n", N::a);
    printf("%d\n", b);
    return 0;    
}

(3) using namespace 命名空间名称 引入 [ 全展开 ]

using namespce N;
int main()
{
    printf("%d\n", N::a);
    printf("%d\n", b);
    Add(10, 20);
    return 0;    
}

  • 关于 全展开部分展开 的讨论:
    对命名空间进行展开 [ 命名空间全展开 ] (3) 其实是一件很危险的事,直接展开,全部暴露,又有冲突的风险。 而 每次指定命名空间(1)又很不方便 。
    指定展开(2)就可以解决问题 。指定展开 常用的
using std::cout;
using std::endl;

int main(){
cout<<"hello world\n";

int a=10;
double b=11.11;

cout << a << endl << b << endl;
cout << a << endl << b << endl;
cout << a << endl << b << endl;
}


三、using namespace std

std命名空间的使用惯例:

std 是 C++标准库 的命名空间,如何展开std使用更合理呢?

  1. 在日常练习中,建议直接 using namespace std 即可,这样就很方便。

  2. using namespace std 展开,标准库就全部暴露出来了,如果我们定义跟库重名的 类型/对象/函数 ,就存在冲突问题。该问题在日常练习中很少出现,但是项目开发中代码较多、规模大,就很容易出现。所以建议在项目开发中使用,像 std::cout 这样使用时指定命名空间 + using std::cout 展开常用的库对象/类型等方式。



四、C++ 输入&输出 [ IO流 ]

(一)代码实现

(1)包含头文件 # include<iostream>

  1. io _ int & out 的缩写 + stream 流 => iostream 输入输出流

后面使用cout,cin时,必须 包含< iostream >头文件 以及 按命名空间使用方法使用std
如下图


(2)cout cin endl

c(console 控制台)out,cin

  • cout 标准输出对象(控制台) [ linux 叫其 终端 ]
    在这里插入图片描述

  • cin 标准输入对象(键盘)

cout和cin是全局的流对象。

  • endl [ endline 结束这一行 ] 是特殊的C++符号(C++换行符),表示换行输出

他们都包含在< iostream >头文件中。


(3)<< _ 流插入运算符,>> _ 流提取运算符


实际上 cout和cin分别是ostream和istream类型的对象>>和<<也涉及运算符重载等知识
这些知识我们我们后续才会学习,所以我们这里只是简单学习他们的使用。后面我们还有有一个章节更深入的学习IO流用法及原理


(二)C++输入输出 的特性:可 自动识别变量类型

使用C++输入输出更方便,不需要像printf/scanf输入输出时那样,需要手动控制格式
C++的输入输出可以自动识别变量类型

#include <iostream>
using namespace std;

int main()
{    
   //IO流
   // 可以自动识别变量的类型
   // << 流插入
   std::cout << "hello world";
   int a = 10;
   double b = 11.11;
   
            //int //字符串 //double //字符
   std::cout << a << "\n"<< b << '\n';
   std::cout << a << std::endl << b << std::endl;
   
   return 0;
}


(三)cout,cin更复杂的用法

关于cout和cin还有很多更复杂的用法,比如控制浮点数输出精度控制整形输出进制格式等等。

因为C++兼容C语言的用法所以,这种在C++的复杂用法,可以在C中用其基本语法来实现

int main(){
cout << "hello world\n";

int a = 10;
double b = 11.11;

cout << a << endl << b << endl;
printf("%.1lf\n",b);      //C++兼容C语言的用法,这种在C++的复杂用法,可以在C中用其基本语法来实现
}

这些又用得不是很多,我们这里就不展开学习了。后续如果有需要,我们再配合文档学习。


(三).h 头文件 以及 std命名空间

注意:早期标准库将所有功能 在全局域中实现声明在.h后缀的头文件中使用时只需包含对应头文件即可。

后来将其实现在std命名空间下,为了和C头文件区分,也为了正确使用命名空间规定C++头文件不带.h;旧编译器(vc 6.0)中还支持<iostream.h>格式,后续编译器已不支持,因此推荐使用 <iostream.h> + std 的方式。



五、缺省参数

(一) 缺省参数 概念

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

void Func(int a = 0)
{
 cout<<a<<endl;
}
int main()
{
 Func();     // 没有传参时,使用参数的默认值
 Func(10);   // 传参时,使用指定的实参
return 0;
}


(二) 缺省参数分类

  • 全缺省参数
void Func(int a = 10, int b = 20, int c = 30)
 {
     cout<<"a = "<<a<<endl;
     cout<<"b = "<<b<<endl;
     cout<<"c = "<<c<<endl;
 }
  • 半缺省参数
    半缺省参数必须 从右往左 依次来给出,不能间隔着给
void Func(int a, int b = 10, int c = 20)
 {
     cout<<"a = "<<a<<endl;
     cout<<"b = "<<b<<endl;
     cout<<"c = "<<c<<endl;
 }

注意:

  1. 缺省参数不能在 函数声明 和 定义 中同时出现
    如果 声明与定义位置同时出现 ,恰巧两个位置提供的值不同,那编译器就无法确定到底该用那个缺省值。
//a.h
  void Func(int a = 10);
  
  // a.cpp
  void Func(int a = 20)
 {}
  1. 缺省值必须是 常量 或者 全局变量
  2. C语言不支持(编译器不支持)


六、函数重载

C不支持同名函数,C都是靠函数名来识别函数的,但功能类似,只是因为 形参列表 (参数个数 或 类型 或 类型顺序)不同 ,而因此要开辟很多不同的函数名。这是一件很麻烦且效率低下的事情。

对此,C++做出改进,提出了函数重载的概念。



(一) 函数重载概念

函数重载:是函数的一种特殊情况,C++允许在 同一作用域中 声明几个功能类似同名函数,这
些同名函数的 形参列表(参数个数 或 类型 或 类型顺序)不同 ,常用来处理实现 功能类似数据类型不同 的问题。

  • [ 返回值可用可不用 ]
    会出现不知道该调用谁。


(二)函数重载

(1)参数类型不同

#include<iostream>
using namespace std;

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

int main(){
 Add(10, 20);
 Add(10.1, 20.2);
}

(2)参数个数不同

// 2、参数个数不同
void f()
{
 cout << "f()" << endl;
}
void f(int a)
{
 cout << "f(int a)" << endl;
}

int main()
{ 
 f();//f()则不知道该调用 f()还是f(int a)  
 f(10);     //f(10)可以很明确的是传给f(int a);
}

(3)参数类型顺序不同

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

int main()
{
 f(10, 'a');
 f('a', 10);
 return 0;

}


(三)函数重载 底层原理_名字修饰(name Mangling)

为什么C++支持函数重载,而C语言不支持函数重载呢?

在C/C++中,一个程序要运行起来,需要经历以下几个阶段:预处理编译汇编链接
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

  1. 预处理(.i 文件)
    预处理指令,头文件展开


  1. 编译(.s 文件)
    语法分析,词法分析,语义分析,符号汇总

(语法束)检查语法,生成汇编代码 [指令级代码]( 给我们看的 )


2.1 函数调用在编译阶段,call(地址)
[ 只有声明 ( 但编译器让它过 ),没有地址 ]
[会 在链接阶段,通过根据汇编阶段生成的.o文件符号表( 找到对应函数名的地址 )将call()里面的地址进行链接 ]
在这里插入图片描述


2.2 并且将参数带入进行修饰产生 [ 函数名修饰规则 ]
由于Windows下vs的修饰规则过于复杂,而Linux下g++的修饰规则简单易懂,下面我们使用了g++演示了这个修饰后的名字。

通过下面我们可以看出 gcc(C)的函数修饰后名字不变。而 g++(C++)的函数修饰后变成_Z+函数长度+函数名+类型首字母

  • 采用C语言编译器编译后结果
    在这里插入图片描述
    结论:在linux下,采用gcc编译完成后,函数名字的修饰没有发生改变。

  • 采用C++编译器编译后结果
    在这里插入图片描述结论:在linux下,采用g++编译完成后,函数名字的修饰发生改变,编译器将函数参数类型信息添加到修改后的名字中。

  • Windows下名字修饰规则
    在这里插入图片描述

对比Linux会发现,windows下vs编译器对函数名字修饰规则相对复杂难懂,但道理都是类似的,我们就不做细致的研究了

【扩展学习:C/C++函数调用约定和名字修饰规则–有兴趣好奇的同学可以看看,里面有对vs下函数名修饰规则讲解】
C/C++ 函数调用约定

通过这里就理解了 C语言没办法支持重载,因为同名函数没办法区分。而C++是通过函数修饰规则来区分,只要参数不同,修饰出来的名字就不一样,就支持了重载

如果两个函数 函数名和参数是一样 的,返回值不同不构成重载 的,因为调用时编译器没办法区分
返回值可用可不用 ,会出现不知道该调用谁的情况。



  1. 汇编( 可重定位目标文件 .o 文件)
    3.1 形成符号表。(汇编调用函数时,才会call 地址 ,因此符号表就是在这时产生的)

    符号表:
    形式 “ 函数名 地址 ” [ 函数名 和 地址 的映射 ]
    在这里插入图片描述
    3.2 汇编指令 -> 二进制指令 [ 转换成二进制的机器码( CPU 能认识的 )] -> test.o



  1. 链接
    合并段表
    符号表的合并 和 符号表的重定位
    合并到一起,链接一些没有确定的函数地址等 [ 用函数名去找地址( 函数名修饰规则 )]

实际项目通常是由多个头文件和多个源文件构成,而通过C语言阶段学习的编译链接,我们可以知道,【当前a.cpp中调用了b.cpp中定义的Add函数时】,编译后链接前,a.o的目标文件中没有Add的函数地址,因为Add是在b.cpp中定义的,所以Add的地址在b.o中。那么怎么办呢?

所以 链接阶段就是专门处理这种问题,链接器看到a.o调用Add,但是没有Add的地址,就会到b.o的符号表中找Add的地址,然后链接到一起

那么链接时,面对Add函数,链接接器会使用哪个名字去找呢?这里就通过我们前面 编译阶段讲的 每个编译器都有自己的 函数名修饰规则来找。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

SharePoint 的 Web Parts 是什么

Web Parts 可以说是微软 SharePoint 的基础组件。 根据微软自己的描述&#xff0c;Web Parts 是 SharePoint 对内容进行构建的基础&#xff0c;可以想想成一块一块的砖块。 我们需要使用这些砖块来完成一个页面的构建。 我们可以利用 Web Parts 在 SharePoint 中添加文本&am…

时序预测 | MATLAB实现基于BP-Adaboost的BP神经网络结合AdaBoost时间序列预测

时序预测 | MATLAB实现基于BP-Adaboost的BP神经网络结合AdaBoost时间序列预测 目录 时序预测 | MATLAB实现基于BP-Adaboost的BP神经网络结合AdaBoost时间序列预测预测效果基本介绍模型描述程序设计参考资料 预测效果 基本介绍 1.MATLAB实现基于BP-Adaboost的BP神经网络结合AdaB…

NZ系列工具NZ08:图表添加标签工具

我的教程一共九套及VBA汉英手册一部&#xff0c;分为初级、中级、高级三大部分。是对VBA的系统讲解&#xff0c;从简单的入门&#xff0c;到数据库&#xff0c;到字典&#xff0c;到高级的网抓及类的应用。大家在学习的过程中可能会存在困惑&#xff0c;这么多知识点该如何组织…

(二)正点原子I.MX6ULL u-boot移植

一、概述 这里使用的是NXP官方2022.04发布的uboot&#xff0c;移植到正点原子阿尔法开发板&#xff08;v2.1&#xff09; u-boot下载&#xff1a;gitgithub.com:nxp-imx/uboot-imx.git 移植是基于NXP的mx6ull_14x14_evk 二、编译NXP官方uboot 进入NXP的u-boot目录 先在Makefile…

通过设置响应头解决跨域问题

网上很多文章都是告诉你直接Nginx添加这几个响应头信息就能解决跨域&#xff0c;当然大部分情况是能解决&#xff0c;但是我相信还是有很多情况&#xff0c;明明配置上了&#xff0c;也同样会报跨域问题。 这大概率是因为&#xff0c;服务端没有正确处理预检请求也就是OPTIONS请…

Word 插入的 Visio 图片显示为{EMBED Visio.Drawing.11} 解决方案

World中&#xff0c;如果我们插入了Visio图还用了Endnote&#xff0c; 就可能出现&#xff1a;{EMBED Visio.Drawing.11}问题 解决方案&#xff1a; 1.在相应的文字上右击&#xff0c;在出现的快捷菜单中单击“切换域代码”&#xff0c;一个一个的修复。 2.在菜单工具–>…

配置交换机将Log发送到日志服务器

文章目录 一、配置说明二、配置步骤推荐阅读 一、配置说明 配置将实现如下&#xff1a; 配置交换机将Log发送到日志服务器。 将信息等级高于等于 debug 的日志信息将会发送到日志服务器上。 允许输出日志信息的模块为default所有应用模块日志信息。 SW1为我们日志源交换机…

【Redis】redis的下载安装

目录 1.window安装 2.Linux安装 下载 解压 移动redis目录 编译 1.window安装 在github 下载redis的安装包 https://github.com/microsoftarchive/redis/releases 下载完后安装相应的目录下&#xff0c;比如我是放在c盘的Program Files下 开启redis&#xff0c;双击运行…

CCF ChinaSoft 2023 论坛巡礼|程序设计教育论坛

2023年CCF中国软件大会&#xff08;CCF ChinaSoft 2023&#xff09;由CCF主办&#xff0c;CCF系统软件专委会、形式化方法专委会、软件工程专委会以及复旦大学联合承办&#xff0c;将于2023年12月1-3日在上海国际会议中心举行。 本次大会主题是“智能化软件创新推动数字经济与社…

SQL练习---511.游戏玩法分析 I

题目描述 分析 题目描述很简单&#xff0c;找出用户第一次登陆的时期&#xff0c;很简单一个用户有多个记录&#xff0c;因此按用户分组即可&#xff0c;但是不知道日期能否求出最小值&#xff0c;事实证明还是可以的。 题解 select player_id,min(event_date) first_login f…

Linux的命令——关于操作用户及用户组的命令

目录 1.Linux的命令格式 2.用户与用户组管理 2.1用户管理 添加用户 设置用户密码 删除用户 修改用户 2.2用户组管理 新增用户组 删除用户组 修改用户组属性 用户组切换 用户组管理 用户切换 1. su 2.sudo 1.Linux的命令格式 Linux系统中几乎所有操作&#xff0…

基于STM32控制直流电机加减速正反转仿真设计

**单片机设计介绍&#xff0c;基于STM32控制直流电机加减速正反转仿真设计 文章目录 一 概要二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 本设计由STM32F103、L298N电机驱动电路、按键电路组成。通过按键可以控制电机&#xff0c;正转、反转、加…

8年经验之谈 —— 接口测试框架中的鉴权处理!

接口自动化测试中通常都有鉴权机制&#xff0c;就是判断是否在登录状态下&#xff0c;已登录方可调用接口&#xff0c;未登录则不可调用。本文将带领大家学习使用rest-assured框架实现基于cookies和token的鉴权关联&#xff0c;实现接口自动化测试。 1、基于cookies的鉴权关联…

Http状态码502常见原因及排错思路(实战)

Http状态码502常见原因及排错思路 502表示Bad Gateway。当Nginx返回502错误时&#xff0c;通常表示Nginx作为代理服务器无法从上游服务器&#xff08;如&#xff1a;我们的后端服务器地址&#xff09;获取有效的响应。导致这种情况的原因有很多&#xff1a; 后端服务器故障ngin…

基于SpringBoot+Vue的博物馆管理系统

基于springbootvue的博物馆信息管理系统的设计与实现~ 开发语言&#xff1a;Java数据库&#xff1a;MySQL技术&#xff1a;SpringBootMyBatisVue工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 主页 登录界面 管理员界面 用户界面 摘要 基于SpringBoot和Vue的博物馆…

vue3响应式api

响应式api——compositon api setup&#xff1a; 不要再想this问题执行是在beforeCreated之前 beforeCreated&#xff1a;也就是创建了一个实例 created&#xff1a;挂载了数据 通过形参props接收&#xff0c;只读 以后所有代码都写到setup中 判断是否只读&#xff1a;isReadon…

lc228. 汇总区间

暴力解法&#xff1a;遍历数组&#xff0c;判断数组是否连续递增。将连续递增的数据的首尾的数据分别存储在map集合的key和value之中&#xff0c;输出时判断首尾值是否相同采用两种方式输出 复杂度分析 时间复杂度&#xff1a;O(n)&#xff0c;其中 n 为数组的长度。空间复杂…

2023亚太杯数学建模A题思路

文章目录 0 赛题思路1 竞赛信息2 竞赛时间3 建模常见问题类型3.1 分类问题3.2 优化问题3.3 预测问题3.4 评价问题 4 建模资料5 最后 0 赛题思路 &#xff08;赛题出来以后第一时间在CSDN分享&#xff09; https://blog.csdn.net/dc_sinor?typeblog 1 竞赛信息 2023年第十三…

【LeetCode】每日一题 2023_11_12 每日一题 Range 模块

文章目录 刷题前唠嗑题目&#xff1a;Range 模块题目描述代码与解题思路 刷题前唠嗑 LeetCode? 启动&#xff01;&#xff01;&#xff01; 嗯&#xff1f;怎么是 hard&#xff0c;好长&#xff0c;可恶&#xff0c;看不懂&#xff0c;怎么办 题目&#xff1a;Range 模块 题…

线索二叉树(存储结构,线索化,寻找前驱/后继)

目录 1.线索二叉树1.中序线索二叉树2.后序线索二叉树3.先序线索二叉树 2.线索二叉树的存储结构3.二叉树的线索化1.中序线索化2.先序线索化3.后序线索化 4.寻找前驱/后继1.中序线索二叉树找后继2.中序线索二叉树找中序前驱3.先序线索二叉树找先序后继4.先序线索二叉树找先序前驱…