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

简介: 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++关键字、命名空间、输入&输出,后续持续讲解!

相关文章
|
3月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
169 0
|
3月前
|
存储 安全 编译器
c++入门
c++作为面向对象的语言与c的简单区别:c语言作为面向过程的语言还是跟c++有很大的区别的,比如说一个简单的五子棋的实现对于c语言面向过程的设计思路是首先分析解决这个问题的步骤:(1)开始游戏(2)黑子先走(3)绘制画面(4)判断输赢(5)轮到白子(6)绘制画面(7)判断输赢(8)返回步骤(2) (9)输出最后结果。但对于c++就不一样了,在下五子棋的例子中,用面向对象的方法来解决的话,首先将整个五子棋游戏分为三个对象:(1)黑白双方,这两方的行为是一样的。(2)棋盘系统,负责绘制画面。
47 0
|
7月前
|
存储 缓存 C++
C++ 容器全面剖析:掌握 STL 的奥秘,从入门到高效编程
C++ 标准模板库(STL)提供了一组功能强大的容器类,用于存储和操作数据集合。不同的容器具有独特的特性和应用场景,因此选择合适的容器对于程序的性能和代码的可读性至关重要。对于刚接触 C++ 的开发者来说,了解这些容器的基础知识以及它们的特点是迈向高效编程的重要一步。本文将详细介绍 C++ 常用的容器,包括序列容器(`std::vector`、`std::array`、`std::list`、`std::deque`)、关联容器(`std::set`、`std::map`)和无序容器(`std::unordered_set`、`std::unordered_map`),全面解析它们的特点、用法
C++ 容器全面剖析:掌握 STL 的奥秘,从入门到高效编程
|
6月前
|
存储 分布式计算 编译器
C++入门基础2
本内容主要讲解C++中的引用、inline函数和nullptr。引用是变量的别名,与原变量共享内存,定义时需初始化且不可更改指向对象,适用于传参和返回值以提高效率;const引用可增强代码灵活性。Inline函数通过展开提高效率,但是否展开由编译器决定,不建议分离声明与定义。Nullptr用于指针赋空,取代C语言中的NULL。最后鼓励持续学习,精进技能,提升竞争力。
|
7月前
|
安全 编译器 程序员
C++ noexcept 关键字的关键作用
`noexcept` 关键字在 C++ 中扮演着重要角色,通过正确使用 `noexcept`,可以提升程序的性能、增强代码的可读性和安全性,并且有助于编译器进行优化。在编写 C++ 代码时,应仔细考虑每个函数是否应该声明为 `noexcept`,以充分利用这一特性带来的优势。通过本文的介绍,希望开发者能够更好地理解和应用 `noexcept` 关键字,从而编写出更加高效、健壮的 C++ 程序。
236 8
|
7月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
3月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
92 0
|
5月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
179 12
|
6月前
|
设计模式 安全 C++
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
129 16