C++中的类与对象深度解析

 更新时间:2022年02月09日 14:42:27   作者:你算哪一个bug?  
这篇文章主要为大家详细介绍了C++中的类与对象,使用数据库,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

初始化列表

引论

//初始化列表的引出
class B
{
public:
	B()
	{
		cout << "我是B的构造函数" << endl;
	}
private:
	int _b;
};
class A
{
public:
	A()
	{
		cout << "我是A的构造函数" << endl;
	}
private:
	int _a;
	B b;
};
int main()
{
	A a;
	return 0;
}

image-20220203215753999

汇编验证:

image-20220203220240209

初始化列表

c++推荐使用初始化列表进行初始化

什么是初始化列表?

一个冒号开始,以逗号分隔的数据成员列表,每个成员变量后面放一个括号,括号中放初始值或者表达式

:_a(1)

,_aa(2)

这个就是初始化列表

//初始化列表使用demo
class A
{
public:
	A()
		:_a(1)//初始化列表
		, _aa(2)
	{
		cout << _a << " " << _aa << endl;
	}
private:
	int _a;
	int _aa;
};
int main()
{
	A a;
	return 0;
}

image-20220203221620952

  • 存在自定义类型时,初始化列表,不写也认为有

只是认为啊,不写初始化列表会在函数体之前调用自定义类型的构造函数,写了初始化列表是会进入函数体(花括号)然后调用,具体的可以观察汇编

  • 成员变量只能在初始化列表里出现一次
  • 成员变量包含引用和const变量时,必须在初始化列表里初始化

引用和const变量必须初始化

  • 成员变量的声明次序就是初始化顺序,与在初始化列表里先后顺序无关

image-20220203222641177

初始化列表是推荐使用的,如果初始化列表不能解决问题,混着用(在构造函数体内初始化)就行了

explicit关键字

引论

//int赋给对象demo
class A
{
public:
	A(int){}
};
int main()
{
	A a = 1;
	return 0;
}

image-20220203223944155

这么写是合法的,赋值的过程发生了隐式类型转换,前提是必须有A(int这样的构造函数)

//多个int赋给对象demo
class A
{
public:
	A(int,int){}
};
int main()
{
	A a = {1,2};
	return 0;
}

image-20220203224101796

C++11支持多参数转换,C++98不支持

explicit关键字使用

前面提到,int可以赋给对象是隐式类型转换,如果要禁止这种用法,则用explicit修饰对应的构造函数

//explicit使用demo
class A
{
public:
	explicit A(int){}
};
int main()
{
	A a = 1;//error
	return 0;
}

image-20220203224344113

static成员 

静态成员变量:static修饰的成员变量

静态成员函数:static修饰的成员函数

静态成员变量不是单单属于某一个对象的,一个类创建的多个对象使用这个静态成员变量时使用的也是同一块内存,即所有对象共有该静态成员变量

一份内存,多对象使用

静态成员函数一般用来访问静态成员,没有this指针

由于没有this指针,所以无法访问非静态的成员

计算类的大小时不包括静态成员

image-20220203225355745

计算类的大小可以认为是计算对象的大小,因为每个对象共有静态成员变量,所以不能认为该变量特定属于某一个对象

调用静态成员函数,初始化静态成员变量

//调用static函数和初始化static变量的democlass A{public:static int Print(){cout << "static int Print()" << endl;return _aa;}private:int _a;static int _aa;};int A::_aa = 1;int main(){A::Print();return 0;}//调用static函数和初始化static变量的demo
class A
{
public:
	static int Print()
	{
		cout << "static int Print()" << endl;
		return _aa;
	}
private:
	int _a;
	static int _aa;
};
int A::_aa = 1;
int main()
{
	A::Print();
	return 0;
}

image-20220203225932607

静态成员变量不能给缺省值,必须在类外初始化,因为在类外初始化时才分配空间,所以不在类外初始化就不能用,用了可能会导致链接错误

静态成员函数不能调用非静态成员函数,非静态成员函数可以调用静态成员函数

普通静态函数需要通过this指针调用,而静态成员函数没有this指针–百度

待了解:链接属性

友元

引论

image-20220203232439932

友元

什么是友元?

友元是一种定义在类外部的普通函数或类,但它需要在类体内进行说明,为了与该类的成员函数加以区别,在说明时前面加以关键字friend。–百度百科

友元的作用

突破封装

class Date
{
	friend bool operator==(Date d1, Date d2);

private:
	int _year;
	int _month;
	int _day;
};
bool operator==(Date d1,Date d2)
{
	return d1._year == d2._year
		&& d1._month == d2._month
		&& d1._day == d2._day;
}
int main()
{
	return 0;
}

image-20220203232555062

随笔记录:编译器找一些声明只会往上找

类的声明:class A;

如果我们在一个类内想访问另一个类的私有成员,就需要友元类

class Time
{
public:
	void GetData()
	{
		cout << d._year << d._month << d._day << endl;
	}
private:
	Date d;//借助这个对象
};
class Date
{
	friend class Time;
private:
	int _year;
	int _month;
	int _day;
};
int main()
{
	return 0;
}

image-20220203233823260

友元==突破类域

内部类

基础概念

什么是内部类?

类里面定义一个类,这就叫内部类

内部类就是外部类的友元类,所以内部类可以访问外部类的成员,用法也和友元类很像

计算类的大小时不算内部类

内部类内可以直接访问外部类的静态成员,不需要通过类名

内部类受到类域影响和访问限定符限制

内部类的使用

