【C++】定制删除器和特殊类设计(饿汉和懒汉~)

news2024/11/15 19:28:51

文章目录

  • 定制删除器
  • 一、设计一个只能在堆上(或栈上)创建的类
  • 二、单例模式
    • 1.饿汉模式
    • 2.懒汉模式
  • 总结


定制删除器

我们在上一篇文章中讲到了智能指针,相信大家都会有一个问题,智能指针该如何辨别我们的资源是用new int开辟的还是new int[]开辟的呢,要知道[]必须与delete[]匹配否则会有未知错误的,这个问题我们就交给定制删除器来解决:

int main()
{
	shared_ptr<int> sp1(new int[10]);
	shared_ptr<string> sp2(new string[10]);
	return 0;
}

 比如上面的代码,一旦运行就会立即引发崩溃,这是因为shared_ptr默认的释放方式是delete而不是delete[]。

 我们从文档中可以看到shared_ptr有一个模板参数是del,这个参数其实就是定制删除器,为了解决释放资源的问题。

template <class T>
struct DeleteArray
{
	void operator()(const T* ptr)
	{
		delete[] ptr;
		cout << "delete[]" <<ptr<< endl;
	}
};
int main()
{
	shared_ptr<int> sp1(new int[10],DeleteArray<int>());
	shared_ptr<string> sp2(new string[10],DeleteArray<string>());
	return 0;
}

 我们可以看到定制删除器非常简单,实际上就是一个仿函数,当我们将这个仿函数传给shared_ptr,shared_ptr会在它的构造函数中接收到这个仿函数,然后析构的时候就直接调用这个仿函数了。下面我们看看运行结果:

 可以看到是能成功释放的,不再像之前那样由于new和delete不匹配导致崩溃。当然,这里既然是直接传给构造函数的,那么我们完全可以不用再写仿函数了,直接用lambda会更加的方便,如下所示:

int main()
{
	shared_ptr<int> sp1(new int[10], [](const int* ptr)
		{
			delete[] ptr;
			cout << "delete[] ptr :" << ptr << endl;
		});
	shared_ptr<string> sp2(new string[10], [](const string* ptr)
		{
			delete[] ptr;
			cout << "delete[] ptr(string) :" << ptr << endl;
		});
	return 0;
}

 可以看到这样也是没有问题的如果不是要打印演示的话会比第一种方式更加的简洁,但是我们也说过lambda表达式的底层就是仿函数所以其实也差不了多少。当然我们不仅可以释放资源,还可以关闭文件:

shared_ptr<FILE> sp3(fopen("test", "w"), [](FILE* fp) { fclose(fp); });

下面我们也给自己的shared_ptr搞一个定制删除器模板:

首先我们自己是不能像库里面那样直接将定制删除器传给构造函数的,因为库里面的shared_ptr有好几个类,所以可以直接给构造函数加一个模板来传定制删除器。那么我们该如何解决呢?其实很容易因为我们就一个shared_ptr类,所以只需要给这个类多一个模板参数就可以了。

namespace sxy
{
	template <class T>
	struct Delete
	{
		void operator()(T* ptr)
		{
			delete ptr;
		}
	};
	template <class T, class D = Delete<T>>
	class shared_ptr
	{
	public:
		//保存资源
		shared_ptr(T* ptr = nullptr)
			:_ptr(ptr)
			, _pcount(new int(1))
			, _pmtx(new mutex)
		{

		}
		//拷贝构造
		shared_ptr(const shared_ptr<T>& sp)
			:_ptr(sp._ptr)
			, _pcount(sp._pcount)
			, _pmtx(sp._pmtx)
		{
			_pmtx->lock();
			++(*_pcount);
			_pmtx->unlock();
		}
		void Release()
		{
			bool flag = false;
			_pmtx->lock();
			if (--(*_pcount) == 0)
			{
				_del(_ptr);
				delete _pcount;
				flag = true;
			}
			_pmtx->unlock();
			if (flag)
			{
				delete _pmtx;
			}
		}
		//赋值重载
		//..........
		//释放资源
		~shared_ptr()
		{
			Release();
		}
		//像指针一样
		//..............
	private:
		T* _ptr;
		int* _pcount;
		mutex* _pmtx;
		D _del;
	};
}

 我们这里默认的是delete,当然也可以自己模板特化一个delete[],我们就不再演示了。下面我们先运行起来:

 没有问题,我们再自己传一个[]的试一下:

