C语言中文网
菜鸟教程 - C++教程
c-cpp.com
C++ interview
在C++中,尽量不使用try-catch异常处理,因为开销比较大。
C语言源文件要经过编译、链接才能生成可执行程序:
编译(Compile)会将源文件(.c文件)转换成目标文件。对于VC/VS,目标文件后缀为 .obj
;对于 GCC,目标文件后缀为 .o
。
编译是针对单个源文件的,一次编译操作只能编译一个源文件,如果程序中有多个源文件,就需要多次编译操作。
链接(Link)是针对多个文件的,它会将多个目标文件以及系统中的库、组件等合并成一个可执行程序。
预处理就是处理以 #
开头的命令,例如 #include
等。预处理命令要放在所有函数之外,而且一般都放在源文件的前面。
编译器会将预处理的结果保存到和源文件同名的 .i
文件中,例如 main.c 的预处理结果在 main.i 中。和 .c
一样, .i
也是文本文件,可用编辑器打开查看内容。
举个例子,假如现在开发一个C语言程序,让它暂停5s以后再输出内容,并且要求跨平台,在Windows和Linux下都能运行,怎么办呢?
不同平台下的暂停函数和头文件都不一样:
void Sleep(DWORD dwMilliseconds)
(注意S是大写的),参数的单位是 “毫秒”,位于 uunsigned int sleep (unsigned int seconds)
,参数的单位是 “秒”, 位于 不同的平台下必须调用不同的函数,并引入不同的头文件,否则就会导致编译错误,因为 Windows 平台下没有 sleep() 函数,也没有
#include
//不同的平台下引入不同的头文件
#ifdef _WIN64 //识别windows平台
#include
#elif __linux__ //识别linux平台
#include
#endif
int main() {
//不同的平台下调用不同的函数
#if _WIN64 //识别windows平台
Sleep(5000);
#elif __linux__ //识别linux平台
sleep(5);
#endif
puts("http://c.biancheng.net/");
return 0;
}
#if
、#elif
、#endif
就是预处理命令,它们都是在编译之前由预处理程序来执行的。
对于 Windows 平台,预处理之后的代码变成:
#include
#include
int main() {
Sleep(5000);
puts("http://c.biancheng.net/");
return 0;
}
对于 Linux 平台,预处理之后的代码变成:
#include
#include
int main() {
sleep(5);
puts("http://c.biancheng.net/");
return 0;
}
总结:在不同平台下,编译之前(预处理之后)的源代码都是不一样的。这就是预处理阶段的工作,它把代码当成普通文本,根据设定的条件进行一些简单的文本替换,将替换以后的结果再交给编译器处理。
一般来说,头文件提供接口,源文件提供实现。
编译器规定源文件必须包含函数入口,即 main 函数。头文件专为源代码调用而写的 静态包含文件,可被源代码文件中 #include
编译预处理指令解释。如果将头文件完整拷贝到源代码的指令处,那么编译时相当于在源代码中插入 函数声明。
C++ 编译规则:头文件不会参与编译,每个cpp单独编译,每个cpp即为一个编译单元。编译期间,每个cpp不需要知道其他 cpp 存在,只有到链接阶段才会将编译期间生成的 obj 连接成一个 exe 或者 out 文件。
头文件用来写 类的声明(包括声明类的成员属性和成员方法)、函数原型、#define 常数等。
头文件的格式
#ifndef MYCLASS_H
#define MYCLASS_H
// code here
#endif
#ifndef MYCLASS_H 的意思是 if not define myclass.h
如果引用这个头文件的源文件不存在 myclass.h 这个头文件,那么接下行 #define MYCALSS_H, 引入myclass.h。如果已经引入,直接跳到 #endif。
按照这种格式,目的是为了 防止头文件被重复引用。避免同一个头文件在同一个源文件中被 include 多次,这种错误称为“include嵌套”。例如,存在 cellphone.h 这个头文件引用了 #include “huawei.h”,之后又有 chain.cpp 源文件同时引用了 #include “cellphone.h” 和 #include “huawei.h”,此时 huawei.h 头文件在 chain.cpp 中被引用了两次。
理论上老说,MYCLASS_H 可以任意命名。但为了提高可读性,约定成俗地把头文件命名为 大写和下划线的形式。
#ifndef HUAWEI_H // 防止huawei.h被重复引用
#define HUAWEI_H
#include // 引用标准库
#include "honor.h" // 引用非标准库头文件
...
void Function(); // 全局函数声明
class Mate20{ // 类声明
public: Mate20(); // 构造函数声明
~Mate20(); // 析构函数声明
private:
protected:
};
#endif
shared_ptr
智能指针class object
{
private:
int value;
public:
object(int x = 0) :value(x) {}
~object() {}
int& Value() { return value; }
const int& Value( ) const { return value; }
};
int main()
{
shared_ptr
shared_ptr 是一种引用计数型智能指针(smart pointer),包含两个元素:指针、引用计数。所谓引用计数(reference counting),记录 有多少个 shared_ptrs 共同指向一个对象。一旦最后一个这样的指针被销毁,即 某个对象的引用计数为0,则这个对象会被自动删除,这在非环形数据结构中防止资源泄露是很有帮助的。
注意:如果多线程对同一个 shared_ptr 对象进行读和写,则必须加锁,否则容易造成“空悬指针”的后果。多线程读写 shared_ptr 所指向的对象,不管是相同的 shared_ptr 对象,还是不同的 shared_ptr 对象,都需要 加锁保护。
// 构造函数无参数
shared_ptr pCameraManager = make_shared();
// 构造函数有参数
shared_ptr pIoManager = make_shared(ioCardName);
shared_ptr apa(new object(10))
需要为 Object 对象和 RefCnt 对象各分配一次内存。printf是 print format 的缩写,表示 “格式化打印”,即在屏幕上格式化输出(显示)。
printf 比 puts 更加强大,不仅可以输出字符串,还可以输出整型、小数、单个字符等,输出的格式也可以自定义,例如:
%d
,d是 decimal 的缩写,表示十进制数,%d 表示以十进制整型的格式输出。
在 puts 函数中,可以将一个较长的字符串分割成几个较短的字符串,这样使得长文本的格式更加整齐。
#include
int main()
{
puts(
"C语言中文网,一个学习C语言和C++的网站,他们坚持用工匠的精神来打磨每一套教程。"
"坚持做好一件事情,做到极致,让自己感动,让用户心动,这就是足以传世的作品!"
"C语言中文网的网址是:http://c.biancheng.net"
);
return 0;
}
注意:这只是形式上的分割,编译器在编译阶段将会合并为一个字符串,并放在一块连续的内存中。
说明 | 字符型 | 字符串 | 短整型 | 整型 | 长整型 | 单精度浮点型 | 双精度浮点型 | 无类型 |
---|---|---|---|---|---|---|---|---|
类型 | char | string | short | int | long | float | double | void |
长度 | 1 | ~ | 2 | 4 | 4 | 4 | 8 | |
输出 | %c | %s | ~ | %d | ~ | %f | ~ |
#define,防止头文件被多重包含。#define头文件保护命名,全大写,例如:
#ifndef XJ_APP_SERVER_H
#define XJ_APP_SERVER_H
……
#endif // XJ_APP_SERVER_H
将头文件包含次序标准化,可增加可读性,次序如下:
C库头文件 ---》 QT/C++库头文件 ---》 其他库头的文件 ---》 项目内的头文件
C++命名规范
c++编程命名规范
C_C++变量命名规则
类名是 名词,每个单词以大写字母开头,不包含下划线,且名称前加大写字母C,例如:
CXJAppServer
CWebServer
函数名是 “动词” 或 “动词+名词”;
取值与设值函数与变量名匹配,例如:
int index_;
int GetIndex()
{
returnindex_;
};
void SetIndex(int _index)
{
index_ =_index;
};
函数的名称由一个或多个单词组成,例如:“GetName()”,“SetValue()”;
回调函数结尾+CallBack,例如:NotifyCallBack();
事件函数结尾+Event,例如:ModifyEvent();
信号、槽函数:
signals:
void askIndexSignal();
private slots:
void setIndexSlot();
全大写,单词间用_分开,例如:
const string MAX_FILENAME255;
全大写,单词间用_分开,例如:
#define PI_RAUD3.14159265
变量的命名 | 变量名由作用域前缀+类型前缀+一个或多个单词组成。为便于界定,每个单词的首字母要大写。 对于某些用途简单明了的局部变量,也可以使用简化的方式,如:i, j, k, x, y, z .... |
||||||||||||||||||||
作用域前缀 | 作用域前缀标明一个变量的可见范围。作用域可以有如下几种:
|
||||||||||||||||||||
类型前缀 | 类型前缀标明一个变量的类型,可以有如下几种:
|
||||||||||||||||||||
推荐的组成形式 | 变量的名字应当使用"名词"或者"形容词+名词"。例如:"nCode", "m_nState","nMaxWidth" .... |
.h
头文件对应的 .cpp
源文件有相同的文件名。
C++ 信号处理
信号是由操作系统传给进程的中断,会提早终止一个程序。在UNIX、Linux、Mac OX或 Windows系统上,通过按 Ctrl+C 产生中断。有些信号不能被程序捕获,但是下表所列信号可以在程序中捕获,并可以基于信号采取适当的动作,这些信号定义在 C++ 头文件 中。
信号 | 描述 |
---|---|
SIGABRT | 程序的异常终止,如调用 abort。 |
SIGFPE | 错误的算术运算,比如除以零或导致溢出的操作。 |
SIGILL | 检测非法指令。 |
SIGINT | 程序终止(interrupt)信号。 |
SIGSEGV | 非法访问内存。 |
SIGTERM | 发送到程序的终止请求。 |
vector> ret;
ret.push_back(1,1)//会报错,因为没有构造一个临时对象
ret.push_back(pair(1,1))//不会报错,可以构成了一个pair对象
ret.emplace_back(1,1)//不会报错,可以直接在容器的尾部创建对象
push_back()
:先向容器尾部添加一个右值元素(临时对象),然后调用 构造函数 构造出这个临时对象,最后调用 移动构造函数 将这个临时对象放入容器中,并释放这个临时对象。简单理解,分为两步:(1)构造临时对象,(2)移动临时对象。
最后调用的不是拷贝构造函数,而是 移动构造函数。因为需要释放临时对象,所以通过 std::move 进行移动构造,可以避免不必要的拷贝操作。
emplace_back()
:在容器尾部添加一个元素,调用 构造函数 原地构造,不需要触发拷贝构造和移动构造,因此比 push_back()
更加高效。
push_back() 只接收一个传参,即 push_back只接受对象(实例);emplace_back() 接受一个参数列表,即 emplace_back() 除了接受对象,还能接受构造函数的参数。emplace_pack() 仅通过使用 构造参数 传入参数的时候更高效。
detectorThreads.emplace_back(&XJAppServer::startDetector, &xjServer, detectorIdx);
emplace_back():
1) 调用 有参构造函数
push_back():
1) 调用 有参构造函数,创建临时对象;
2) 调用 移动构造函数,移动到 vector 中;
3) 调用 析构函数, 销毁临时对象
thread
线程(1)默认构造函数 | thread() |
---|---|
(2)初始化构造函数 | template |
(3)拷贝构造函数 | thread(const thread&) = delete |
(4)move构造函数 | thread(thread&& x) |
当前线程阻塞,等待子线程结束。
当前线程和子线程分离,不必等待子线程结束,即子线程变成守护线程。
获取线程id。
如果一个线程正在执行,那么它是 joinable 的。
下列任一情况,都是非 joinable 的:
C++ 中using 的使用
using的作用:
#include
using namespace std; // 引入命名空间
class ClassOne
{
public:
int w;
protected:
int a;
};
class ClassTwo
{
public:
using ModuleType = ClassOne; // 指定别名
};
template
class ClassThree : private ClassType
{
public:
using typename ClassType::ModuleType; // 在子类中引用基类的成员
ModuleType m;
ClassThree() = default;
virtual ~ClassThree() = default;
};
void main()
{
ClassThree::ModuleType a;
}
using namespace std;
指定别名,一般都是 using a = b
这样的形式
// ModuleType 是ClassOne的一个别名
using ModuleType = ClassOne;
// value_type 是_Ty的一个别名, `value_type a` 和 `_Ty a` 是同样的效果。
template>class vector: public _Vector_alloc<_Vec_base_types<_Ty, _Alloc>>
{
public:
using value_type = _Ty;
using allocator_type = _Alloc;
}
在子类中引用基类的成员,一般都是 using CBase::a
的形式。
/*
因为类ClassThree是个模板类,它的基类是 ClassType,需要加 typename 修饰,
这个 typename 和 using 本身没什么关系。
如果 ClassType不是模板类,这行代码可以写成:
using ClassType::ModuleType;
*/
using typename ClassType::ModuleType;
标准库里面的每个命名空间代表了一个的独立的概念。
C++11计时器:chrono库介绍
chrono库是一个模板库,使用简单,功能强大,只需要理解三个概念:duration
、time_point
、clock
。
#include
using namespace std;
CLOCK
时钟chrono库定义了三种不同的时钟:
// 依据系统的当前时间(不稳定)
std::chrono::system_clock;
// 以统一的速率运行(不能被调整)
std::chrono::steady_clock;
// 提供最高精度的计时周期
std::chrono::high_resolution_clock;
三种时钟的区别:
ratio
时间比率问题引入
时间精度,即时间分辨率。抛开 时间量纲 单论分辨率,就是一个比率。如:1000/1、10/1、1/1、1/10、1/1000。
这些比率加上距离量纲就变成了距离分辨率,加上时间量纲就变成了 时间分辨率。为此,C++11定义了 ration
模板类,用于表示比率,定义如下:
std::ratio
表示时钟周期,时间单位为秒。
ratio 是一个分数类型的值,其中 N 表示分子(秒),D表示分母(周期)。
常用的时间单位
ratio<3600, 1> hours (3600秒为一个周期,表示一小时)
ratio<60, 1> minutes
ratio<1, 1> seconds
ratio<1, 1000> millisecond
ratio<1, 1000000> microseconds
ratio<1, 1000000000> nanosecons
duration
持续的时间std::chrono::duration
,表示持续的一段时间,单位是由 radio <60,1> 决定的,int 表示这段时间值的类型,函数返回的类型还是一个时间段 duration。
std::chrono::duration>
std::chrono::duration>
由于各种时间段 duration 表示不同,chrono库提供了 duration_cast 类型转换函数。
// 将 duration 转换成另一种类型的 duration
duration_cast();
// 表示一段时间的长度
count();
#include
#include
#include
using namespace std::chrono;
using namespace std;
int main()
{
auto start = steady_clock::now();
for(int i=0;i<100;i++)
cout<<"nice"<(end - start);
cout<<"程序用时="<
time_point
时间点std::chrono::time_point()
表示一个具体时间,例如:上个世纪80年代,你的生日,今天下午,火车出发时间等。一个 time_point 必须有一个 clock 计时。
// 设置一个高精度时间点
time_point high_resolution_clock::now()
在C++中,模板分为函数模板和类模板两种。熟练的C++程序员,在编写函数时都会考虑能否将其写成 函数模板,编写类时都会考虑能否将其写成 类模板,以便实现重用。
一般来说,数据的值 可以通过 函数参数传递。在函数定义时数据的值是未知的,只有 等到函数调用时接收到了实参才能确定其值,这就是 值的参数化。
在 C++ 中,数据的类型 也可以通过参数来传递,在函数定义时可以不指明具体的数据类型,当发生函数调用时,编译器可以 根据传入的实参自动判断数据类型,这就是 类型的参数化。值(Value)和类型(Type)是数据的两个主要特征,在C++中都可以被参数化。
函数模板,实际上是建立一个 通用函数,不具体指定所用到的数据类型(包括返回值类型、形参类型、局部变量类型),而是用一个 虚拟的类型 来代替(实际上用一个 标识符 来占位),等发生函数调用时,再根据传入的实参来逆推出真正的类型,这个通用函数就称为 函数模板(Function Template)。简单理解为,使用泛型参数的函数(functions with generic parameters)。
在函数模板中,数据的值和类型都被参数化了,发生函数调用时编译器会根据传入的实参来推演形参的 值 和 类型 。换个角度说,函数模板除了支持值的参数化,还支持类型的参数化。
template
返回值类型 函数名(形参列表){
//在函数体中可以使用类型参数
}
template void Swap(T *a, T *b){
T temp = *a;
*a = *b;
*b = temp;
}
说明:template
是定义函数模板的关键字,后面紧跟尖括号 <>
,尖括号包围的是类型参数(虚拟的类型,即类型占位符)。typename
用来声明具体的类型参数,这里的类型参数就是 T
。从整体上来看,template
被称为 模板头。模板头和函数头是一个不可分割的整体,可以换行,都中间不能有分号。
模板头中包含的参数可以用在函数定义的各个位置,包括:返回值、形参列表和函数体;本例在形参列表和函数体中都使用了类型参数 T
。
函数模板被编译了两次:
//交换 int 变量的值
void Swap(int *a, int *b){
int temp = *a;
*a = *b;
*b = temp;
}
//交换 float 变量的值
void Swap(float *a, float *b){
float temp = *a;
*a = *b;
*b = temp;
}
//交换 char 变量的值
void Swap(char *a, char *b){
char temp = *a;
*a = *b;
*b = temp;
}
//交换 bool 变量的值
void Swap(bool *a, bool *b){
char temp = *a;
*a = *b;
*b = temp;
}
改成函数模板
#include
using namespace std;
template void Swap(T *a, T *b){
T temp = *a;
*a = *b;
*b = temp;
}
int main(){
//交换 int 变量的值
int n1 = 100, n2 = 200;
Swap(&n1, &n2);
cout<
引用 不但使得函数模板定义简洁明了,也使得调用函数方便很多,整体来看,引用让编码更加漂亮。
template void Swap(T *a, T *b){
T temp = *a;
*a = *b;
*b = temp;
}
改为
template void Swap(T &a, T &b){
T temp = a;
a = b;
b = temp;
}
#include
using namespace std;
template void Swap(T &a, T &b){
T temp = a;
a = b;
b = temp;
}
int main(){
//交换 int 变量的值
int n1 = 100, n2 = 200;
Swap(n1, n2);
cout<
C++ 除了支持函数模板,还支持 类模板(Class Template),类模板是使用泛型参数的类(classes with generic parameters)。
模板头和类头是一个不可分割的整体,可以换行,都中间不能有分号。
template
class 类名{
//TODO:
};
在类外定义成员函数时,需要带上模板头,格式为:
template
返回值类型 类名<类型参数1 , 类型参数2, ...>::函数名(形参列表){
//TODO:
}
类模板
template //这里不能有分号
class Point{
public:
Point(T1 x, T2 y): m_x(x), m_y(y){ }
public:
T1 getX() const; //获取x坐标
void setX(T1 x); //设置x坐标
T2 getY() const; //获取y坐标
void setY(T2 y); //设置y坐标
private:
T1 m_x; //x坐标
T2 m_y; //y坐标
};
类的成员函数
在类外定义成员函数时,template 后面的 类型参数 要和类声明时的一致。
template //模板头
T1 Point::getX() const /*函数头*/ {
return m_x;
}
template
void Point::setX(T1 x){
m_x = x;
}
template
T2 Point::getY() const{
return m_y;
}
template
void Point::setY(T2 y){
m_y = y;
}
与函数模板不同的是,类模板在实例化时必须显式地指明数据类型,编译器不能根据给定的数据推演出函数类型。
使用对象变量的方式来实例化
Point p1(10, 20);
Point p2(10, 15.5);
Point p3(12.4, "东经180度");
使用对象指针的方式实例化
Point *p1 = new Point(10.6, 109.3);
Point *p = new Point("东经180度", "北纬210度");
注意:赋值号两边都要指明具体的数据类型,且要保持一致。下面的写法是错误的:
//赋值号两边的数据类型不一致
Point *p = new Point(10.6, 109);
//赋值号右边没有指明数据类型
Point *p = new Point(10.6, 109);
泛化之美–C++11可变模版参数的妙用
C++ 运算符重载的基本概念
C++预定义的运算符,只能用于基本数据类型的运算,例如:int、char等,不能用于对象的运算。为了能在对象之间使用运算符,就需要重载运算符。例如,数学上两个复数可以直接进行 +
、-
等运算,但在C++中,直接将 +
或 -
用于复数对象是不允许的,这时需要对运算符进行重载。
c = a - b
,等价于 c = a.operator - (b)
;c = a + b
,等价于 c = operator + (a, b)
;class Complex // 复数类
{
public:
// 构造函数,如果不传参数,默认把实部和虚部初始化为0
Complex(double r = 0.0, double i = 0.0):m_real(r),m_imag(i) { }
// 重载-号运算符,属于成员函数
Complex operator-(const Complex & c)
{
// 返回一个临时对象
return Complex(m_real - c.m_real, m_imag - c.m_imag);
}
// 打印复数
void PrintComplex()
{
cout << m_real << "," << m_imag << endl;
}
// 将重载+号的普通函数,定义成友元函数
// 目的是为了友元函数能访问对象的私有成员
friend Complex operator+(const Complex &a, const Complex &b);
private:
double m_real; // 实部的值
double m_imag; // 虚部的值
};
// 重载+号运算符,属于普通函数,不是对象的成员函数
Complex operator+(const Complex &a, const Complex &b)
{
// 返回一个临时对象
return Complex(a.m_real + b.m_real, a.m_imag + b.m_imag);
}
int main()
{
Complex a(2,2);
Complex b(1,1);
Complex c;
c = a + b; // 等价于c = operator+(a,b)
c.PrintComplex();
c = a - b; // 等价于 c = a.operator-(b)
c.PrintComplex();
return 0;
}
输出结果:
3,3
1,1
// 重载-号运算符,属于成员函数
Complex Complex::operator-(const Complex & c)
{
// 返回一个临时对象
return Complex(m_real - c.m_real, m_imag - c.m_imag);
}
值得思考的问题:
const Complex & c
常引用类型,而不是 Complex
类型呢?Complex
对象,而不是 Complex &
呢?分析原因:
Complex
普通对象类型,在入参的时候,就会调用默认的拷贝构造函数,产生一个临时对象,这会增大开销,所以采用引用的方式。同时,为了防止引用的对象被修改,所以定义成了 const Complex & c
常引用类型。Complex
对象。所谓“泛型”,指的是算法只要 实现一遍,就能适用于多种数据类型,泛型的优势在于能够减少重复代码的编写。泛型程序设计(generic programming)是一种算法在实现时 不指定 具体要操作的 数据类型 的程序设计方法。
泛型是一种编程思想,不依赖于具体的编程语言。大多数面向对象的语言都支持泛型编程,例如:C++,C#,Java等。
C++里的泛型,通过模板以及相关性质表现的。
问题引入:
模板类 SigmaTraits
叫做 traits template
,它含有参数类型 T 的一个 特性(trait)
,即 ReturnType
。
template
inline typename SigmaTraits::ReturnType Sigma (const T const* start, const T const* end)
{
typedef typename SigmaTraits::ReturnType ReturnType;
ReturnType s = ReturnType();
while (start != end)
{
s += *start++;
}
return s;
};
template <template T> class SigmaTraits {};
template <> class SigmaTraits<char> {
public:typedef int ReturnType;
};
tempalte <> class SigmaTraits<short> {
public: typedef int ReturnType;
};
tempalte <> class SigmaTraits<int> {
public: typedef long ReturnType;
};
template <> class SigmaTraits<unsigned int> {
public: typedef unsigned long ReturnType;
};
template <> class SigmaTraits<float> {
public: typedef double ReturnType;
};
迭代器是泛化的指针(generalization of points)。在STL中迭代器是容器和算法之间的接口,算法通常以迭代器作为输入参数。
迭代器的基本思想:
C++(笔记)浅析vector容器的实例
C++容器
Vector是一个封装了动态大小数组的顺序容器,简单理解为能够存放任意类型的动态数组。Vector支持动态空间大小调整,随着元素增加,Vector内部会自动扩充内存空间。
vector::at();
进行边界检查,如果访问越界则抛出exception,访问效率不如 operator[]。
vector::operator[];
类似于数组操作,没有边界检查,访问效率高。
Deque是一个能够存放任意类型的双向队列。Deque提供的函数与Vector类似,新增了两个函数:
Deque采用了与Vector不同的内存管理方法:大块分配内存。
List是一个能够存放任意类型的 双向链表(double linked list)。