设计模式(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;
}
设计模式(4)-序列生成器之单例模式相关推荐
- Java设计模式之创建型:单例模式
一.什么是单例模式: 单例模式可以确保系统中某个类只有一个实例,该类自行实例化并向整个系统提供这个实例的公共访问点,除了该公共访问点,不能通过其他途径访问该实例.单例模式的优点在于: 系统中只存在一个 ...
- php工厂模式和单例模式,php 设计模式之工厂模式、单例模式、注册树模式
php 设计模式之工厂模式.单例模式.注册树模式 在软件工程中,创建型设计模式承担着对象创建的职责,尝试创建适合程序上下文的对象,对象创建设计模式的产生是由于软件工程设计的问题,具体说是向设计中增加复 ...
- 扩增子分析解读4去嵌合体,非细菌序列,生成代表性序列和OTU表
写在前面 之前发布的<扩增子图表解读>系列,相信关注过我的朋友大部分都看过了(链接直达7月文章目录).这些内容的最初是写本实验室的学生们学习的材料,加速大家对同行文章的解读能力. < ...
- AAAI 2022 | Diaformer: 采用症状序列生成的方式做自动诊断
本文约5000字,建议阅读10分钟 本文为你介绍在智能医学应用领域 AI Drive 分享了他们的工作"Diaformer". 自动诊断是智能医学应用领域的一个重要方向,其多阶段的 ...
- 【ICLR2022】序列生成的目标侧数据增强
来源:专知 本文为论文,建议阅读5分钟本文提出了一种生成端的数据增强方法. 论文题目:Target-Side Data Augmentation for Sequence Generation 作者: ...
- Petuum提出序列生成学习算法通用框架
近日,来自人工智能创业公司 Petuum 的研究人员发表论文,提出序列生成学习算法的通用框架--广义的熵正则化策略优化框架(Generalized Entropy-Regularized Policy ...
- 1.6 语言模型和序列生成-深度学习第五课《序列模型》-Stanford吴恩达教授
←上一篇 ↓↑ 下一篇→ 1.5 不同类型的循环神经网络 回到目录 1.7 对新序列采样 语言模型和序列生成 (Language Model and Sequence Generation) 在自然语 ...
- matlab生成均匀部分散点图,应用halton序列生成均匀散点图
前言 一门课的作业要用RBF-DQ方法计算流场.简单来讲就是要在无网格的条件下用高精度格式实现流场的仿真计算.生成散点图时,用蒙特卡洛的方法得到的散点图分布不是很均匀,于是想到一种叫做halton的序 ...
- em算法 实例 正态分布_Petuum提出序列生成学习算法通用框架
近日,来自人工智能创业公司 Petuum 的研究人员发表论文,提出序列生成学习算法的通用框架--广义的熵正则化策略优化框架(Generalized Entropy-Regularized Policy ...
最新文章
- 关于软件开发环境的思考
- Sharepoint 2010 新特性笔记
- IB component change - CL_IBCOMPONENT_IL~CHANGE_COMPONENT
- C++笔记-lambda表达式需要注意的地方
- 【C语言】判断某一正整数是否为完数
- 计算机组成与体系结构——计算机结构——2020.11.19
- Bottlerocket:一套专用型容器操作系统
- # Please enter the commit message for your changes. Lines starting # with ‘#‘ will be ignored
- mathtype2022数学公式编辑器快捷键及操作技巧分享教程
- 使用打印方法将caj、pdz转pdf格式
- maya阿诺德渲染失败_[转载]Arnold 渲染器for maya 安装不了的解决办法
- 学习RAID磁盘阵列
- 82 将真分数分解为埃及分数
- 房屋安全鉴定的建筑结构检测技术
- 软件测试设计——按类型划分
- Apache NiFi系统管理员指南 [ 四 ]
- 计算机程序设计c++ 8-6:数组指针相关应用
- Redis学习:基本知识点(转载)
- 小式智能客服机器人严肃升级!!
- windows 共享文件夹