享元模式的应用之权限管理系统(混合了组合模式)

使用场景

(1)系统中存在大量的相似对象

(2)细粒度的对象都具有较接近的外部状态,而且内部状态与环境无关,也就是说对明没有特定的身份。

(3)需要缓冲池的场景

(4)如果不考虑对象的外部状态,可以用相对较少的共享对象取代很多组合对象,可以使用享元模式来共享对象,然后组合对象来使用这些共享对象。

【编程实验】权限管理系统


//声明文件

//******************************************************************************************
//结构型模式:享元模式
//场景:权限管理。
//几个概念
//1.安全实体:如某个数据表
//2.权限:指很对安全实体进行的操作,如查看、修改、删除等。
//几个问题
//1.因安全实体的权限可能被成千上万的人共享。如“人员列表”的查询权限
//  为了减少创建对象,可以将“人员列表”的“查询”权限作为一个享元对象,放入享元工厂
//2.为了增加实用性,这里采用享元模式+组合模式的方式实现了多层次的权限管理。
//  组合:将安全实体的权限进行组合,如“查看”+“修改” = “操作”权限,由于“操作”是组
//  合的权限,所以无须在享元工厂中缓存,即这个组合对象是个UnsharedConcreteFlyweight
//  对象。
//3. 享元工厂的垃圾回收:创建一个线程,专门负责过期的垃圾回收

#include <iostream>
#include <string>
#include <vector>
#include <map>
#include <sstream>
#include <ctime>
#include <windows.h>
#include <process.h>

using namespace std;



//
// 字符串分割
// 
// -------------------------------------------------------------------------
// 函数     : Split
// 功能     : 分割STL标准字符串
// 返回值   : void 
// 参数     : Container<std::basic_string<CharT> >& v 存放分割结果
// 参数     : const std::basic_string<CharT>& s 待分割字符串
// 参数     : const std::basic_string<CharT>& c 分割字符串
// -------------------------------------------------------------------------
template<typename CharT, template<typename S, typename Q = std::allocator<S> > class Container>
void Split(Container<std::basic_string<CharT> >& v, const std::basic_string<CharT>& s, const std::basic_string<CharT>& c);


//*******************************************辅助类******************************
//测试数据(在内存中模拟数据库中的值)
class CDataBase{
public:
	static vector<string> vecSingle;//用来存放单独授权数据的值
    //其中的key为组合数据的id,value为该组合包含的多条授权数据的值
	static map<string, vector<string>> mpMulti; //用来存放组合授权数据的值
};
//**********************************享元类************************
//享元接口:描述授权数据的接口
class CPermitWeight{
public:
	virtual ~CPermitWeight();
	virtual bool Match(string SecurityEntity, string Permit) = 0;
	virtual void Add(CPermitWeight* pPermit) = 0; //享元模式与组合模式的结合。为Flyweight添加子Flyweight对象
	virtual void Disp() = 0;//输出自身权限
	virtual bool IsAdmin() = 0;//判断是不是组合权限
};
//具体享元对象(封装授权数据中重复出现的部分)
//由于add是针对组合对象的,而这个可共享的是叶子对象,所以这里抛出
//异常就可以了。
//享元工厂里存放的是这个对象,如“薪资数据表的查看权限”。因为某一权限
//可能被分配给成上千万个人,所以需要缓存。即享元
class CSingleFly : public CPermitWeight{
private://内部状态
	string strSecurityEntity;//安全实体
	string strPermit; //权限
public:
	 //构造函数,传入状态数据,包含安全实体和权限的数据,用“,”分隔
	CSingleFly(string state);
	~CSingleFly();
	string GetSecurityEntity();
	string GetPermit();
	bool Match(string securityentity, string permit);
	void Add(CPermitWeight* pPermit); //叶子对象,什么都不做!这里也可以抛出异常!
	void Disp();//输出属性(成员变量即可)
	bool IsAdmin();//返回false表示是叶子节点
};

//不需要共享的享元对象的实现,也是组合模式中的组合对象
class CMultiFly : public CPermitWeight{
private: //记录每个组合对象所包含的子组件
	vector<CPermitWeight*> vRight;
public:
	~CMultiFly();
	void Add(CPermitWeight* permit);
	bool Match(string securityentity, string permit);
	void Disp();//输出每个叶子节点的权限
	bool IsAdmin();//返回true表示是分支节点
};
//******************************************************************
//享元工厂
class CAdmin{
private:
	static map<string, CPermitWeight*> mpPermit;
	static multimap<string, CPermitWeight*> mpUser;
public:
	static CPermitWeight* GetWeight(string key);
	//从数据库中获取某人所拥有的权限
	static void DispAllPermit();
	static void RemoveAllPermit();
	static void DispAllUser();
	static void RemoveAllUser();
	static vector<CPermitWeight*> QueryPermitByUser(string user);
    //判断某用户对某个安全实体是否拥有某种权限
	static bool HasPermit(string user, string securityentity, string permit);
};


