设计模式(4)-序列生成器之单例模式

场景:序列生成器

系统中统一的序列生成程序,整个系统统一一套!那么就用单例模式吧!

首先看看单例模式

1)类持有一个自己的实例,而且还是个静态实例。

2)类的构造函数为私有属性。

3)用以获得实例的方法为静态方法。

看看类图

然后看一段试例程序:

#include <iostream>

using namespace std;

class Singleton{
private:   
	Singleton();//注意:构造方法私有   
	virtual ~Singleton();   
	static Singleton* instance;//惟一实例   
	int var;//成员变量(用于测试)   
public:   
	static Singleton* GetInstance();//工厂方法(用来获得实例)   
	int getVar();//获得var的值   
	void setVar(int);//设置var的值 
};
//构造方法实现 
Singleton::Singleton()   
{   
	this->var = 20;   
	cout<<"Singleton Constructor"<<endl;   
} 
Singleton::~Singleton()
{
	if(instance != NULL)
	{
		delete instance;
	}
}
//初始化静态成员   
//Singleton* Singleton::instance=new Singleton();  
Singleton* Singleton::instance=NULL;
Singleton* Singleton::GetInstance()   
{   
	if(instance == NULL)
	{
		instance = new Singleton();
	}
	return instance;   
}   
//seter && getter含数   
int Singleton::getVar()   
{   
	return this->var;   
}   
void Singleton::setVar(int var)   
{   
	this->var = var;   
}   

int main(int argc, char* argv[])
{
	Singleton *ton1 = Singleton::GetInstance();
	Singleton *ton2 = Singleton::GetInstance();
	cout<<"ton1 var = "<<ton1->getVar()<<endl;   
	ton1->setVar(150);  
	cout<<"ton2 var = "<<ton2->getVar()<<endl;
	return 0;
}

1、构造方法私有

那么,就意味着,只能在Singleton的成员函数中,才能调用Singleton的构造函数来创建实例。在Singleton之外,不能创建Singleton对象的实例。

2、代码中,定义了GetInstance方法,只能通过GetInstance方法来获取Singleton对象的实例,单例就是在GetInstance方法中控制的。

首先,Singleton有一个 static Singleton* instance;//惟一实例 

Singleton* Singleton::instance=NULL; 在这里初始化为NULL。

Singleton* Singleton::GetInstance()   {    if(instance == NULL)  {   instance = new Singleton();  }  return instance;   }  

上面的函数,就是通过instance来实现单例的。

当第一次调用GetInstance时,instance 为NULL,所以会执行 instance = new Singleton(); 把这个新建的实例保存到静态成员instance,并返回这个指针。

第二次到第N次调用GetInstance时,由于instance不为空,所以会直接返回instance 。也就是第一次调用GetInstance创建的那个实例。

所以这样就实现了,单实例。

意思就是说,Singleton对象的实例,只会被创建一次,就是说内存中,只存在一个Singleton的实例,就是所谓,单实例。

弄个生成单例的实例程序吧!

#include <sys/sem.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <time.h>
#include <sys/time.h>
#include <iostream>  

using namespace std;  


#define MAXID	9999
static struct sembuf op_open={1,-1,0  };

class GenHH{
	private:
		GenHH();//注意:构造方法私有 
    	virtual ~GenHH();    
    	static GenHH* instance;//惟一实例
		int opensem(key_t semkey);
		int creatsem(key_t semkey,int bigcount);
		int sem_open(int semid);
		unsigned int gen_seq();
	public:
		static GenHH* getInstance();//工厂方法(用来获得实例) 
		unsigned int gen_hh();
}

GenHH::~GenHH()
{
	if(instance != NULL)  
    {  
        delete instance;  
    }
}

//初始化静态成员
GenHH* GenHH::instance=NULL;
GenHH* GenHH::getInstance()
{
	if(instance == NULL)  
    {  
        instance = new Singleton();  
    }  
    return instance; 
}

unsigned int GenHH::gen_hh()
{
	unsigned int hh;
	char	chh[9];

	memset(chh,0,9);

	sprintf(chh,"%05d%04d",time(NULL)%100000,gen_seq());

	hh = atoi(chh);

	return hh;
}

