C++入门——关键字|命名空间|输入&输出

本文涉及的产品
.cn 域名,1个 12个月
简介: C++入门——关键字|命名空间|输入&输出

前言:

       今天我们又开启了一个崭新的大门——C++面向对象编程语言,C++是怎么来的呢?答案是:因为C语言的有很多不足,我们的祖师爷用着不爽,就不断更改,就改出来了一门新的语言,C++。C++语言兼容C语言,所以C的代码在C++中也可以使用。


0、本节知识点安排的目的

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

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

       (2)为后续类和对象学习打基础。



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

       C++总计63个关键字,C语言32个关键字。

       PS:下面我们只是看一下C++有哪些关键字,不对关键字进行具体的讲解,目前我们只需知道命名时要避开这些关键字。后面我们学到在细讲。

二、命名空间

        C++要解决的第一个问题就是命名冲突的问题。

       代码演示:我们定义一个全局变量,取名为rand。

#include<stdio.h>
#include<stdlib.h>//预处理,对rand的声明
int rand = 10;//定义变量rand
int main()
{
  printf("%d\n", rand);
  return 0;
}


当我们运行程序后,报错:C2365 “rand”:重定义;以前的定义是“函数”。

       我们知道,rand是stdlib.h这个头文件声明的一个函数,但是这里我又定义了一个变量rand,所以此时就发生了命名冲突。

总结:

       (1)可能发生命名冲突的类型:

1、我们跟库冲突 —— 在C/C++中,变量、函数和后面要学到的类都是大量存在的,这些变量、函数和类的名称将都存在于全局作用域中,所以可能会导致很多冲突。

2、我们互相之间冲突 —— 在大型项目中,常常需要程序员互相分工合作完成,在命名的时候,难免可能会相同,所以可能会导致冲突。

(2)那怎么解决命名冲突的问题呢?

       答案是:在C++中我们引入了namespace —— 命名空间。使用命名空间的目的就是对标识符的名称进行本地化,以避免命名冲突或名字污染,namespace关键字的出现就是针对这种问题的。

1、命名空间定义

       关键字namespace是用来定义命名空间的,namespace后面跟命名空间的名字,然后接一对{}即可,{}中即为命名空间的成员。

       即定义出一个命名空间域,用命名空间域来隔离,解决命名冲突。

(1)域

1、同一个域不能定义同名的变量,不同域可以定义同名的变量。

2、域有的地方可以影响访问,有的地方可以影响生命周期。

3、局部域与全局域即影响访问,也影响生命周期。

4、访问变量时:编译器默认先局部找,找不到再去全局找。(都找不到就报错)

5、分类:有①局部域;②全局域;③命名空间域;④类域

讲解:

       ①访问变量时:编译器默认先局部找,找不到再去全局找。

#include<stdio.h>
int a = 10;//全局域
int main()
{
  int a = 0;//局部域
  printf("%d\n", a);//直接访问a,这里打印局部域的a
  //那我们想打印全局域的a,该怎么做呢?
  //答案是:使用 ::域作用限定符,指定方式:域名::变量名
  //全局作用域没有域名,所以 ::左边是空白就代表它去全局域访问
  printf("%d\n", ::a);
  return 0;
}

运行结果:

说明:

1、编译器的访问顺序:局部 > 全局,找不到或命名冲突则报错

2、局部与全局都存在一个同名变量a时,想要访问全局的a —— 使用::域作用限制符。

3、域作用限制符的指定方式:域名::变量名

4、全局域没有域名,所以::左边是空白就代表它去全局域访问。

(2)命名空间的定义讲解

       在上面我们那个代码中,rand发生了命名冲突的问题,我们现在使用namespace定义一个命名空间来解决命名冲突的问题。

       代码演示1:正常的命名空间定义

#include<stdio.h>
#include<stdlib.h>//#include包含头文件stdlib.h,在预处理指令时,它会打开指定的头文件,
        //并将其中的代码插入到包含该指令的源文件中,然后再进行编译。
namespace wjs
{
  int rand = 10;
}
int main()
{
  //直接访问rand:编译器是现在局部域找,找不到再去全局找,全局找不到就报错了。
  //要访问命名空间中的成员:
  //①展开命名空间域 —— 编译时是否去命名空间中搜索,即是将命名空间中声明的成员
  //暴露到当前作用域内可以直接使用(注意:并没有改变其所在的域)。
  //②指定访问命名空间域
  printf("%d\n", wjs::rand);
  return 0;
}