//实现文件

//
// 字符串分割
// 
// -------------------------------------------------------------------------
// 函数     : Split
// 功能     : 分割STL标准字符串
// 返回值   : void 
// 参数     : Container<std::basic_string<CharT> >& v 存放分割结果
// 参数     : const std::basic_string<CharT>& s 待分割字符串
// 参数     : const std::basic_string<CharT>& c 分割字符串
// -------------------------------------------------------------------------
template<template<typename S, typename Q = std::allocator<S> > class Container>
void Split(Container<std::basic_string<char> >& v, const std::basic_string<char>& s, const std::basic_string<char>& c)
{
	if (0 == c.length())				  return;

	std::basic_string<char>::size_type pos1 = 0;
	std::basic_string<char>::size_type pos2 = 0;

	pos1 = 0;
	pos2 = s.find(c);
	while (std::basic_string<char>::npos != pos2){
		v.push_back(s.substr(pos1, pos2 - pos1));
		pos1 = pos2 + c.size();
		pos2 = s.find(c, pos1);
	}

	if (pos1 != s.length()) {
		v.push_back(s.substr(pos1));
	}
}


//*******************************************辅助类******************************
//测试数据(在内存中模拟数据库中的值)
//单独授权数据
static vector<string>::value_type init_value1[] = 
{
	vector<string>::value_type("ZhangSan,PersonTable,Query,1"),
	vector<string>::value_type("LiSi,PersonTable,Query,1"),
	vector<string>::value_type("LiSi,OptSalaryTable,,2"),
	vector<string>::value_type("ZhangSan0,PersonTable,Query,1"),
	vector<string>::value_type("ZhangSan1,PersonTable,Query,1"),
	vector<string>::value_type("ZhangSan2,PersonTable,Query,1")
};
vector<string> CDataBase::vecSingle(init_value1, init_value1+6);
//组合授权数据
static vector<string>::value_type init_value2[] =
{
	vector<string>::value_type("SalaryTable,Query"),
	vector<string>::value_type("SalaryTable,Modify")
};
static vector<string> CompositePermit(init_value2, init_value2+2);

static map<string, vector<string>>::value_type init_value3[] = 
{
	map<string, vector<string>>::value_type("OptSalaryTable", CompositePermit)
};
map<string, vector<string>> CDataBase::mpMulti(init_value3, init_value3+1);

//**********************************享元类************************
//享元接口:描述授权数据的接口
CPermitWeight::~CPermitWeight(){}

//具体享元对象(封装授权数据中重复出现的部分)
//由于add是针对组合对象的,而这个可共享的是叶子对象,所以这里抛出
//异常就可以了。
//享元工厂里存放的是这个对象,如“薪资数据表的查看权限”。因为某一权限
//可能被分配给成上千万个人,所以需要缓存。即享元
//构造函数,传入状态数据,包含安全实体和权限的数据,用“,”分隔
CSingleFly::CSingleFly(string state)
{
	vector<string> vString;
	Split(vString, state, ",");
	strSecurityEntity = vString[0];
	strPermit = vString[1];
}
CSingleFly::~CSingleFly(){ cout << "~CSingleFly" << endl;}
string CSingleFly::GetSecurityEntity(){return strSecurityEntity;}
string CSingleFly::GetPermit(){return strPermit;}
bool CSingleFly::Match(string securityentity, string permit)
{
	return (strSecurityEntity == securityentity && strPermit == permit);
}
void CSingleFly::Add(CPermitWeight* pPermit){}
void CSingleFly::Disp() { cout << strSecurityEntity << " : " << strPermit << endl; }
bool CSingleFly::IsAdmin(){return false;}
//不需要共享的享元对象的实现,也是组合模式中的组合对象
CMultiFly::~CMultiFly(){ cout << "~CMultiFly" << endl;}
void CMultiFly::Add(CPermitWeight* permit){ vRight.push_back(permit);}
bool CMultiFly::Match(string securityentity, string permit)
{
	bool bMatched = false;
	for(vector<CPermitWeight*>::iterator it = vRight.begin(); it != vRight.end(); it++){
		if((*it)->Match(securityentity, permit)){
			bMatched = true; break;
		}
	}
	return bMatched;
}