int main()
{
	sxy::shared_ptr<int,DeleteArray<int>> sp1(new int[10]);
	sxy::shared_ptr<string,DeleteArray<string>> sp2(new string[10]);
	return 0;
}

 可以看到运行起来也是没有问题的,但是我们实现的定制删除器是不可以传lambda表达式的,这是因为lambda表达式无法作为一个参数类型。

注意:unique_ptr也和我们自己设计的删除器一样,都是不可以用lambda表达式的,这里大家可以自行去验证。


一、设计一个只能在堆上(或栈上)创建的类

我们在设计只能在堆上创建的类之前,先设计一个不能被拷贝的类,因为后面的类都会用到不能被拷贝的特点。

拷贝只会放生在两个场景中:拷贝构造函数以及赋值运算符重载,因此 想要让一个类禁止拷贝,
只需让该类不能调用拷贝构造函数以及赋值运算符重载即可。
C++98
将拷贝构造函数与赋值运算符重载只声明不定义,并且将其访问权限设置为私有即可。
class BanCopy
{
    // ...
    
private:
    BanCopy(const BanCopy& bc);
    BanCopy& operator=(const BanCopy& bc);
    //...
};
原因:
1. 设置成私有:如果只声明没有设置成 private ,用户自己如果在类外定义了,就可以不
能禁止拷贝了
2. 只声明不定义:不定义是因为该函数根本不会调用,定义了其实也没有什么意义,不写
反而还简单,而且如果定义了就不会防止成员函数内部拷贝了。
C++11引入了delete关键字要禁用某个函数就更加的方便了:
C++11 扩展 delete 的用法, delete 除了释放 new 申请的资源外,如果在默认成员函数后跟上
=delete ,表示让编译器删除掉该默认成员函数。
class BanCopy
{
    // ...
    
private:
    BanCopy(const BanCopy& bc) = delete;
    BanCopy& operator=(const BanCopy& bc) = delete;
    //...
};

下面我们思考如何实现只能在堆上创建的类,首先如果只能在堆上创建那么我们肯定是不能直接用构造函数创建对象的,所以要把构造函数私有,其次如果将一个堆上的对象拷贝给栈上的对象也不符合要求,所以拷贝构造肯定也要禁掉,下面我们实现一下:

class HeapOnly
{
public:
	static HeapOnly* CreatObj() 
	{
		return new HeapOnly;
	}
	void print()
	{
		cout << "print()" << endl;
	}
private:
	HeapOnly()
	{}
	HeapOnly(const HeapOnly&) = delete;
};
int main()
{
	HeapOnly* hp = HeapOnly::CreatObj();
	hp->print();
	return 0;
}

可以看到我们将构造函数设为私有,这样就不会有随意的对象被创建,只能通过我们的creat函数接收在堆上开辟的对象,注意我们的Creat接口一定是静态的,因为我们将构造封掉了没有这个类的对象,如果不设置为静态的那么谁来调用这个接口呢,其次我们还有禁掉拷贝构造,为了防止下面这种情况:

HeapOnly ht(*hp);

对于赋值重载来讲我们是没必要考虑的,因为赋值重载一定是针对于已经创建过的对象,我们将构造函数和拷贝构造禁掉就避免了栈对象的出现,被创建的一定是堆上的,那么堆上的对象赋值还是在堆上。

当然我们还有第二种只在堆上创建的类的思想:

class HeapOnly
{
public:
	HeapOnly()
	{}
	void Destroy()
	{
		this->~HeapOnly();
	}
private:
	~HeapOnly()
	{}
	HeapOnly(const HeapOnly&) = delete;
};

我们将析构函数设为私有,这样只要是栈上创建的对象都会编译报错,因为无法调用其析构函数,但是指针却可以正常的开辟空间,当我们将析构函数设为私有后那么该如何释放空间呢?只需要加一个成员函数,让这个成员函数调用类内的析构就可以了,这里不能直接调用析构需要用this指针指向一下否则会有编译错误。

 可以看到是没有问题的。

下面我们实现只在栈上创建的类:

只在栈上创建我们只需要禁掉operator new和operator delete以及禁止static变量即可。

class StackOnly
{
public:
	static StackOnly CreatObj()
	{
		return StackOnly();
	}
	void print()
	{
		cout << "print()" << endl;
	}
private:
	StackOnly()
	{}
};