说明:

1、wjs命名空间的名字,一般开发中是用项目名字做命名空间名,平时练习大家可以用自己名字缩写即可。

2、命名空间的定义必须在全局域中。(因为命名空间就是在解决全局域中声明的名称与库里面的名称发生冲突的问题,所以命名空间的定义必须在全局域中)

3、编译器不能自动访问命名空间中的成员,想要访问必须:

       ①展开命名空间域 ——编译时去命名空间中搜索,即是将命名空间中声明的成员暴露到当前作用域内,可以直接使用(注意:并没有改变其所在的域)

       ②指定访问命名空间域

4、#include包含头文件stdlib.h —— 在预处理指令时,它会打开指定的头文件,并将其中的代码插入到包含该指令的源文件中,然后再进行编译。

5、头文件只是提供了这些名称的声明,让编译器知道他们的存在。

6、命名空间中可以定义变量、函数、类型。

 代码演示2:命名空间可以嵌套

#include<stdio.h>
namespace N1
{
  int a;
  int b;
  int Add(int left, int right)
  {
    return left + right;
  }
  namespace N2
  {
    int c = 10;
    int d;
    int Sub(int left, int right)
    {
      return left - right;
    }
  }
}
int main()
{
  //访问N1中的a
  printf("%d\n", N1::a);
  //访问N2中的C
  printf("%d\n", N1::N2::c);
  return 0;
}

代码演示3:同一个工程中允许存在多个相同名称的命名空间,编译器最后会合成到同一个命名空间中。

test.h文件:

namespace wjs
{
  int a = 10;
}

    test.cpp文件:

#include<stdio.h>
#include"test.h"
namespace wjs
{
  int rand = 10;
}
int main()
{
  printf("%d\n", wjs::rand);
  printf("%d\n", wjs::a);
  return 0;
}

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

2、命名空间的使用

在前面我们已经知道,编译器不能自动访问命名空间中的成员。想要访问必须:

①展开命名空间域 ——编译时去命名空间中搜索,即是将命名空间中声明的成员暴露到当前作用域内,可以直接使用(注意:并没有改变其所在的域)。

②指定访问命名空间域。

现在我们对其详细的讲解,把命名空间的使用分为三种方式

(1)加命名空间名称及作用域限定符

       适用于使用次数少的,代码演示:

//为什么C++标准库中的头文件都没有后缀名.h了?
//答案是:早期标准库将所有功能在在全局域中实现,声明在.h后缀的头文件中,
// 使用时只需包含对应头文件即可,后来C++标准引入了命名空间,
// 就把C++的标准库的定义实现都放到std这个命名空间中了,
// std是C++标准库的命名空间名。为了与C头文件作区分,也为了正确使用命名空间,
// 规定C++头文件不带.h。如C中有一个string.h的头文件,C++中也有一个string的头文件
#include<iostream>
int main()
{
  //编译器不能自动访问命名空间中的成员。
  //方式1:加命名空间名称及作用域限制符 —— 编译器直接去指定命名空间中搜索访问,
  //不会去局部域,全局域中搜索。
  std::cout << "hello world!" << std::endl;
  return 0;
}

说明:1、命名空间名称::成员 —— 编译器直接去指定命名空间中搜索访问该成员,不会去局部域和全局域中搜索。

2、为什么C++标准库中的头文件都没有后缀名.h?

       答案是:早期标准库将所有功能在全局域中实现,声明在.h后缀的头文件中,使用时只需包含对应的头文件即可(如老编译器VC 6.0还支持),后来C++标准引入了命名空间,就把C++的标准库的定义实现都放在std这个命名空间中了,std是C++标准库的命名空间名。为了与C头文件作区分,也为了正确使用命名空间,规定C++头文件不带.h。如C中有一个string.h的头文件,C++中也有一个string的头文件。

3、在项目中我们常常使用指定命名空间的方法来访问成员。

(2)使用using将命名空间中的某个成员引入

       当使用次数多时,我们每一次都要指定命名空间,那也太麻烦了,这里我们可以展开常用的名称,就不用再每一次都指定命名空间了。

