参见:面相对象的程序设计
在C++中,运算符重载是一种形式的多态,允许开发者为已有的运算符赋予自定义的行为。运算符重载的实质是函数重载。重载运算符可以是成员函数或全局函数(友元函数),但必须至少有一个操作数是用户定义的类型。
.
、 ::
、 ?:
和 sizeof
。=
,应该通常作为类的成员函数来重载。ReturnType operator${符号}(params...)
{
// Do sth...
return ...;
}
${符号}
为需要重载的运算符,前后可以加空格。
可以重载的运算符有:
算数运算符 | + 、 - 、 * 、 / 、 % |
关系运算符 | == 、 != 、 < 、 > 、 <= 、 >= |
逻辑运算符 | |
赋值运算符 | |
位运算符 | |
单目运算符 | |
自增、自减运算符 | |
动态内存操作运算符 | |
其他运算符 |
Dataview (inline field '='): Error: -- PARSING FAILED -------------------------------------------------- > 1 | = | ^ Expected one of the following: '(', 'null', boolean, date, duration, file link, list ('[1, 2, 3]'), negated field, number, object ('{ a: 1, b: 2 }'), string, variable
不可重载的运算符有:
.
->
::
? :
sizeof
假设我们需要对如下的虚数类实现其加法运算:
class Complex {
private:
double real;
double image;
public:
Complex(double real, double image);
std::string to_string(void);
};
Complex::Complex(double real, double image)
{
this->real = real;
this->image = image;
}
std::string Complex::to_string(void)
{
using namespace std;
string symbol = "";
image >= 0 ? symbol = "+" : symbol = "-";
string str = std::format("{}{}{}i\r\n", real, symbol, fabs(image));
return str;
}
基于上述类,可以基于上述类和友元函数实现全局函数定义的运算符 +
的重载,Demo如下:
class Complex {
private:
double real;
double image;
public:
Complex(double real, double image);
std::string to_string(void);
friend Complex operator+(const Complex& comp1, const Complex& comp2);
};
Complex::Complex(double real, double image)
{
this->real = real;
this->image = image;
}
std::string Complex::to_string(void)
{
using namespace std;
string symbol = "";
image >= 0 ? symbol = "+" : symbol = "-";
string str = std::format("{}{}{}i\r\n", real, symbol, fabs(image));
return str;
}
Complex operator+(const Complex& comp1, const Complex& comp2)
{
return Complex(comp1.real + comp2.real, comp1.image + comp2.image);
}
与上述友元函数的运算符重载不同的是,成员函数实现的运算符重载不再需要额外传递一次自身。即成员函数实现的运算符重载会比友元函数少一个函数参数。
如果需要重载的运算符为双目运算符,则只需要设置一个参数作为右侧运算量。
如果需要重载的运算符为单目运算符,则不需要另外设置参数,使用自身进行运算即可。
因此,可以基于上述类和成员函数实现运算符 ==
的重载,Demo如下:
Dataview (inline field '='): Error: -- PARSING FAILED -------------------------------------------------- > 1 | = | ^ Expected one of the following: '(', 'null', boolean, date, duration, file link, list ('[1, 2, 3]'), negated field, number, object ('{ a: 1, b: 2 }'), string, variable
class Complex {
private:
double real;
double image;
public:
Complex(double real, double image);
std::string to_string(void);
bool operator==(const Complex& comp);
};
bool Complex::operator==(const Complex& comp)
{
return (this->real == comp.real && this->image == comp.image);
}
直接使用对若干个对象进行运算就是运算符的隐式调用,例如:
Complex comp1(1, -1);
Complex comp2(-2, 3);
Complex comp_sum = comp1 + comp2;
而除了显示调用以外,还有如下的隐式调用方式:
Complex comp1(1, -1);
Complex comp2(-2, 3);
// 全局函数重载的显示调用方式:
Complex comp_sum = operator+(comp1, comp2);
// 成员函数重载的显示调用方式
comp1.operator==(comp2);
需要注意的是,函数重载的方式不同,其对应的显示调用方式也不同。
在x86架构下,sizeofstring = 28
;
在x86_64架构下,sizeofstring = 40
;
而 sizeofstring
的值不随字符串内容发生改变。
string可以作为struct的成员,其size计算符合内存对齐等要求。
string(const char *s); |
构造方法,用 c_str 初始化 |
|
string(int n,char c); |
构造方法,构造一个含有 n 个 c 的字符串 |
|
STL全名为Standard Template Library,意为标准模板库或泛型库,是C++中的一个重要组件。其主要包含如下组件:
STL容器主要有如下三类:
std::array
std::vector
std::deque
std::list
std::set
std::multiset
std::map
std::multimap
std::unordered_set
std::unordered_multiset
std::unordered_map
std::unordered_multimap
std::vector
是C++的动态大小的数组实现,其元素被顺序存储,因此其可以被迭代器和引索顺序访问。其会自动扩展其所需要的内存空间,并且通常其所占用的内存比同大小的静态数组要多。其空间的动态分配仅会发生在其所保留的额外空间耗尽时触发。
vector的常用操作的时间复杂度:
vector中的模板类型需要满足如下要求:
但是需要注意慎用bool类型作为vector的元素,除非明确地要使用 vector<bool>
的特性。
方法 | 含义 | 备注 |
---|---|---|