C++ 类的成员函数
一、普通成员函数
1、普通成员函数的声明
普通成员函数必须在类中声明,声明方式与声明普通函数的方式一样
class People
{
public:
void getWeight();
void getHeight(); //成员函数
};
2、普通成员函数的定义方式
成员函数的定义有两种方式:
1)、在类中定义,此时该函数默认为内联函数;
2)、在类外定义,但要使用作用域限定符(::)
class Rectangle
{
private:
float height;
float width; //数据成员
public:
float getWidth()
{
return width; //类中定义,默认为内联函数
}
float getHeight();
};
//类外定义
float Rectangle::getHeight()
{
return height;
}
3、普通成员函数的调用方式
1)、通过对象使用成员运算符(.)调用;
2)、通过对象指针使用间接成员运算符(->)调用
二、构造函数
1、构造函数的分类
1)、自定义构造函数(可有参数也可以没有参数);
2)、默认构造函数;
3)、拷贝构造函数
当没有自定义构造函数时,编译器会提供一个默认的构造函数;当自定义构造函数后,编译器不再提供默认构造函数
2、构造函数的特点
1)、构造函数没有返回值和声明类型;
2)、构造函数名与类名相同;
3)、构造函数可以有参数也可以没有参数;
4)、构造函数的函数体可以为空也可以不为空;
5)、构造函数可在类中定义也可在类外定义;
6)、构造函数不能声明为const型的;
7)、构造函数的参数可以有默认值;
8)、类可以有多个构造函数
3、构造函数的作用
用来初始化对象的数据成员
4、构造函数何时调用
构造函数在创建对象时自动调用
5、构造函数的调用方式
1)、显式调用
2)、隐式调用
6、构造函数的定义方式
1)、在类中定义;
2)、可在类外定义,此时需要使用作用域限定符(::)
7、构造函数初始化成员的方式
1)、在函数体内一一赋值初始化;
2)、采用初始化列表初始化
赋值初始化示例:
#include <iostream>
using namespace std;
class Student
{
private:
int m_age;
float m_score;
public:
Student() { }
Student(int age, float score)
{
m_age = age;
m_score = score;
}
void show()
{
cout << "age = " << m_age
<< " score = " << m_score
<< endl;
}
};
int main()
{
Student stu = Student(16, 90); //显式调用
stu.show();
Student stu1(18, 96); //隐式调用
stu1.show();
return 0;
}
初始化列表示例
#include <iostream>
using namespace std;
class Student
{
private:
int m_age;
float m_score;
public:
Student(int age, float score) : m_age(age), m_score(score)
{
}
void show()
{
cout << "age = " << m_age
<< " score = " << m_score
<< endl;
}
};
int main()
{
Student stu = Student(16, 90); //显式调用
stu.show();
Student stu1(18, 96); //隐式调用
stu1.show();
return 0;
}
(一)、自定义构造函数
函数原型
className(argument-list)
{
... //
}
(二)、默认构造函数
1、函数原型
className()
{
}
编译器提供的默认构造函数不接受任何参数,函数体也不执行任何操作;
2、自定义默认构造函数的方式:
1)、给已有的构造函数的所有参数提供默认值;
2)、通过函数重载定义一个没有参数的构造函数
默认构造函数可在类中定义也可在类外定义,在类外定义时需要使用作用域限定符(::)
例如:
class Rectangle
{
private:
float height;
float width;
public:
Rectangle(float w = 0, float h = 0); //默认构造函数定义方式1
Rectangle() //默认构造函数定义方式2
{
height = 0;
width = 0;
}
};
3、默认构造函数的注意事项:
1)、如果在类中没有显式定义构造函数,则C++自动提供默认构造函数;
2)、如果显式定义了构造函数,则C++不会再提供默认构造函数,必须为类提供一个默认构造函数;
3)、默认构造函数只能有一个;
4)、设计类时,通常应提供对所有类成员做隐式初始化的默认构造函数;
5)、默认构造函数初始化对象时对象的初始值是未知的
4、默认构造函数的作用
有了默认构造函数(没有参数或所有参数都有默认值),在创建对象的时候就不必初始化对象
#include <iostream>
using namespace std;
class Student
{
private:
int m_age;
float m_score;
public:
Student(){}
Student(int age, float score)
{
m_age = age;
m_score = score;
}
void show()
{
cout << "age = " << m_age
<< " score = " << m_score
<< endl;
}
};
int main()
{
Student stu; //有了默认构造函数就可以先定义对象再进行初始化,否则就必须在定义的同时进行初始化: Student stu = Student(16, 90);
stu = Student(16, 90);
stu.show();
return 0;
}
(三)、拷贝构造函数
类 T 的拷贝构造函数是非模板构造函数,其首个形参为 T&、const T&、volatile T& 或 const volatile T&,而且要么没有其他形参,要么剩余形参均有默认值
当没有自定义拷贝构造函数时,编译器会提供一个默认的拷贝构造函数;当自定义拷贝构造函数后,编译器不再提供默认拷贝构造函数
1、拷贝构造函数的类型
拷贝构造函数分为两类:
1)、默认拷贝构造函数
2)、自定义拷贝构造函数
2、拷贝构造函数的功能
逐个复制非静态成员的值
3、拷贝构造函数何时调用
拷贝构造函数在以下三种情况下调用:
1)、用类的一个对象初始化另一个对象;
2)、将对象作为实参传递给一个非引用类型的形参(即按值传递);
3)、函数返回对象
4、默认拷贝构造函数
函数原型
className(const className &);
拷贝构造函数说明:
1)、该函数接收一个指向类类型的引用作为参数;
2)、当没有自定义拷贝构造函数时,编译器会提供一个默认的拷贝构造函数作为其类的非 explicit 的 inline public 成员;当自定义拷贝构造函数后,编译器不再提供默认拷贝构造函数;
3)、默认拷贝函数对成员的复制称之为浅拷贝
5、自定义拷贝构造函数
什么情况下需要自定义拷贝构造函数?
当类中有需要用new来分配存储空间的成员时,需要自定义拷贝构造函数,此时的拷贝称之为深拷贝
#include <iostream>
#include <cstring>
using namespace std;
class Str
{
public:
char *p;
public:
Str(char *s)
{
int len = strlen(s);
p = new char[len + 1];
strcpy(p, s);
cout << "调用构造函数" << endl;
}
Str(const Str &s)
{
int len = strlen(s.p);
p = new char[len + 1];
strcpy(p, s.p);
cout << "调用拷贝构造函数" << endl;
}
~Str()
{
if (p != NULL)
{
delete [] p;
}
cout << "调用析构函数" << endl;
}
};
int main()
{
char s[] = "Hello";
Str s1(s);
Str s2 = s1;
cout << s1.p << endl;
cout << s2.p << endl;
return 0;
}
四、析构函数
1、析构函数分类
1)、自定义的析构函数;
2)、默认的析构函数
当没有自定义析构函数时,编译器会提供一个默认的析构函数;当自定义析构函数后,编译器不再提供默认拷贝构造函数
2、析构函数的特点
1)、析构函数声明时须在函数名前面加上~;
2)、析构函数没有参数,故不能重载,故类只能有一个析构函数;
3)、析构函数没有返回值和声明类型;
4)、若在类中没有显式定义析构函数,C++会自动提供一个默认的析构函数;若显式定义了析构函数,则不会再提供默认的析构函数;
5)、若某个类作为基类,则一般将析构函数声明为虚析构函数
6)、析构函数可以声明为纯虚,例如对于需要声明为抽象类,但没有其他可声明为纯虚的适合函数的基类;
7)、析构函数可在类中定义也可在类外定义,在类外定义时需要使用作用域限定符(::)
3、析构函数的作用
析构函数用来销毁对象和为对象的数据成员使用new分配的内存空间
4、析构函数何时调用
以下情况下,析构函数是由编译器决定调用
1)、若创建的是静态存储类对象,则析构函数在程序运行结束时被自动调用;
2)、若创建的是自动存储类对象,则析构函数在程序执行完对象所在的代码块时被自动调用;
3)、若对象是通过new创建的,则析构函数在使用delete释放对象占用的内存时被自动调用
4)、若创建的是临时对象,则程序在结束对该对象的使用时调用析构函数
5、默认析构函数
函数原型
~className()
{
}
6、自定义析构函数
1)、函数原型
~className()
{
}
virtual ~className() //类作为基类时的通常设为virtual
{
}
示例:
#include <iostream>
using namespace std;
class Student
{
public:
Student(){}
~Student(){cout << "对象被销毁" << endl;}
};
int main()
{
{
Student stu;
Student *sp = new Student;
delete sp;
}
return 0;
}
运行结果:
对象被销毁
对象被销毁
2)、何时需要自定义析构函数
在以下情况下需要自定义的析构函数;
1)、当类中有需要用new来分配存储空间的成员时;
2)、当某个类作为另一个类的基类
五、赋值运算符重载函数
1、赋值运算符的分类
1)、自定义的赋值运算符
2)、编译器提供的默认的赋值运算符
当没有自定义赋值运算符时,编译器会提供一个默认的赋值运算符;当自定义赋值运算符后,编译器不再提供默认赋值运算符
2、赋值运算符的功能
与拷贝构造函数相似,赋值运算符也对非静态成员进行逐个复制,如果成员本身是类对象,则程序将使用为这个类定义的赋值运算符来复制该成员,但静态数据不受影响
3、何时使用赋值运算符
将已有的对象赋给另一个对象时
MyClass foo;
MyClass bar (foo); // 初始化对象,调用拷贝构造函数
MyClass baz = foo; // 初始化对象,调用拷贝构造函数
foo = bar; // bar已经被初始化: 调用赋值运算符
4、默认的赋值运算符
函数原型
className & operator=(const className &);
5、自定义的赋值运算符
什么时候需要自定义赋值运算符
1)、用非类类型的值为类类型的对象赋值时(当然,这种情况下我们可以不提供相应的赋值运算符重载函数,而只提供相应的构造函数);
2)、当类中含有指针成员,将同一个类的一个对象赋值给另一个对象时
六、移动构造函数(C++11)
1、函数原型
className (className &&);
类 T 的移动构造函数是非模板构造函数,其首个形参是 T&&、const T&&、volatile T&& 或 const volatile T&&,且无其他形参,或剩余形参均有默认值
2、移动构造函数的原理
拷贝构造函数中,对于指针要采用深拷贝,而移动构造函数中,对于指针采用浅拷贝。类中有指针时浅拷贝之所以危险,是因为两个指针共同指向同一片内存空间,若将第一个指针释放,则另一个指针的指向就合法了。
3、何时调用移动构造函数
同拷贝构造函数
4、隐式声明的移动构造函数
若不对类提供任何用户定义的移动构造函数,且下列各项均为真:
1)、没有用户声明的复制构造函数;
2)、没有用户声明的复制赋值运算符;
3)、没有用户声明的移动赋值运算符;
4)、没有用户声明的析构函数;
则编译器将声明一个移动构造函数,作为其类的非 explicit 的 inline public 成员
#include <iostream>
#include <cstring>
using namespace std;
class Str
{
public:
char *p;
public:
Str(char *s)
{
int len = strlen(s);
p = new char[len + 1];
strcpy(p, s);
cout << "调用构造函数" << endl;
}
Str(Str &&s) //移动构造函数
{
p = s.p;
s.p = nullptr;
cout << "调用移动构造函数" << endl;
}
~Str()
{
delete [] p;
cout << "调用析构函数" << endl;
}
};
int main()
{
char s[] = "Hello";
Str s1(s);
Str s2(move(s1));
cout << s2.p << endl;
return 0;
}
七、移动构赋值运算符函数(C++11)
1、函数原型
className &operator=(className &&);
2、何时调用移动构赋值运算符函数
同赋值运算符
3、移动赋值运算符的作用
执行与移动构造函数相同的工作
4、隐式声明的移动赋值运算符
若不对类类型(struct、class 或 union)提供任何用户定义的移动赋值运算符,且下列各项均为真:
没有用户声明的复制构造函数;
没有用户声明的移动构造函数;
没有用户声明的复制赋值运算符;
没有用户声明的析构函数;
隐式声明的移动赋值运算符不会被定义为弃置,
(C++14 前)
则编译器将声明一个移动赋值运算符,作为其类的 inline public 成员
#include <iostream>
#include <cstring>
using namespace std;
class Str
{
public:
char *p;
public:
Str()
{
}
Str(char *s)
{
int len = strlen(s);
p = new char[len + 1];
strcpy(p, s);
cout << "调用构造函数" << endl;
}
Str(Str &&s)
{
p = s.p;
s.p = nullptr;
cout << "调用移动构造函数" << endl;
}
Str &operator=(Str &&s)
{
p = s.p;
s.p = nullptr;
cout << "调用移动赋值运算符" << endl;
}
~Str()
{
delete [] p;
cout << "调用析构函数" << endl;
}
};
int main()
{
char s[] = "Hello";
Str s1(s);
Str s3;
Str s2(move(s1));
s3 = move(s2);
cout << s3.p << endl;
return 0;
}
八、静态成员函数(static成员函数)
1、函数原型
static type functionName(arguments-list)
{
... //函数体
}
2、静态成员函数的特性
1)、静态成员遵循类成员访问规则(私有、保护、公开);
2)、静态成员函数不关联到任何对象,即使不定义类的任何对象它们也存在,它们无 this 指针,只能访问静态成员(包括静态成员变量和静态成员函数);
3)、静态成员函数不能为 virtual、const 或 volatile;
4)、静态成员函数的地址可以存储在常规的函数指针中,但不能存储于成员函数指针中;
5)、静态数据成员不能为 mutable;
6)、在命名空间作用域中,若类自身具有外部连接(即不是无名命名空间的成员),则类的静态数据成员也具有外部连接。局部类(定义于函数内部的类)和无名类,包括无名类的成员类,不能拥有静态数据成员;
7)、静态成员函数在类外定义时不需要加关键字static
#include <iostream>
using namespace std;
class Student
{
public:
Student(char *name, int age, float score);
void show();
public: //声明静态成员函数
static int getTotal();
static float getPoints();
private:
static int m_total; //总人数
static float m_points; //总成绩
private:
char *m_name;
int m_age;
float m_score;
};
int Student::m_total = 0;
float Student::m_points = 0.0;
Student::Student(char *name, int age, float score): m_name(name), m_age(age), m_score(score)
{
m_total++;
m_points += score;
}
void Student::show(){
cout<<m_name<<"的年龄是"<<m_age<<",成绩是"<<m_score<<endl;
}
//定义静态成员函数
int Student::getTotal(){
return m_total;
}
float Student::getPoints(){
return m_points;
}
int main()
{
(new Student("小明", 15, 90.6)) -> show();
(new Student("李磊", 16, 80.5)) -> show();
(new Student("张华", 16, 99.0)) -> show();
(new Student("王康", 14, 60.8)) -> show();
int total = Student::getTotal();
float points = Student::getPoints();
cout<< "当前共有" << total
<< "名学生,总成绩是" << points
<< ",平均分是" << points / total << endl;
return 0;
}
运行结果:
小明的年龄是15,成绩是90.6
李磊的年龄是16,成绩是80.5
张华的年龄是16,成绩是99
王康的年龄是14,成绩是60.8
当前共有4名学生,总成绩是330.9,平均分是82.725
九、常成员函数(const成员函数)
1、函数原型
type functionName(arguments-list) const;
2、常成员函数的特性
1)、常成员函数需要在声明和定义的时候在函数头部的结尾加上 const 关键字
2)、常对象只能调用常成员函数,普通对象可以调用所有成员函数;
3)、常成员函数中的const修饰的是该成员函数隐含的this指针,常成员函数中不能改变数据成员的值
4)、const成员函数既可使用const数据,也可使用非const数据,但都不能改变值
#include <iostream>
#include <cstring>
using namespace std;
class Test
{
private:
int m_a;
int m_b;
public:
Test(int a, int b) : m_a(a), m_b(b)
{
}
int getA() const
{
return m_a;
}
int getB() const
{
return m_b;
}
};
int main()
{
Test test(1, 2);
int a = test.getA();
int b = test.getB();
cout << "a = " << a << endl;
cout << "b = " << b << endl;
return 0;
}
3、常成员函数的作用
常成员函数中不得修改类中的任何数据成员的值
例如:
class Complex
{
public:
Complex();
Complex(int real, int image);
virtual ~Complex();
//编译出错,因为修改了数据成员,去掉const就正确
void Print() const
{
cout << "real = " << ++m_real << " image = " << ++m_image << endl;
}
//正确
void Print() const
{
cout << "real = " << m_real << " image = " << m_image << endl;
}
private:
int m_real;
int m_image;
};
十、三五法则
1、三法则
如果需要析构函数,则一定需要拷贝构造函数和拷贝赋值操作符
2、五法则
在 C++11 标准中,为了支持移动语义,又增加了移动构造函数和移动赋值运算符,这样共有五个特殊的成员函数,所以又称为“C++五法则”;
3、三五法则
三法则”是针对较旧的 C++89 标准说的,“五法则”是针对较新的 C++11 标准说的;为了统一称呼,后来人们干把它叫做“C++ 三/五法则”
1). 需要析构函数的类也需要拷贝构造函数和拷贝赋值函数;
2). 需要拷贝操作的类也需要赋值操作,反之亦然;
3). 析构函数不能是删除的;
4). 如果一个类有删除的或不可访问的析构函数,那么其默认和拷贝构造函数会被定义为删除的;
5). 如果一个类有const或引用成员,则不能使用合成的拷贝赋值操作
注:如果在类中没有显式定义以下函数,则编译器自动提供
1)、构造函数
2)、析构函数
3)、拷贝构造函数
4)、赋值运算符
5)、地址运算符
参考:
1、《C++ Primer Plus》
2、《Primer C++》
3、https://zh.cppreference.com
4、http://c.biancheng.net/view/2230.html
5、C++ 三五法则