C++特殊类设计【特殊类 || 单例对象 || 饿汉模式 || 懒汉模式】

目录

一,特殊类设计 

1. 只在堆上创建的类

2. 只允许在栈上创建的类 

3. 不能被继承的类

4. 不能被拷贝的类

 5. 设计一个类,只能创建一个对象(单例对象)

饿汉模式

懒汉模式

C++11静态成员初始化多线程安全问题

二, C++类型转换

C语言类型转换存在的问题

1. static_cast(近似类型)

2. reinterpret_cast(不相关类型)

3. const_cast 

4. dynamic_cast(一般用于多态)

5. RTTI(了解)

结语


嗨!收到一张超美的风景图,愿你每天都能顺心!

一,特殊类设计 

1. 只在堆上创建的类

实现方式:
1. 将类的构造函数私有,拷贝构造声明成私有。防止别人调用拷贝在栈上生成对象。
2. 提供一个静态的成员函数,在该静态成员函数中完成堆对象的创建
// 方法一
class HeapOnly
{
public:
	void Destory()
	{
		cout << "~HeapOnly" << endl;
		delete this;
	}

private:
	~HeapOnly()
	{}

	int i;
};
// 将析构函数设为私有,这样在栈中的对象(不需要手动析构),在栈结束时
// 无法自动调用析构函数。

// 方法二:构造函数私有,仅提供一个creat函数来进行构造
class HeapOnly_2
{
public:
	// static 添加原因:我们知道调用成员函数需要对象,但我们这时并没有对象,
	// 因此需要用static修饰。(先有鸡先有蛋问题)
	static HeapOnly_2*  creat_(int b) 
	{
		return new HeapOnly_2(b);
	}
	// 防止默认拷贝,赋值构造生成,在栈上创建
	HeapOnly_2(const HeapOnly_2& b) = delete;
	HeapOnly_2& operator=(const HeapOnly_2& b) = delete;
private:
	HeapOnly_2(int b)
		:i(b)
	{}

	int i;
};

int main()
{
	// HeapOnly it;
	HeapOnly* it = new HeapOnly;
	it->Destory();

	//HeapOnly_2 it2;
	HeapOnly_2* it2 = HeapOnly_2::creat_(1);
	//HeapOnly_2 it3 = *it2; 
	//HeapOnly_2 it3(*it2);
    delete it2
	return 0;
}

2. 只允许在栈上创建的类 

class StackOnly
{
public:
	static StackOnly creat_(int b)
	{
		return StackOnly(b); //意味着拷贝构造不能禁止 
	}

	// creat_返回值是右值,我们拷贝构造参数只设置右值,同时默认构造也不会产生
	StackOnly(StackOnly&& st)
		:_i(st._i)
	{}
	//但这也没有禁止,move一下就又可以了

private:
	// 构造一私有,堆栈上都无法构造创建
	StackOnly(int i)
		:_i(i){}

	StackOnly& operator=(const StackOnly& st) = delete;

	int _i;
};

方法同1类似,但这无法限制住,static对象的拷贝。

3. 不能被继承的类

C++98方式,将构造函数私有,派生类无法调用到构造函数导致无法被继承。

// C++98中构造函数私有化,派生类中调不到基类的构造函数。则无法继承
class NonInherit
{
public:
 static NonInherit GetInstance()
 {
 return NonInherit();
 }
private:
 NonInherit()
 {}
};

C++11新方式,在目标类添加关键词——final,表示该类无法被继承

class A  final
{
    // ....
};

4. 不能被拷贝的类

拷贝只会放生在两个场景中:拷贝构造函数以及赋值运算符重载,因此 想要让一个类禁止拷贝,只需让该类不能调用拷贝构造函数以及赋值运算符重载即可
C++98
将拷贝构造函数与赋值运算符重载只声明不定义,并且将其访问权限设置为私有即可
class CopyBan
{
    // ...
    
private:
    CopyBan(const CopyBan&);
    CopyBan& operator=(const CopyBan&);
    //...
};
原因:
1. 设置成私有:如果只声明没有设置成private,用户自己如果在类外定义了,就可以不能禁止拷贝了
2. 只声明不定义:不定义是因为该函数根本不会调用,定义了其实也没有什么意义,不写反而还简单,而且如果定义了就不会防止成员函数内部拷贝了。