一旦我们将构造函数禁掉,那么static变量和stackOnly* = new都无法创建。当然这个类是封不死的对于下面这个情况:

static StackOnly st = StackOnly::CreatObj();

这也是这个类的缺陷,当然我们也可以直接禁掉operator new ,但是没必要因为我们禁掉析构后就无法调用new,毕竟new是需要构造函数的。

二、单例模式

设计模式:
设计模式( Design Pattern )是一套 被反复使用、多数人知晓的、经过分类的、代码设计经验的
总结 。为什么会产生设计模式这样的东西呢?就像人类历史发展会产生兵法。最开始部落之间打
仗时都是人拼人的对砍。后来春秋战国时期,七国之间经常打仗,就发现打仗也是有 套路 的,后
来孙子就总结出了《孙子兵法》。孙子兵法也是类似。
使用设计模式的目的:为了代码可重用性、让代码更容易被他人理解、保证代码可靠性。 设计模
式使代码编写真正工程化;设计模式是软件工程的基石脉络,如同大厦的结构一样。
单例模式:
一个类只能创建一个对象,即单例模式,该模式可以保证系统中该类只有一个实例,并提供一个
访问它的全局访问点,该实例被所有程序模块共享 。比如在某个服务器程序中,该服务器的配置
信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再
通过这个单例对象获取这些配置信息,这种方式简化了在复杂环境下的配置管理。

1.饿汉模式

我们先写出代码然后进行讲解:

#include <map>
class InfoSingleton
{
public:
	static InfoSingleton& GetInstance()
	{
		return _sin;
	}
	void insert(string name,int salary)
	{
		_info[name] = salary;
	}
	void print()
	{
		for (auto& e : _info)
		{
			cout << e.first << " : " << e.second << endl;
		}
		cout << endl;
	}
private:
	InfoSingleton()
	{}
	InfoSingleton(const InfoSingleton&) = delete;
	InfoSingleton& operator=(const InfoSingleton&) = delete;
	map<string, int> _info;
	static InfoSingleton _sin;
};

首先单例模式是只能创建一个对象,所以我们必须将构造函数设为私有,上面的场景是一个工资管理表用map存放,既然只有一个类我们就直接加了一个私有静态成员,这个成员就是我们唯一使用的类,注意我们还需要将这个静态成员在类外初始化:

InfoSingleton InfoSingleton::_sin;

然后我们设计一个getinstance的接口可以返回这个私有对象,注意返回的是引用因为只有一个类不能产生拷贝。insert和print接口是为了演示所以设计出来了,然后我们要将拷贝构造和赋值都禁掉,还是那个原因只有一个对象。接下来我们运行起来看看:

int main()
{
	InfoSingleton::GetInstance().insert("张三", 20);
	InfoSingleton& sl = InfoSingleton::GetInstance();
	sl.insert("李四", 200);
	sl.insert("王五", 600);
	sl.insert("赵六", 3100);
	sl.print();
	return 0;
}

上面的代码中我们给出了两种拿到这个类的唯一对象的方法,一种是利用静态成员函数的特性直接使用,另一个是引用接收,下面我们看看结果:

 结果也是没有问题的,下面我们说一说饿汉模式的缺点:

首先因为静态成员变量的原因,我们的类会在main函数开始之前就创建对象,这么做就会有一些缺点,比如:单例对象初始化的时候数据太多,会导致启动慢。2.多个单例类如果有初始化依赖关系的话,那么饿汉模式是无法保证哪个类先初始化的(比如:A和B都是单例类,我们要求B先初始化然后再初始化A,但是饿汉模式是无法控制顺序的)。

2.懒汉模式

懒汉模式和饿汉模式的区别在于,懒汉模式是等需要用到对象的时候再创建,下面我们写一下代码:

class InfoSingleton
{
public:
	static InfoSingleton* GetInstance()
	{
		//第一次获取单例对象的时候创建对象
		if (_psin == nullptr)
		{
			return new InfoSingleton;
		}
		return _psin;
	}
	void insert(string name, int salary)
	{
		_info[name] = salary;
	}
	void print()
	{
		for (auto& e : _info)
		{
			cout << e.first << " : " << e.second << endl;
		}
		cout << endl;
	}
private:
	InfoSingleton()
	{}
	InfoSingleton(const InfoSingleton&) = delete;
	InfoSingleton& operator=(const InfoSingleton&) = delete;
	map<string, int> _info;
	static InfoSingleton* _psin;
};
InfoSingleton* InfoSingleton::_psin = nullptr;