代码演示:

#include<iostream>
using std::cout;//展开std中的cout
using std::endl;
int main()
{
  //下面cout,endl我们就可以不用每一次都指定命名空间std了
  cout << "hello world" << endl;
  cout << "hello world" << endl;
  cout << "hello world" << endl;
  return 0;
}

说明:

1、using 命名空间名::某个成员 —— 展开命名空间中的某个成员,编译时去命名空间中搜索,即将这个成员暴露到当前作用域,可以直接使用

2、当我们经常使用命名空间中的某个成员时,建议使用,因为展开常用的名称,就不用再每一次都指定命名空间了。

(3)使用using namespace 命名空间名称引入

       直接展开命名空间域,代码演示:

#include<iostream>
using namespace std;//展开命名空间
int main()
{
  //下面cout,endl我们就可以不用每一次都指定命名空间std了
  cout << "hello world" << endl;
  cout << "hello world" << endl;
  cout << "hello world" << endl;
  return 0;
}

说明:

1、展开命名空间域 ——编译时去命名空间中搜索,即是将命名空间中声明的成员暴露到当前作用域内,可以直接使用,所以直接展开会有风险,失去了命名空间的含义,我们定义的名称(如:变量名、函数名、类名)如果跟库重名,就报错了。

2、建议项目中不要去展开,日常练习我们可以为了方便可以去直接展开,项目中建议指定命名空间访问。

三、C++输入&输出

       代码演示:

#include<iostream>
using namespace std;//展开命名空间
int main()
{
  int a = 10;
  double b = 3.14;
  cout << a << endl;//输入a
  cout << b << '\n';//输出b
  cin >> a >> b;//输入a,b
  cout << a << " " << b << endl;//输出a 空格 b 
  return 0;
}

说明:

1、使用cout标准输出对象(控制台)和cin标准输入对象(键盘)时,必须包含头文件以及按命名空间使用方法使用std。

2、cout和cin是全局的流对象,endl是特殊的C++符号,表示换行输出,他们都包含在头文件中。

3、<<流插入运算符,>>流提取运算符。(可以理解为流水的方向)

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

5、实际上cout和cin分别是ostream和istream类型的对象,>>和<<也涉及运算符重载等知识,这些知识我们后续才会学习,所以我们这里只是简单学习他们的使用。

6、关于cout和cin还有很多更复杂的用法,比如控制浮点数输出精度,控制整形输出格式等等。因为C++兼容C语言的用法,我建议使用C的方法即可了因为C++相较于C格式复杂一些,有兴趣也可以去了解C++的。(总结:C++兼容C,哪个使用方便就使用哪个)


C++入门今天就先讲解了C++关键字、命名空间、输入&输出,后续持续讲解!

相关文章
|
23天前
|
安全 编译器 C++
C++ `noexcept` 关键字的深入解析
`noexcept` 关键字在 C++ 中用于指示函数不会抛出异常,有助于编译器优化和提高程序的可靠性。它可以减少代码大小、提高执行效率,并增强程序的稳定性和可预测性。`noexcept` 还可以影响函数重载和模板特化的决策。使用时需谨慎,确保函数确实不会抛出异常,否则可能导致程序崩溃。通过合理使用 `noexcept`,开发者可以编写出更高效、更可靠的 C++ 代码。
28 0
|
3月前
|
C++
C++入门13——详解多态2
C++入门13——详解多态2
94 1
|
3月前
|
存储 安全 编译器
【C++打怪之路Lv1】-- 入门二级
【C++打怪之路Lv1】-- 入门二级
34 0
|
3月前
|
自然语言处理 编译器 C语言
【C++打怪之路Lv1】-- C++开篇(入门)
【C++打怪之路Lv1】-- C++开篇(入门)
40 0
|
3月前
|
分布式计算 Java 编译器
【C++入门(下)】—— 我与C++的不解之缘(二)
【C++入门(下)】—— 我与C++的不解之缘(二)
|
3月前
|
编译器 Linux C语言
【C++入门(上)】—— 我与C++的不解之缘(一)
【C++入门(上)】—— 我与C++的不解之缘(一)
|
1天前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
32 18
|
1天前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
28 13
|
1天前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
18 5
|
1天前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
16 5