C++11
C++11扩展delete的用法,delete除了释放new申请的资源外,如果在默认成员函数后跟上=delete,表示让编译器删除掉该默认成员函数。
class CopyBan
{
    // ...
    CopyBan(const CopyBan&)=delete;
    CopyBan& operator=(const CopyBan&)=delete;
    //...
};

 5. 设计一个类,只能创建一个对象(单例对象)

设计模式:
设计模式(Design Pattern)是一套 被反复使用、多数人知晓的、经过分类的、代码设计经验的总结。我们学过的有适配器,迭代器,包装器,以及扩展:工厂,观察者模式。下面的单例模式也是
单例模式:
一个类只能创建一个对象,即单例模式,该模式可以保证系统中该类只有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息,这种方式简化了在复杂环境下的配置管理。单例模式有两种实现模式:

饿汉模式

就是说不管你将来用不用,程序启动时就先创建一个唯一的实例对象
class SingleCase
{
public:
	static SingleCase* GetInstace()
	{
		return _sc;
	}
	
	size_t GetSize()
	{
		return _vc.size();
	}

	void Add(const string& str)
	{
		_mtx.lock();
		_vc.push_back(str);
		_mtx.unlock();
	}
	
	void print()
	{
		for (auto& it : _vc)
		{
			cout << it << endl;
		}
	}

	SingleCase(const SingleCase& it) = delete;
	SingleCase& operator=(const SingleCase& it) = delete;
private:
	SingleCase()
	{}

	mutex _mtx;
	vector<string> _vc;
	static SingleCase* _sc;
};
// 声明定义分离,_sc是也在类成员中(能调用类成员),虽然看起来像在类外。
SingleCase* SingleCase::_sc = new SingleCase;

 饿汉的缺点:

1. 单例对象的构造可能会占用大量资源,对应用程序的打开比较慢。

2. 不适合多个单例对象之间存在依赖关系生成。如单例A,单例B,必须得先有A再有B,如果在同一个源文件中跟声明顺序相同;但如果不在同一个源文件中,产生顺序就不确定。

优点:代码简单,支持一些高并发场景

懒汉模式

如果单例对象构造十分耗时或者占用很多资源,比如加载插件啊, 初始化网络连接啊,读取文件啊等等,而有可能该对象程序运行时不会用到,那么也要在程序一开始就进行初始化,就会导致程序启动时非常的缓慢。 所以这种情况使用懒汉模式( 延迟加载)更好。

class SingleCase_lazy
{
public:
	static SingleCase_lazy* GetInstace()
	{
		// 处理大多数情况,提高效率
		if (_sc == nullptr)
		{
			SingleCase_lazy::new_mtx.lock();
			// 两程序几乎同一时间进入该
			// 作用域时,再进行判断
			if (_sc == nullptr) 
			{
				// 加锁防止_sc发生覆盖
				_sc = new SingleCase_lazy;
			}
			SingleCase_lazy::new_mtx.unlock();
		}
		return _sc;
	}

	

	size_t GetSize()
	{
		return _vc.size();
	}

	void Add(const string& str)
	{
		_vmtx.lock();
		_vc.push_back(str);
		_vmtx.unlock();
	}

	void print()
	{
		for (auto& it : _vc)
		{
			cout << it << endl;
		}
	}

	 static void Destory()
	{
		new_mtx.lock();
		if (_sc)
		{
			cout << "delete _sc" << endl;
		}
		new_mtx.unlock();

		
		delete _sc;
	}

	class De
	{
	public:
		~De()
		{
			Destory();
		}
	};