我们可以看到饿汉模式和懒汉模式代码上的区别是:

饿汉模式用的是静态成员变量,懒汉模式用的静态成员变量指针,并且懒汉模式只有第一次获取单列对象的时候才创建对象,这就解决了刚刚饿汉模式启动的时候初始化数据太多导致启动慢的问题,并且懒汉模式如果在有依赖关系的情况下是可以控制先初始化某个类再初始化某个类的。

不知道大家有没有发现,我们现在写的懒汉模式是有一个很明显的问题的,那就是如果是多线程的情况下第一次获取单例对象的时候有可能多new了对象,面对这个情况我们可以直接加锁解决问题。

//懒汉模式
class InfoSingleton
{
public:
	static InfoSingleton* GetInstance()
	{
		//第一次获取单例对象的时候创建对象
		//lock_guard<mutex> lock(_mtx);
		if (_psin == nullptr)
		{
			lock_guard<mutex> lock(_mtx);
			if (_psin == nullptr)
			{
				return new InfoSingleton;
			}
		}
		return _psin;
	}
private:
	InfoSingleton()
	{}
	InfoSingleton(const InfoSingleton&) = delete;
	InfoSingleton& operator=(const InfoSingleton&) = delete;
	map<string, int> _info;
	static InfoSingleton* _psin;
	static mutex _mtx;
};
InfoSingleton* InfoSingleton::_psin = nullptr;
mutex InfoSingleton::_mtx;

首先我们定义锁的时候必须是静态的,这是因为getinstance这个函数就是静态的,这个函数里是无法调用普通成员函数的,只能调用静态成员函数,并且我们的锁只需要保护第一次来判断是否需要创建对象的情况,如果写成下面这样的代码就会造成资源浪费:

static InfoSingleton* GetInstance()
	{
			lock_guard<mutex> lock(_mtx);
			if (_psin == nullptr)
			{
				return new InfoSingleton;
			}
		return _psin;
	}

为什么会造成资源浪费呢?因为第一次创建后后面的几次我们只需要返回这个对象的指针,这个时候是不需要加锁的,而像上面的代码我们即使已经创建过一次对象了进来后还是要加锁解锁消耗资源,这就造成了资源的浪费,所以我们多做一个判断就能解决这个问题。注意:饿汉模式是没有这个线程安全的问题的,因为饿汉模式在main函数之前就创建好对象了,main函数之前是不会有两个线程去创建对象的。

当然一般情况下我们的单例对象是不考虑释放的,不过如果需要释放该如何释放呢?其实和创建的时候一样,写一个静态的接口即可,代码如下:

static void DelInstance()
	{
		lock_guard<mutex> lock(_mtx);
		if (_psin != nullptr)
		{
			delete _psin;
			_psin = nullptr;
		}
	}

当然也有人想到如果有些人就是忘记手动释放资源那就麻烦了,所以又出现了一个自动的释放资源的方法:

class GC
	{
	public:
		~GC()
		{
			if (_psin)
			{
				cout << "~GC()" << endl;
				DelInstance();
			}
		}
	};

GC是这个类对象的内部类,这个类的析构函数就是如果我们没有手动释放单例类的资源,那么GC这个对象出了作用域会自动帮我们进行销毁。

 下面我们将程序运行起来看看能否释放:

 上面是我们自己手动释放后,GC就没有帮我们释放,下面我们再看看自己不手动释放GC是否会帮我们释放:

 所以我们总结一下,GC有下面两个好处:

可以手动调用主动回收,也可以让它在程序结束时自动回收。

当然下面还有一种更简单的懒汉模式的实现方法:

class InfoSingleton
{
public:
	static InfoSingleton& GetInstance()
	{
		static InfoSingleton sin;
		return sin;
	}
	void insert(string name, int salary)
	{
		_info[name] = salary;
	}
	void print()
	{
		for (auto& e : _info)
		{
			cout << e.first << " : " << e.second << endl;
		}
		cout << endl;
	}
private:
	InfoSingleton()
	{}
	InfoSingleton(const InfoSingleton&) = delete;
	InfoSingleton& operator=(const InfoSingleton&) = delete;
	map<string, int> _info;
};