unsigned int GenHH::gen_seq()
{
	int	seq,kid;
	int	semid,semval;

	struct timeval  tv;


	union semun {
		int val;
		struct semid_ds *buf;
                unsigned short *array;
        } semctl_arg;
	

	kid=ftok("/etc/hosts",'m');
	if(kid<0){
		printf("system Error! Can't find  /etc/hosts!\n");
		gettimeofday(&tv, NULL);
		return tv.tv_usec % MAXID ;
	}

	semid=opensem(kid);
	if(semid<=0){
		semid=creatsem(kid,MAXID);
		if(semid<0){
			gettimeofday(&tv, NULL);
			return tv.tv_usec % MAXID ;
		}
	}

	semval=semctl(semid,1,GETVAL,0);
	if(semval<=2){
		semctl_arg.val=MAXID;
                if ((semctl(semid,1,SETVAL,semctl_arg)) < 0 ){
			gettimeofday(&tv, NULL);
			return tv.tv_usec % MAXID ;
		}
	}
	
	sem_open(semid);
	semval=semctl(semid,1,GETVAL,0);


	return MAXID-semval;
}

int GenHH::opensem(key_t semkey)
{
	int semid;

    semid=semget(semkey,2,0);
    if(semid<0){
            printf("semaphoreid get error!\n");
            return -1;
    }

     return semid;
}

int GenHH::creatsem(key_t semkey,int bigcount)
{
	int semid,semval;
	union semun {
		int val;
		struct semid_ds *buf;
                unsigned short *array;
	} semctl_arg;

	semid=semget(semkey,2,IPC_CREAT|0600);
	if(semid<0){
		return -1;
	}

	if((semval=semctl(semid,1,GETVAL,0))<0)
		printf("GETVAL error!\n");
	else if(semval==0){
		semctl_arg.val=1;
		if(semctl(semid,0,SETVAL,semctl_arg)<0)
			printf("SETVAL error\n");

		semctl_arg.val=bigcount;
		if(( semctl(semid,1,SETVAL,semctl_arg)) < 0 )
			printf("setval error\n");
	}

	return semid;
}

int GenHH::sem_open(int semid)
{
	while(( semop(semid,&op_open,1) ) < 0 ){
		if( errno==EINTR ) {
			usleep(5000);
			continue;
		}
		printf("sem op_open error!\n");
		return -1;
	}
	return 0;
}

int main(int argc, char* argv[])  
{
	GenHH *genHH1 = GenHH::getInstance();
	GenHH *genHH2 = GenHH::getInstance();

	cout<<genHH1->gen_hh()<<endl;
	cout<<genHH2->gen_hh()<<endl;
	return 0;
}

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏开发与安全

从零开始学C++之对象的使用(三):static 与单例模式、auto_ptr与单例模式、const 用法小结、mutable修饰符

一、static 与单例模式 单例模式也就是简单的一种设计模式,它需要: 保证一个类只有一个实例,并提供一个全局访问点 禁止拷贝 #include <i...

24600
来自专栏C/C++基础

设计模式(7)——单例模式(Singleton Pattern,创建型)

使用设计模式可以提高代码的可复用性、可扩充性和可维护性。单例模式(Singleton Pattern),确保一个类只有一个实例,并提供一个全局访问点。

10420
来自专栏Clive的技术分享

PHP实现单例模式

<?php /** * 单例模式实现 */ class Singleton { //静态变量保存全局实例 private static $ins...

33770
来自专栏乐沙弥的世界

MySQL数据类型 -- 数值型

版权声明:本文为博主原创文章,欢迎扩散,扩散请务必注明出处。 https://blog.csdn.net/robinson_0612/art...

16030
来自专栏desperate633

第14课 组合查询创建组合查询union的使用规则

组合查询很容易理解就是讲多个查询的结果放在一起显示 使用UNION关键字进行查询的组合

9020
来自专栏Python

表的数据类型

一 介绍 存储引擎决定了表的类型,而表内存放的数据也要有不同的类型,每种数据类型都有自己的宽度,但宽度是可选的 详细参考: http://www.runoob....

22270
来自专栏JavaEE

mybatis的association以及collection的用法association:一对一关联(has one)collection:一对多关联(has many)

71180
来自专栏java一日一条

MySql数据库索引原理

第二部分结合MySQL数据库中InnoDB数据存储引擎中索引的架构实现讨论聚集索引、非聚集索引及覆盖索引等话题。

34830
来自专栏JavaQ

温故而知新-MySQL数据类型

选择数据类型的原则 MySQL支持多种数据类型,选择合适的数据类型存储数据对MySQL存储引擎来说至关重要,下面的一些原则可以在选择数据类型的时候做出更合适的选...

31970
来自专栏Zephery

数据库

1.SQL执行的顺序 ? 2.NULL “空值” 和”NULL”的概念: 1:空值(‘’)是不占用空间的,判断空字符用 = ‘’ 或者 <> ‘’ 来进行处...

38480

扫码关注云+社区

领取腾讯云代金券