	SingleCase_lazy(const SingleCase_lazy& it) = delete;
	SingleCase_lazy& operator=(const SingleCase& it) = delete;
private:
	SingleCase_lazy()
	{}
	static mutex new_mtx;
	mutex _vmtx;
	vector<string> _vc;
	static SingleCase_lazy* _sc; //无法显示调用析构,我们通过内部类,调用
};
SingleCase_lazy* SingleCase_lazy::_sc = nullptr;
mutex SingleCase_lazy::new_mtx; // 静态函数的锁
SingleCase_lazy::De _de;

总结:设计一个单例对象,我们需要满足:1. 把数据放一个类中,把这个类设计成单例类。 2. 保持进程唯一,需要将拷贝,赋值都禁止。 3. 设计模式是饿汉,还是懒汉

C++11静态成员初始化多线程安全问题

class SingleInstance {
public:
	static SingleInstance* GetInstance() {
		// 局部静态变量实现的懒汉式单例,C++11后线程安全
		// 在C++11之前,不能保证多线程安全
		static SingleInstance instance;
		return &instance;
	}
private:
	SingleInstance();
	~SingleInstance();
	SingleInstance(const SingleInstance& signal) = delete;
	SingleInstance& operator=(const SingleInstance& signal) = delete;
};

二, C++类型转换

标准C++为了加强类型转换的可视性,引入了四种命名的强制类型转换操作符:
static_cast、reinterpret_cast、const_cast、dynamic_cast

 

C语言类型转换存在的问题

C风格的转换格式很简单,但是有不少缺点的:
1. 隐式类型转化有些情况下可能会出问题:比如数据精度丢失
2. 显式类型转换将所有情况混合在一起,代码不够清晰
因此C++提出了自己的类型转化风格,注意因为C++要兼容C语言,所以C++中还可以使用C语言的转化风格

1. static_cast(近似类型)

static_cast用于非多态类型的转换(静态转换),编译器隐式执行的任何类型转换都可用static_cast,但它不能用于两个不相关的类型进行转换。
int main()
{
  double d = 12.34;
  int a = static_cast<int>(d);
  cout<<a<<endl;
  return 0;
}

2. reinterpret_cast(不相关类型)

reinterpret_cast操作符通常为操作数的位模式提供较低层次的重新解释,用于将一种类型转换为 另一种不同的类型
int main()
{
 double d = 12.34;
 int a = static_cast<int>(d);
 cout << a << endl;
 // 这里使用static_cast会报错,应该使用reinterpret_cast
 //int *p = static_cast<int*>(a);
 int *p = reinterpret_cast<int*>(a);
 return 0;
}

3. const_cast 

const_cast 最常用的用途就是删除变量的 const 属性,方便赋值
int main()
{
	volatile const int i = 10;
	int* it = const_cast<int*>(&i);
	*it = 2;
	
	cout << i << endl;  // 10
	cout << *it << endl; // 2
	// i为什么没有被修改?答:i被编译器优化,i被10替换到寄存中,并没有在内存中获取
	//在C++中,volatile关键字用于告诉编译器该变量的值可能会在程序的控制之外被改变,
	//因此编译器不应该对该变量进行优化。这意味着每次访问该变量时,编译器都会从内存
    // 中读取最新的值,而不是使用之前缓存的值 (来自gpt的回答)
	return 0;
}

4. dynamic_cast(一般用于多态)

dynamic_cast 用于将一个父类对象的指针 / 引用转换为子类对象的指针或引用 ( 动态转换 )

 

注意:
1. dynamic_cast 只能用于父类含有 虚函数 的类。

 2. dynamic_cast会先检查是否能转换成功,能成功则转换,不能则返回0。

class A
{
public :
virtual void f(){}
};
class B : public A
{};
void fun (A* pa)
{
// dynamic_cast会先检查是否能转换成功,能成功则转换,不能则返回
B* pb1 = static_cast<B*>(pa);
B* pb2 = dynamic_cast<B*>(pa);
cout<<"pb1:" <<pb1<< endl;
cout<<"pb2:" <<pb2<< endl;
}
int main ()
{
  A a;
  B b;
  fun(&a);
  fun(&b);
  return 0;
}