为什么这种方法也被称为懒汉模式呢?这是因为getinstance这个函数内部定义的静态变量sin是一个局部静态变量,一个局部的静态成员变量在初始化的时候是在main函数后初始化,所以和我们new的效果一样,并且还不用加锁保护。这个方式就利用了一个知识点:静态的局部变量是在main函数之后才创建初始化的。但是对于这样的写法有一个点需要注意:

 在C++11之前,上面红框的部分是不能保证sin的初始化是线程安全的,在C++11之后,可以保证sin的初始化是线程安全的。


 总结

设计模式中最重要的就是单例模式了,这个模式在面试中是高频考点大家一定要学会。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

通用分页【上】

一、什么是通用分页&#xff1f; 通用分页是一种常用的分页方式&#xff0c;也被称为“前端分页”。它是指在获取数据的时候&#xff0c;将大量的数据分成若干页以便于用户查看和操作。通用分页通常是在前端完成的&#xff0c;也就是在用户界面上进行处理。通用分页的优点是可…

[Flask] 初识Flask

Flask是一个使用Python编写的轻量级Web应用框架 为了更好地管理Python应用&#xff0c;通常情况下都会在虚拟环境中安装Flask框架。在虚拟环境下&#xff0c;使用如下命令安装Flask pip install flask 我们可以直接使用pycharm工具创建虚拟环境&#xff0c;只需要在创建项目的…

chatGPT AI对话聊天绘画系统开发:打开人工智能AI社交聊天系统开发新时代

人工智能技术的快速发展和普及&#xff0c;催生了众多创新应用&#xff0c;其中&#xff0c;AI社交聊天系统成为当下市场的热门话题&#xff0c;本文将详细介绍开发属于自己的ChatGPT的过程&#xff0c;并探讨当下市场因Chat AI聊天系统所带来的影响性。 AI社交聊天系统的潜力与…

Springboot Mybatis 自定义顺序排序查询,指定某个字段

前言 与本文无关 "我进去了" ....... 正文 今天要讲些什么&#xff1f; 其实很简单&#xff0c;就是查询数据的时候&#xff0c;想根据自己指定的字段的自定义顺序&#xff0c;做排序查询数据。 本篇文章会讲到的几个点 &#xff1a; 1. 单纯sql 怎么实现 排序2. …

miniconda 安装 windows+linux

虽然常用anaconda&#xff0c;但是有时候只需要管理环境的时候&#xff0c;用miniconda更方便 如果想安装anaconda&#xff0c;可以参考这篇&#xff1a;anaconda安装 一、linux下 1.下载 链接&#xff1a;miniconda文档 cd /usr/localwget https://repo.anaconda.com/mini…

分布式处理框架 MapReduce

3.2.1 什么是MapReduce 源于Google的MapReduce论文(2004年12月)Hadoop的MapReduce是Google论文的开源实现MapReduce优点: 海量数据离线处理&易开发MapReduce缺点: 实时流式计算 3.2.2 MapReduce编程模型 MapReduce分而治之的思想 数钱实例&#xff1a;一堆钞票&#xff0…

Inno Setup 安装包制作软件

推荐一个开源的安装包制作软件&#xff1a;Inno Setup   Inno Setup是一个免费的安装制作软件&#xff0c;小巧、简便、精美是其最大特点&#xff0c;支持pascal脚本&#xff0c;能快速制作出标准Windows2000风格的安装界面&#xff0c;足以完成一般安装任务。 下载地址&…

进阶接口自动化测试——认证/代理/超时配置/错误异常(详细)

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 认证 1、基本认证…

【Android】Touch 事件分发逻辑梳理和避坑逻辑(上层设置了setOnTouchListener的事件监听但是没有起作用的原因)