//内部类使用demo
class A
{
public:
	class B
	{
	public:
		B(const A& a)
		{
			cout << "我是内部类B" << endl;
			cout << "我可以访问外部类A的变量_a:" <<a._a << endl;
		}
	private:
		int _b;
	};
	A(){}
	A(const B& b)
	{
		cout << b._b << endl;//error
	}
private:
	int _a=1;
};
int main()
{
	A a;
	A::B b(a);
	return 0;
}	

image-20220204001232793

其实C++不咋用内部类,Java喜欢用内部类

补充

析构顺序例题

类A、B、C、D,问下面程序中析构函数的调用顺序?

C c;
int main()
{
	A a;
	B b;
	static D d;
	return 0;
}

答案:析构顺序 B A D C

构造顺序:C A B D

析构顺序是D B C A吗?不是

  • 析构函数的调用时期:对象声明周期结束后
  • 静态的变量存储在全局区,main函数结束后会销毁栈帧

①因为a,b都是局部对象,先构造则后析构,构造时是AB,则析构肯定是BA

换个角度理解,栈的特点是先进后出,那a先入栈就应该后销毁,所以b先调用析构函数

②剩下C D,C是全局对象,D是静态局部对象,这两个谁先析构?

静态局部变量先析构,全局变量C再析构

D先析构,C后析构,即DC

全局对象和静态局部对象的释放优先级在网上没有找到很好的解释

个人理解:CD都存在全局区,所以CD的构造顺序和析构顺序应该是相反的,即构造是CD,则析构是DC

组合①②,得到BADC

这种题的技巧:把局部变量作为一组,把全局和静态变量作为一组,写出两个相应的构造顺序,再逆置一下就得到相应的析构顺序,又因为局部变量先析构,再拼接两组的析构顺序得到答案

可以把上面那段代码拷到编译器上,然后自己写代码验证答案

总结

  • 初始化列表提供了一种更好的初始化的方式,如果初始化列表不能单独完成任务,就结合构造函数体完成初始化任务
  • explicit可以禁止内置类型和自定义类型的转换,具体操作就是修饰对应的构造函数
  • static成员可以牵扯出很多东西,比如静态成员函数可不可以非静态变量等等,抓住关键点:静态成员函数没有this指针,static成员变量始终是一块内存,类外初始化才会分配空间
  • 友元主要解决了我们在类外不能访问私有成员变量的问题,本质上破坏了封装,不建议大量使用
  • 内部类,C++一般不怎么用,内部类理解为外部类的友元,同时受到访问限定符的限制,类外使用内部类得用::突破类域限制
  • 面向对象是在模拟是在抽象模拟我们的现实世界!

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容! 

相关文章

  • C++实现ini文件读写的示例代码

    C++实现ini文件读写的示例代码

    这篇文章主要介绍了C++如何实现读写ini配置文件,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2022-05-05
  • C++实现LeetCode(146.近最少使用页面置换缓存器)

    C++实现LeetCode(146.近最少使用页面置换缓存器)

    这篇文章主要介绍了C++实现LeetCode(146.近最少使用页面置换缓存器),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C++ Boost Spirit入门教程

    C++ Boost Spirit入门教程

    Boost是为C++语言标准库提供扩展的一些C++程序库的总称。Boost库是一个可移植、提供源代码的C++库,作为标准库的后备,是C++标准化进程的开发引擎之一,是为C++语言标准库提供扩展的一些C++程序库的总称
    2022-11-11
  • 浅析iterator与指针的区别

    浅析iterator与指针的区别

    指针和iterator都支持减法运算,指针-指针得到的是两个指针之间的距离,迭代器-迭代器得到的是两个迭代器之间的距离
    2013-10-10
  • 利用简洁的C语言代码解决跳台阶问题与约瑟夫环问题

    利用简洁的C语言代码解决跳台阶问题与约瑟夫环问题

    这篇文章主要介绍了利用简洁的C语言代码解决跳台阶问题与约瑟夫环问题的方法,跳台阶问题与约瑟夫环问题是常见的基础算法题目,需要的朋友可以参考下
    2016-02-02
  • C++贪心算法处理多机调度问题详解

    C++贪心算法处理多机调度问题详解

    贪心算法(又称贪婪算法)是指,在对问题求解时,总是做出在当前看来是最好的选择。也就是说,不从整体最优上加以考虑,他所做出的仅是在某种意义上的局部最优解
    2022-06-06
  • C++读入XML文件示例

    C++读入XML文件示例

    本篇文章主要介绍了C++读入XML文件,读取和设置xml配置文件是最常用的操作,TinyXML是一个开源的解析XML的C++解析库,感兴趣的小伙伴们可以参考一下。
    2016-12-12
  • 深入分析C++中声明与定义的区别

    深入分析C++中声明与定义的区别

    C++学了这么多年你知道为什么定义类时,类的定义放在.h文件中,而类的实现放在cpp文件中。它们为什么能够关联到一起呢?你知道什么东西可以放在.h文件中,什么不能。什么东西又可以放在cpp文件中。如果你忘记了或是压根就不明白,那么读过此文你会清晰无比!!
    2014-09-09
  • C语言算法的定义及分析详解

    C语言算法的定义及分析详解

    这篇文章主要为大家详细介绍了C语言算法的定义及分析,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-09-09
  • C++用mysql自带的头文件连接数据库

    C++用mysql自带的头文件连接数据库

    现在正做一个接口,通过不同的连接字符串操作不同的数据库。要用到mysql数据库。通过网上的一些资料和自己的摸索,大致清楚了C++连接mysql的方法。可以通过2种方法实现。第一种方法是利用ADO连接,第二种方法是利用mysql自己的api函数进行连接。今天主要来讲解下使用API
    2016-07-07

最新评论