5. RTTI(了解)

RTTI Run-time Type identification 的简称,即: 运行时类型识别
C++ 通过以下方式来支持 RTTI
1. typeid 运算符(获取类型)
2. dynamic_cast 运算符
3. decltype

结语

   本小节就到这里了,感谢小伙伴的浏览,如果有什么建议,欢迎在评论区评论,如果给小伙伴带来一些收获请留下你的小赞,你的点赞和关注将会成为博主创作的动力

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/445561.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Android14 Handle机制

Handle是进程内部, 线程之间的通信机制. handle主要接受子线程发送的数据, 并用此数据配合主线程更新UI handle可以分发Message对象和Runnable对象到主线程中, 每个handle实例, 都会绑定到创建他的线程中, 它有两个作用,: (1) 安排消息在某个主线程中某个地方执行 (2) 安排…

工作纪实46-关于微服务的上线发布姿势

蓝绿部署 在部署时&#xff0c;不需要将旧版本的服务停掉&#xff0c;而是将新版本与旧版本同时运行&#xff0c;新版本测试无误之后再将旧版本停掉。这样可以避免再升级的过程中如果失败服务不可用的问题&#xff0c;因为同时部署了两个版本的程序&#xff0c;使得硬件资源是…

算法设计.

文章目录 1. 贪心算法&#xff1a;只看当前1.1 零钱兑换问题&#xff1a;力扣322 2. 活动选择问题3. 动态规划3.1 不同路径&#xff1a;3.2 0-1背包问题3.3 完全背包问题3.4 零钱兑换-动态规划 4. 最长公共字串--动态规划5. 最长公共子序列6. 最长递增子序列7. 打家劫舍8. 全排…

分销商城小程序怎么做_打造高效分销商城小程序的秘诀

在数字化浪潮席卷全球的今天&#xff0c;小程序成为了连接线上线下的重要桥梁。其中&#xff0c;分销商城小程序因其独特的裂变传播能力和低门槛的创业模式&#xff0c;受到了越来越多创业者和商家的青睐。那么&#xff0c;如何打造一个高效、吸引人的分销商城小程序呢&#xf…

数据库(一)初步认识数据库系统

什么是数据库&#xff1f; 表&#xff1a;以按行按列形式组织及展现的数据 如下便是一个表&#xff0c;也叫关系&#xff0c;描述了一批相互有关联关系的数据 数据库&#xff1a;起源于规范化表&#xff08;如成绩单&#xff09;的处理&#xff0c;简称DB&#xff0c;是相互有…

基于鳑鲏鱼优化算法(Bitterling Fish Optimization,BFO)的无人机三维路径规划

一、无人机路径规划模型介绍 无人机三维路径规划是指在三维空间中为无人机规划一条合理的飞行路径&#xff0c;使其能够安全、高效地完成任务。路径规划是无人机自主飞行的关键技术之一&#xff0c;它可以通过算法和模型来确定无人机的航迹&#xff0c;以避开障碍物、优化飞行…

童装WP模板

童装WP模板 https://www.wpniu.com/moban/6359.html

JDBC和连接池

JDBC和连接池 大纲 JDBC连接数据库的方式 具体案例 JDBC 需求&#xff1a;满足Java程序能对多个不同的数据库进行操作&#xff0c;而创建了一种接口&#xff0c;实现对数据库的规范 连接数据库的方式 1.方法1 先创建一个Driver对象&#xff0c;然后设置连接到的数据…

微信小程序提示确认框

如图所示&#xff0c;如何弹出微信小程序自带默认弹框&#xff1f; 代码如下&#xff1a; wx.showModal({ title: 确认, content: 确定要删除吗&#xff1f;, success (res) { if (res.confirm) { console.log(用户点击确定) } else if (res.cancel) { console.log(用…

物联网电气融合实训室建设方案

