一、从C语言到C++(一)
C++介绍
C语言和C++的联系
- 继承关系:C++是由C语言发展而来的,因此C++与C语言具有很深的渊源和联系。C++在设计时充分考虑了对C语言的兼容性,使得大多数C语言代码都可以不加修改地在C++环境中编译和运行。
- 扩展与增强:C++在C语言的基础上增加了许多新的特性和功能,特别是面向对象编程(OOP)的支持,使得C++在编程时更加灵活和强大。这些新增的特性包括类、封装、继承、多态等。
- 编译方式:尽管C++已经发展成为一门独立的编程语言,但现有的许多C/C++编译器仍然可以同时支持C和C++代码的编译。这些编译器通过不同的前端处理C和C++的语法和语义,但共享后端和优化器等基础设施。
C++介绍
- 基本特性:C++是一种静态类型的、编译式的、通用的、大小写敏感的编程语言,支持过程化编程、面向对象编程和泛型编程。C++被认为是一种中级语言,它综合了高级语言和低级语言的特点。
- 面向对象编程:C++完全支持面向对象的程序设计,包括封装、继承、多态和抽象等四大特性。这些特性使得C++在程序设计时更加灵活和易于维护。
- 泛型编程:C++支持泛型编程,可以实现代码的复用和泛化。泛型编程的思想是将代码中与数据类型无关的部分抽象出来,形成通用的代码框架,从而实现更高效、更灵活的代码编写。
- 标准库:标准的C++由三个重要部分组成:核心语言、C++标准库和标准模板库(STL)。这些库提供了大量的函数和数据结构,用于操作文件、字符串、数据结构等,极大地提高了编程效率。
- ANSI标准:为了确保C++的便携性,ANSI标准规定了C++的语法和语义。由于ANSI标准已经稳定使用了很长时间,所有主要的C++编译器的制造商都支持ANSI标准。
- 创始人与发展:C++是由Bjarne Stroustrup于1979年在新泽西州美利山贝尔实验室开始设计开发的。最初命名为"带类的C",后来在1983年更名为C++。
综上所述,C++和C语言虽然都是编程语言,但C++在C语言的基础上增加了许多新的特性和功能,特别是在面向对象编程和泛型编程方面的支持,使得C++在编程时更加灵活和强大。同时,C++也继承了C语言的许多优点,如简洁、高效和可移植性等。
头文件
从C语言到C++的演进中,头文件(header files)的角色和用法也发生了一些变化。在C语言中,头文件主要用于声明函数、变量、宏和类型定义等,以便在多个源文件中共享这些声明。而在C++中,头文件的使用方式与C语言类似,但也有一些扩展和新的特性。
-
声明和定义分离:
- 在C++中,通常将类的声明(即类的接口)放在头文件中,而将类的成员函数的具体实现(即定义)放在源文件中。这样可以使头文件更加简洁,并且可以在多个源文件中包含同一个头文件而不会导致多重定义错误。
-
包含保护:
- 为了防止头文件被多次包含(即多重包含)导致的编译错误,C++程序员通常会在头文件的开头和结尾使用预处理器指令来创建包含保护。这通常是通过定义一个唯一的标识符(如使用
#ifndef
,#define
, 和#endif
)来实现的。
cpp// myclass.h #ifndef MYCLASS_H #define MYCLASS_H // 类的声明 class MyClass { public: // ... }; #endif // MYCLASS_H
- 为了防止头文件被多次包含(即多重包含)导致的编译错误,C++程序员通常会在头文件的开头和结尾使用预处理器指令来创建包含保护。这通常是通过定义一个唯一的标识符(如使用
-
模板和头文件:
- C++中的模板(包括函数模板和类模板)通常需要在头文件中定义,因为模板的实例化是由编译器在编译时完成的,并且需要看到模板的完整定义。因此,模板的头文件通常不仅包含声明,还包含定义。
-
命名空间:
- C++引入了命名空间(namespaces)来组织代码,避免命名冲突。在头文件中使用命名空间可以使代码更加清晰,并允许开发者在不同的命名空间中定义相同名称的类或函数。
-
内联函数:
- C++支持内联函数(inline functions),这些函数通常在头文件中定义,以便编译器在调用点处内联展开函数体,提高执行效率。
-
标准库头文件:
- C++标准库提供了大量的头文件,这些头文件包含了标准库中的类和函数的声明。这些头文件通常以
<iostream>
,<vector>
,<string>
等形式包含,与C语言中的标准库头文件(如<stdio.h>
,<stdlib.h>
)在命名和用法上有所不同。
- C++标准库提供了大量的头文件,这些头文件包含了标准库中的类和函数的声明。这些头文件通常以
-
#pragma once:
- 一些编译器支持
#pragma once
指令作为包含保护的一种替代方法。这是一个非标准的扩展,但它在许多编译器上都被支持,并且在某些情况下可能比使用#ifndef
/#define
/#endif
更加简洁和方便。
- 一些编译器支持
请注意,尽管C++在头文件的使用上有一些新的特性和约定,但C++仍然与C语言兼容,因此许多C语言中的头文件和包含约定在C++中仍然有效。然而,随着C++的发展,许多新的编程风格和最佳实践已经形成,并在现代C++代码中得到了广泛应用。
命名空间
从C语言过渡到C++时,一个显著的特性是C++引入了**命名空间(Namespace)**的概念。命名空间是C++中用于解决命名冲突的一种方法,它允许开发者将相关的名称(如变量名、函数名、类名等)组织在一起,形成一个逻辑上的分组。
定义命名空间
在C++中,你可以使用namespace
关键字来定义一个命名空间。例如:
cpp
namespace MyNamespace {
int variable = 42;
void myFunction() {
// ...
}
class MyClass {
public:
// ...
};
}
在这个例子中,我们定义了一个名为MyNamespace
的命名空间,并在其中声明了一个整数变量variable
、一个函数myFunction()
和一个类MyClass
。
使用命名空间中的名称
要在命名空间外部访问命名空间中的名称,你需要使用作用域解析运算符::
。例如:
cpp
int main() {
std::cout << MyNamespace::variable << std::endl; // 假设我们包含了<iostream>
MyNamespace::myFunction();
MyNamespace::MyClass obj;
// ...
return 0;
}
注意,在这个例子中,我们还使用了std
命名空间中的cout
对象,它是C++标准库中的一个输出流对象。
使用using
声明或指令
如果你不想每次都使用作用域解析运算符来访问命名空间中的名称,你可以使用using
声明或指令来简化代码。
- using声明:在局部作用域中引入命名空间中的某个特定名称。
cpp
using MyNamespace::myFunction;
myFunction(); // 不需要MyNamespace::前缀
- using指令:在局部作用域或全局作用域中引入整个命名空间中的所有名称。
cpp
using namespace MyNamespace;
myFunction(); // 不需要MyNamespace::前缀
MyClass obj; // 同样不需要MyNamespace::前缀
但是,过度使用using namespace
指令可能会导致命名冲突,特别是在大型项目中,因此通常建议在函数或类的局部作用域中使用它,而不是在全局作用域中。
命名空间与C语言的对比
在C语言中,没有命名空间的概念。为了避免命名冲突,开发者通常会在变量名、函数名等前面加上特定的前缀或后缀来标识它们所属的模块或库。然而,这种方法并不是一种很好的解决方案,因为它可能会导致名称变得冗长且难以阅读。C++的命名空间提供了一种更优雅和灵活的方式来组织和管理代码中的名称。
给命名空间起别名
在编程中,特别是在C++或C#等语言中,你可以给命名空间起别名,以便于更简洁地引用命名空间中的内容。这在命名空间名称较长或需要频繁引用某个命名空间中的元素时特别有用。
在C++中,你可以使用namespace
指令与using
关键字结合来给命名空间起别名。以下是一个示例:
cpp
#include <iostream>
namespace VeryLongNamespaceName {
void printHello() {
std::cout << "Hello from VeryLongNamespaceName!" << std::endl;
}
}
// 给命名空间起别名
namespace VLNN = VeryLongNamespaceName;
int main() {
// 使用别名调用命名空间中的函数
VLNN::printHello();
return 0;
}
在这个例子中,VeryLongNamespaceName
是一个很长的命名空间名称,我们通过 namespace VLNN = VeryLongNamespaceName;
为它创建了一个别名 VLNN
。之后,我们就可以通过这个简短的别名来访问命名空间中的内容了。
请注意,起别名时应选择简洁且有意义的名称,以提高代码的可读性。同时,避免使用可能与现有命名空间或类名冲突的别名。
注意事项
-
定义与使用位置:
- 命名空间使用
namespace
关键字声明,并且必须位于其他代码之前(包括任何非PHP代码以及空白符,除了declare
关键字)。 - 命名空间下的类(包括抽象类和traits)、接口、常量和函数会受命名空间影响。
- 命名空间使用
-
避免重复定义:
- 在头文件中使用命名空间时,不能定义函数或变量,因为这可能会导致多个源文件中出现重复定义的链接错误。
- 如果需要在命名空间中定义变量,可以加上
static
或constexpr
关键字,使其变成静态变量或常量。 - 如果需要在命名空间中定义函数,可以加上
inline
关键字,使其变成内联函数。
-
结构体和类的定义:
- 在头文件中使用命名空间时,可以定义结构体和类,因为这些不是具体的实例,不会导致链接错误。
-
using关键字的使用:
using
关键字用于引入命名空间中的符号,以便在代码中直接使用这些符号,而不需要使用完整的命名空间前缀。- 使用
using
关键字时,建议在引用完所有头文件后使用,以避免符号冲突。 - 尽量避免在头文件中使用
using namespace
语句,因为这相当于引入了命名空间内的所有元素,可能会导致其他文件使用时出现命名冲突。
-
避免全局变量的使用:
- 过多使用全局变量会导致内存占用问题,并可能引发命名冲突,应该尽量避免。
-
注意循环和函数调用:
- 尽量减少循环嵌套次数和函数调用次数,以提高程序性能。
-
命名空间的合并:
- 允许存在多个相同名称的命名空间,编译器会自动将其合并到同一个命名空间中。
-
函数定义的位置:
- 命名空间中的函数,其定义可以放在命名空间内部,也可以放在命名空间外部。放在命名空间外部时,函数前面必须加上命名空间名的前缀。
遵循这些注意事项可以帮助你更好地管理和组织代码,提高代码的可读性和可维护性。
std
在C++中,std
是一个非常重要的命名空间,它包含了C++标准库中的大部分内容。std
是 "standard" 的缩写,表示这个命名空间包含了C++标准所定义的功能和对象。
当你使用C++标准库中的任何功能时,例如输入输出流(如 std::cout
和 std::cin
)、字符串(如 std::string
)、容器(如 std::vector
, std::map
, std::set
等)、算法(如 std::sort
, std::find
等)以及其他许多实用工具时,你通常需要使用 std
命名空间。
有几种方法可以访问 std
命名空间中的名称:
-
使用作用域解析运算符
::
:你可以通过
std::
前缀来明确指定你想要使用的是标准库中的名称。例如:cppstd::cout << "Hello, world!" << std::endl; std::string s = "Example";
-
使用
using
声明 :如果你只想在局部作用域中使用某个特定的标准库名称,你可以使用
using
声明。例如:cppusing std::cout; using std::endl; cout << "Hello, world!" << endl;
但请注意,这只会引入
cout
和endl
到当前作用域,不会引入其他标准库名称。 -
使用
using
指令 :如果你希望在当前文件中使用整个
std
命名空间中的所有名称,你可以使用using
指令。但请注意,这可能会导致命名冲突,特别是当你自己定义的名称与标准库中的名称相同时。因此,在大型项目中通常不推荐这样做。cppusing namespace std; cout << "Hello, world!" << endl;
为了编写清晰、可维护的代码,通常建议只在必要时使用 using
声明,并在整个项目中一致地使用 std::
前缀来访问标准库名称。这有助于避免命名冲突,并使代码更易于阅读和理解。
此外,C++标准库还包含了许多头文件,你需要通过包含这些头文件来使用标准库中的功能。例如,要使用输入输出流,你需要包含 <iostream>
头文件;要使用字符串,你需要包含 <string>
头文件;要使用容器和算法,你需要包含 <vector>
, <map>
, <set>
, <algorithm>
等头文件。
标准输入输出
在C++中,标准输入输出是通过预定义的流对象来实现的,这些对象分别是std::cin
(用于输入)和std::cout
(用于输出)。这两个对象都是C++标准库<iostream>
中的一部分。
std::endl
std::endl
是 C++ 标准库 <iostream>
中定义的一个操纵符(manipulator),它通常与输出流(如 std::cout
)一起使用,以在输出中添加一个新行,并立即刷新输出缓冲区。
源码:
cpp
_EXPORT_STD template <class _Elem, class _Traits>
basic_ostream<_Elem, _Traits>& __CLRCALL_OR_CDECL endl(
basic_ostream<_Elem, _Traits>& _Ostr) { // insert newline and flush stream
_Ostr.put(_Ostr.widen('\n'));
_Ostr.flush();
return _Ostr;
}
std::endl
的功能主要有两个:
- 添加新行 :在输出的末尾添加一个换行符(在 Unix/Linux 系统中是
\n
,在 Windows 中是\r\n
),这样下一次的输出就会从新的一行开始。 - 刷新缓冲区 :调用
std::endl
后,输出缓冲区会被立即刷新,确保到目前为止所有的输出都被发送到它们的目标位置(如控制台、文件等)。如果不刷新缓冲区,输出可能会被缓存起来,直到缓冲区满或遇到其他导致刷新的操作。
使用 std::endl
的一个常见例子是:
cpp
#include <iostream>
int main() {
std::cout << "Hello, World!" << std::endl;
std::cout << "This is a new line." << std::endl;
return 0;
}
在这个例子中,std::endl
确保了 "Hello, World!" 和 "This is a new line." 分别在新的一行输出,并且每个字符串之后都立即刷新了输出缓冲区。
然而,需要注意的是,频繁地刷新输出缓冲区可能会影响程序的性能,因为刷新操作通常是相对昂贵的。因此,在不需要立即看到输出或不需要确保输出顺序的情况下,可以使用 \n
代替 std::endl
来避免不必要的刷新。例如:
cpp
std::cout << "Hello, World!\n";
std::cout << "This is a new line.\n";
// 输出缓冲区不会被立即刷新
使用std::cout
进行输出
std::cout
是一个输出流对象,它允许你将数据发送到标准输出设备(通常是终端或控制台)。你可以使用插入运算符(<<
)将数据发送到std::cout
。
示例:
cpp
#include <iostream>
int main() {
std::cout << "Hello, World!" << std::endl;
return 0;
}
使用std::cin
进行输入
std::cin
是一个输入流对象,它允许你从标准输入设备(通常是键盘)读取数据。你可以使用提取运算符(>>
)从std::cin
读取数据。
示例:
cpp
#include <iostream>
int main() {
int num;
std::cout << "Enter a number: ";
std::cin >> num;
std::cout << "You entered: " << num << std::endl;
return 0;
}
在上面的示例中,程序首先输出一个提示消息,然后等待用户输入一个整数。当用户输入一个整数后,程序读取该整数并将其存储在变量num
中,然后输出该整数。
格式化输出
C++标准库提供了许多操纵符来控制输出的格式。例如,你可以使用std::setw
和std::setfill
来设置字段宽度和填充字符,或者使用std::fixed
和std::setprecision
来设置浮点数的输出格式。
示例(设置字段宽度和填充字符):
cpp
#include <iostream>
#include <iomanip> // 引入iomanip头文件以使用setw和setfill
int main() {
int num = 123;
std::cout << std::setw(5) << std::setfill('0') << num << std::endl; // 输出:00123
return 0;
}
示例(设置浮点数的输出格式):
cpp
#include <iostream>
#include <iomanip> // 引入iomanip头文件以使用fixed和setprecision
int main() {
double pi = 3.14159265358979323846;
std::cout << std::fixed << std::setprecision(2) << pi << std::endl; // 输出:3.14
return 0;
}