void CMultiFly::Disp()
{
	for(vector<CPermitWeight*>::iterator it = vRight.begin(); it != vRight.end(); it++){
		CPermitWeight* pPermit = (*it);
		cout << pPermit << " : "; pPermit->Disp();
	}
}
bool CMultiFly::IsAdmin(){return true;}
//******************************************************************
//享元工厂
CPermitWeight* CAdmin::GetWeight(string key)
{
	CPermitWeight* pPermit = mpPermit[key];
	if(pPermit == NULL){
		pPermit = new CSingleFly(key);
		mpPermit[key] = pPermit;
	}
	return pPermit;
}
//从数据库中获取某人所拥有的权限
void CAdmin::DispAllPermit(){
	cout << "******************************************" << endl;
	for(map<string, CPermitWeight*>::iterator it = mpPermit.begin(); it != mpPermit.end(); it++){
		cout << it->first << " : " << it->second << endl;
	}
	cout << "******************************************" << endl << endl;
}
void CAdmin::RemoveAllPermit(){
	for(map<string, CPermitWeight*>::iterator it = mpPermit.begin(); it != mpPermit.end(); ){
		cout << it->first << " : " << it->second << endl;
		CPermitWeight* pPermit = it->second;
		delete pPermit;
		it = mpPermit.erase(it);
	}
}

void CAdmin::DispAllUser(){
	cout << "******************************************" << endl;
	for(multimap<string, CPermitWeight*>::iterator it = mpUser.begin(); it != mpUser.end(); it++){
		cout << it->first << " : " << it->second << endl;
		CPermitWeight* pPermit = it->second;
		pPermit->Disp();
	}
	cout << "******************************************" << endl << endl;
}
void CAdmin::RemoveAllUser(){
	for(multimap<string, CPermitWeight*>::iterator it = mpUser.begin(); it != mpUser.end(); ){
		CPermitWeight* pPermit = it->second;
		if(pPermit->IsAdmin()){
			cout << it->first << " : " << it->second << endl; delete pPermit;
		}
		it = mpUser.erase(it);
	}
}
vector<CPermitWeight*> CAdmin::QueryPermitByUser(string user)
{
	vector<CPermitWeight*> vPermit;
	vector<string> vString;
	for(vector<string>::iterator it = CDataBase::vecSingle.begin(); it != CDataBase::vecSingle.end(); it++){
		vString.clear(); Split(vString, *it, ","); if(vString.size() < 4)			break;
		if(vString[0] == user){
			CPermitWeight* pPermit = NULL;
			if(vString[3] == "2"){
				pPermit = new CMultiFly();//表示组合
				cout << "UserName : " << user << "<<<<<<<<<<<<<<<<<<<<<<<<<" << endl;
				vector<string> vTmp = CDataBase::mpMulti[vString[1]];
				for(vector<string>::iterator it = vTmp.begin(); it != vTmp.end(); it++){
					pPermit->Add(CAdmin::GetWeight(*it));
				}
			}
			else			pPermit = CAdmin::GetWeight(vString[1] + "," + vString[2]);
			vPermit.push_back(pPermit);
			mpUser.insert(make_pair(user, pPermit));
		}
	}
	return vPermit;
}

//判断某用户对某个安全实体是否拥有某种权限
bool CAdmin::HasPermit(string user, string securityentity, string permit){
	bool bHas = false;
	vector<CPermitWeight*> vPermit = QueryPermitByUser(user);
	for(vector<CPermitWeight*>::iterator it = vPermit.begin(); it != vPermit.end(); it++){
		if((*it)->Match(securityentity, permit)){
			bHas = true; break;
		}
	}
	return bHas;
}


map<string, CPermitWeight*> CAdmin::mpPermit;
multimap<string, CPermitWeight*> CAdmin::mpUser;

//测试客户端

void main()
{
	cout << "ZhangSan Query PersonTable : " << CAdmin::HasPermit("ZhangSan", "PersonTable", "Query") << endl;
	cout << "LiSi Query SalaryTable : " << CAdmin::HasPermit("LiSi", "SalaryTable", "Query") << endl;
	cout << "LiSi Modify SalaryTable : " << CAdmin::HasPermit("LiSi", "SalaryTable", "Modify") << endl;
	cout << "ZhangSan0 Query SalayTable : " << CAdmin::HasPermit("ZhangSan0", "SalaryTable", "Query") << endl;
	cout << "ZhangSan1 Modify SalaryTable : " << CAdmin::HasPermit("ZhangSan1", "SalaryTable", "Modify") << endl;
	cout << "ZhangSan2 Query PersonTable : " << CAdmin::HasPermit("ZhangSan2", "PersonTable", "Query") << endl;
	CAdmin::DispAllPermit(); 
	CAdmin::DispAllUser(); 
	CAdmin::RemoveAllUser();
	CAdmin::RemoveAllPermit();
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值