1 教学实训总体设计 1.1 建设背景 &#xff08;一&#xff09;政策推动与战略部署 近年来&#xff0c;物联网技术在全球范围内得到了广泛的关注和应用。作为信息技术的重要组成部分&#xff0c;物联网在推动经济转型升级、提升社会管理水平、改善民生福祉等方面发挥着重要作…

二、TensorFlow结构分析(5)案例

案例&#xff1a; minimize(error) 代码&#xff1a; def linear_regression():# 自实现线性回归# 1&#xff09;准备数据X tf.random.normal(shape[100,1])y_true tf.matmul(X,[[0.8]]) 0.7# 2&#xff09;构造模型# 定义模型参数 用 变量weights tf.Variable(initial_v…

Linux的基本权限

一、对shell的浅显认识 shell是操作系统下的一个外壳程序&#xff0c;无论是Linux操作系统&#xff0c;还是Windows操作系统&#xff0c;用户都不会直接对操作系统本身直接进行操作&#xff0c;需要通过一个外壳程序去间接的进行各种操作 在Linux的shell外壳就是命令行&#…

Storyboard动画、EventTrigger事件触发器

就是动画&#xff0c;要注意的就是EventTrigger中的SourceName就是想要实现这个功能的按钮 <StackPanel Orientation"Vertical"><Rectanglex:Name"rect"Width"200"Height"40"Fill"Pink" /><StackPanel Orie…

java之lombok

Lombok是一个实用的java类库&#xff0c;能通过注解的形式自动生成构造器&#xff0c; getter setter equsls hashcode tostring等方法 并且可以自动化生成日志变量&#xff0c;简化java开发&#xff0c;提高效率 作用 导入 <dependency><groupId>org.projectlomb…

Servlet API 详细讲解

Servlet API 详细讲解 API就是一组类和方法的集合&#xff0c;servlet 中的 类是非常多的&#xff0c;咱们只需要学习 3个类即可。 HttpServletHttpServletRequest&#xff08;服务器如何读取客户端响应&#xff09;HttpServletResponse&#xff08;服务器如何把响应返回给客…

C++ 中的头文件和源文件

#include<>一般用于包含系统头文件&#xff0c;诸如stdlib.h、stdio.h、iostream等&#xff1b; 类库目录下查找失败&#xff0c;编译器会终止查找&#xff0c;直接报错&#xff1a;No such file or directory. #include""一般用于包含自定义头文件&#xff…

Ubuntu18.04 下使用 Pybind11实现 C++ 调用 Python 函数和类的示例

Pybind11 是一个轻量级的库&#xff0c;它提供了在 C 中无缝集成 Python 代码的能力。使用 Pybind11&#xff0c;你可以很容易地从 C 调用 Python 代码&#xff0c;反之亦然。下面我将通过一个简单的例子来展示如何在 Ubuntu 系统上使用 Pybind11 从 C 调用 Python 接口。 安装…

css--浮动

一. 浮动的简介 在最初&#xff0c;浮动是用来实现文字环绕图片效果的&#xff0c;现在浮动是主流的页面布局方式之一。 二. 元素浮动后的特点 &#x1f922;脱离文档流。&#x1f60a;不管浮动前是什么元素&#xff0c;浮动后&#xff1a;默认宽与高都是被内容撑开&#xff0…

数据结构系列-链表实现

&#x1f308;个人主页: 会编辑的果子君 &#x1f4ab;个人格言:“成为自己未来的主人~” #define _CRT_SECURE_NO_WARNINGS #include"List.h" void ListTest01() {LTNode* plist LTInit();LTPushFront(plist, 1);LTPushFront(plist, 2);LTPushFront(plist, 3);…

VBA_NZ系列工具NZ02:VBA读取PDF使用说明

我的教程一共九套及VBA汉英手册一部&#xff0c;分为初级、中级、高级三大部分。是对VBA的系统讲解&#xff0c;从简单的入门&#xff0c;到数据库&#xff0c;到字典&#xff0c;到高级的网抓及类的应用。大家在学习的过程中可能会存在困惑&#xff0c;这么多知识点该如何组织…