背景 在项目中发现我明明在最上层的activity中的一个DrawerLayout对象设置了如下代码: /*** 超级白板的整体点击事件* 保证topBar在合适的时机出现*/binding.layoutMainDrawer.setOnTouchListener { _, event ->if (event.action MotionEvent.ACTION_DOWN) {val isVisib…

【毕业设计】爱琴海——基于HTML5的婚庆用品商城网页设计

一、内容简介 (一)背景与意义 “婚俗”是指结婚的风俗&#xff0c;各国各族人民按照自己的习俗&#xff0c;举行各具特色的婚礼&#xff0c;具有各自浓厚的民族独特风采。婚俗元素在是中国婚俗文化的媒介&#xff0c;承载了中华儿女对幸福和吉祥的追求。在中国婚俗文化的发展过…

【BI数据可视化】Docker部署metabase

开源世界从来不缺分享与馈赠&#xff0c;前一篇已经分享过【BI数据可视化】使用Docker快速部署Superset_wenchun001的博客-CSDN博客&#xff0c;今天我们继续来介绍一款开源的数据可视化系统 Metabase&#xff0c;帮助您解决数据可视化的难题。 Metabase是什么&#xff1f; 在传…

DOTA-cyclo(RGDfK),RGD肽指含有由Arg-Gly-Asp三个氨基酸组成的序列多肽,螯合剂多肽

试剂基团反应特点&#xff08;Reagent group reaction characteristics&#xff09;&#xff1a; DOTA-cyclo(RGDfK)螯合剂多肽&#xff0c;RGD肽是指含有由Arg-Gly-Asp三个氨基酸组成的序列多肽&#xff0c;有直线肽和环肽之分。它们是许多细胞外基质蛋白&#xff08;如VN、FN…

关于邮件协议的理解、实验以及java mail的实现

1、几种电子邮件系统组成 传统电子邮件系统 一个传统电子邮件系统由以下几个部分组成 UA&#xff08;用户代理&#xff09;邮箱服务器电子邮件协议 邮件发送协议&#xff08;SMTP&#xff09;与邮件接受协议&#xff08;POP3,IMAP&#xff09; 用户代理 UA (User Agent) 定…

mac上mysql启动报错问题Unable to lock ./ibdata1 error: 35

记录一次mac安装mysql启动报错问题Unable to lock ./ibdata1 error: 35。首先&#xff0c;这是第一次在公司新领的mac上面去安装mysql&#xff0c;在去年换新电脑之前&#xff0c;老电脑也安装过&#xff0c;没出现过这种问题。其次&#xff0c;自己的几台windows笔记本也安装过…

VS2019 彻底卸载、安装C\C++\C#

彻底卸载 进入“控制面板 - 程序和功能”&#xff0c;找到并选择VS2019&#xff0c;单击右键选择“卸载”。 在弹出的卸载向导中&#xff0c;我们需要选择要卸载的组件&#xff0c;如果要彻底卸载VS2019&#xff0c;则需要全部选择&#xff0c;并依次执行卸载操作。 在VS2019…

要从HTML中提取img标签的src属性(图片链接),可以使用正则表达式方式。

1. 定义提取src属性的正则表达式: const srcRegex /<img\s(?:[^>]*?\s)?src\s*\s*(["])((?:[^\1"]|\\\1|.)*?)\1/g 这个正则会匹配类似<img src"http://example.com/1.jpg">中的src属性和括号中的连接。2. 调用字符串的matchAll()方法…

系统集成项目管理工程师(项目管理知识[二])

第八章 项目进度管理 1、前导图法4种类型依赖关系&#xff1a; (1)结束-开始的关系&#xff08;F-S型&#xff09; (2)结束-结束的关系&#xff08;F-F型&#xff09; (3)开始-开始的关系&#xff08;S-S型&#xff09; (4)开始-结束的关系&#xff08;S-F型&#xff09; 2、…

Fastjson漏洞

Fastjson 概述 Fastjson是阿里巴巴公司开源的一款json解析器&#xff0c;它可以解析 JSON 格式的字符串&#xff0c;支持将 Java Bean 序列化为 JSON 字符串&#xff0c;也可以从 JSON 字符串反序列化到JavaBean。 漏洞版本 Fastjson <1.2.24 反序列化远程命令执行漏洞 F…

如何关闭带有隐患的端口?一键关闭危险端口?封锁电脑端口工具

你是否担心自己的Windows电脑存在安全隐患&#xff1f;毕竟&#xff0c;黑客攻击时刻都在发生&#xff01;今天&#xff0c;我将为大家分享一些有关Windows危险端口的知识&#xff0c;以及如何一键关闭高危端口&#xff0c;保障你的电脑安全。希望这些能够帮助你更好地保护自己…

Hidl编程实战(一)——定义HAL服务

1. 概述 hidl基本知识可以参考官网 安卓官网-hidl 也讲解了C和Java实现hidl 本文讲解CHal服务的创建 2. 文件的创建 aosp整编过的代码&#xff0c;可以直接choosecombo后使用hidl-gen工具。如果没有整编过&#xff0c;可以单编hidl-gen工具。 hidl-gen工具可以用来协助创建h…