《剑指offer》面试题2:实现Singleton 模式
面试题2:实现Singleton 模式
题目:设计一个类,我们只能生成该类的一个实例。
只能生成一个实例的类是实现了Singleton (单例)模式的类型。由于设计模式在面向对象程序设计中起着举足轻重的作用,在面试过程中很多公司都喜欢问一些与设计模式相关的问题。在常用的模式中, Singleton是唯一一个能够用短短几十行代码完整实现的模式。因此,写一个Singleton的类型是一个很常见的面试题。
单例模式有三种经典的设计方案:
- 延时加载,也称为懒汉模式 ,需要的时候才创建对象,也就是下面方法一用到的
- 双重锁模式,需要的时候才创建对象,线程安全,懒汉模式升级版,也就是下面方法二用到的
- 贪婪加载,也称为饿汉模式 , 程序执行前就创建好对象,也就是下面方法三用到的
- 测试,简单测试。
解法一:只适用于单线程环境
由于要求只能生成一个实例,因此我们必须把构造函数设为私有函数以禁止他人创建实例。我们可以定义一个静态的实例,在需要的时候创建该实例。下面定义类型Singletonl就是基于这个思路的实现;
class SingleTon
{
public:
static SingleTon* getInstance()
{
if (instance== NULL)
{
instance= new SingleTon(); //堆区申请对象
}
return instance;
}
private:
SingleTon(){} //隐藏构造函数接口
SingleTon(const SingleTon&);//拷贝构造函数
static SingleTon* instance; //静态指针可以不依赖对象调用,保存唯一实例地址
};
SingleTon* SingleTon::instance= NULL;
上述代码在Singleton的静态属性Instance中,只有在instance为null的时候才创建一个实例以避免重复创建。同时我们把构造函数定义为私有函数,这样就能确保只创建一个实例。
注意:只适用于单线程环境,在多线程中,如果一个现成在刚完成if (instance== NULL)这段代码时,还没有来的及创建对象时,另一个线程也执行到这个阶段,如此一来,程序就会出现问题,也不满足我们单例模式的目的。
加同步锁:if (instance== NULL)这段语句前后加锁,实现每次只能有一个线程访问
通常在解决线程安全类问题常用的方法就是加锁,限制对临界区的访问。下面这段代码虽然解决线程安全问题,但程序的效率却低了不少。
class SingleTon
{
public:
static SingleTon* getInstance()
{
lock(); //加锁 注:并未具体实现此函数,这里仅仅做演示
if (instance== NULL)
{
instance= new SingleTon(); //堆区申请对象
}
unlock(); //解锁
return instance;
}
private:
SingleTon(){} //隐藏构造函数接口
SingleTon(const SingleTon&);//拷贝构造函数
static SingleTon* instance; //静态指针可以不依赖对象调用,保存唯一实例地址
};
SingleTon* SingleTon::instance= NULL;
加锁解锁机制: 线程A在执行到lock()时,如果发现无法加锁,即有别的线程B正在使用(加锁)此临界资源,当A发现可以进行加锁操作时,即B已经执行完成并进行了解锁操作,此时A再加锁(防止别的线程执行),当A线程执行完遇到代码unlock()时进行解锁操作。
但是,还有一个问题,当唯一的一个实例已经被创建出来后,后面的线程执行到此段代码时,无需再构建实例,但lock() unlock()会被反复执行,程序进行了许多不必要的开销,降低了效率。
难道真的鱼(线程安全)与熊掌(效率)不可兼得乎?
方法二:双重锁模式
双重锁模式:在加锁前再判断一次,是否需要执行以下内容。虽然叫双重锁,可并不是真正的加了两重锁哦,只是利用了一点技巧而已,可见,有时候只要多想一点点总会有办法的嘛。
class SingleTon
{
public:
static SingleTon* getInstance()
{
if (instance== NULL) //对象已经生成,无需加锁、解锁操作
{
lock();
if (instance== NULL)
{
instance= new SingleTon();
}
unlock();
}
return instance;
}
private:
SingleTon(){}
SingleTon(const SingleTon&);
static SingleTon* instance;
};
方法三:提前创建实例
由于静态变量在代码段存放,在程序刚被加载时就已经存在,因此可以利用这一特点重新设计程序,直接构造实例并让静态的指针instance接收。
class SingleTon
{
public:
static SingleTon* getInstance()
{
return instance;
}
private:
SingleTon(){}
SingleTon(const SingleTon&);
static SingleTon* instance;
};
SingleTon* SingleTon::instance= new SingleTon();
测试:
Person类,属性有mname mage msex
class Person
{
public:
static Person* getInstance(char* name, int age, bool sex)
{
if (pm == NULL)
{
pm = new Person(name, age, sex);
}
return pm;
}
~Person()
{
delete[] mname;
mname = NULL;
}
private:
Person(char* name, int age, bool sex)
{
mname = new char[strlen(name) + 1]();
strcpy_s(mname, strlen(name) + 1, name);
mage = age;
msex = sex;
}
char* mname;
int mage;
bool msex;
static Person* pm;
};
Person* Person::pm = NULL;
int main()
{
char name[] = "zhangsan";
Person* p1 = Person::getInstance(name, 25, true);
Person* p2 = Person::getInstance(name, 25, true);
Person* p3 = Person::getInstance(name, 25, true);
Person* p4 = Person::getInstance(name, 25, true);
return 0;
}
运行结果:

可以看到运行结果中,不论getInstance方法调用多少次,始终只有一个实例产生。
《剑指offer》面试题2:实现Singleton 模式的更多相关文章
- 剑指Offer:面试题15——链表中倒数第k个结点(java实现)
问题描述 输入一个链表,输出该链表中倒数第k个结点.(尾结点是倒数第一个) 结点定义如下: public class ListNode { int val; ListNode next = null; ...
- 剑指offer面试题3 二维数组中的查找(c)
剑指offer面试题三:
- 剑指Offer——笔试题+知识点总结
剑指Offer--笔试题+知识点总结 情景回顾 时间:2016.9.23 12:00-14:00 19:00-21:00 地点:山东省网络环境智能计算技术重点实验室 事件:笔试 注意事项:要有大局观, ...
- C++版 - 剑指offer之面试题37:两个链表的第一个公共结点[LeetCode 160] 解题报告
剑指offer之面试题37 两个链表的第一个公共结点 提交网址: http://www.nowcoder.com/practice/6ab1d9a29e88450685099d45c9e31e46?t ...
- C++版 - 剑指offer 面试题23:从上往下打印二叉树(二叉树的层次遍历BFS) 题解
剑指offer 面试题23:从上往下打印二叉树 参与人数:4853 时间限制:1秒 空间限制:32768K 提交网址: http://www.nowcoder.com/practice/7fe2 ...
- C++版 - 剑指offer 面试题39:判断平衡二叉树(LeetCode 110. Balanced Binary Tree) 题解
剑指offer 面试题39:判断平衡二叉树 提交网址: http://www.nowcoder.com/practice/8b3b95850edb4115918ecebdf1b4d222?tpId= ...
- Leetcode - 剑指offer 面试题29:数组中出现次数超过一半的数字及其变形(腾讯2015秋招 编程题4)
剑指offer 面试题29:数组中出现次数超过一半的数字 提交网址: http://www.nowcoder.com/practice/e8a1b01a2df14cb2b228b30ee6a92163 ...
- C++版 - 剑指Offer 面试题39:二叉树的深度(高度)(二叉树深度优先遍历dfs的应用) 题解
剑指Offer 面试题39:二叉树的深度(高度) 题目:输入一棵二叉树的根结点,求该树的深度.从根结点到叶结点依次经过的结点(含根.叶结点)形成树的一条路径,最长路径的长度为树的深度.例如:输入二叉树 ...
- C++版 - 剑指offer 面试题24:二叉搜索树BST的后序遍历序列(的判断) 题解
剑指offer 面试题24:二叉搜索树的后序遍历序列(的判断) 题目:输入一个整数数组,判断该数组是不是某二叉搜索树的后序遍历的结果.如果是则返回true.否则返回false.假设输入的数组的任意两个 ...
- C++版 - 剑指Offer 面试题45:圆圈中最后剩下的数字(约瑟夫环问题,ZOJ 1088:System Overload类似)题解
剑指Offer 面试题45:圆圈中最后剩下的数字(约瑟夫环问题) 原书题目:0, 1, - , n-1 这n个数字排成一个圈圈,从数字0开始每次从圆圏里删除第m个数字.求出这个圈圈里剩下的最后一个数字 ...
随机推荐
- 点击弹出div内容包括遮罩层
效果: HTML: <!-- 添加分组按钮 --> <div class="group_add" onclick = "docu ...
- MySQL第五讲
内容回顾 单表操作 """ 1.配置文件先统一设置成utf8 \s 2.无论你怎么改都没有生效 你的机器上不止一个mysql文件 C有一个 D有一个 3.百度搜索 sho ...
- vue--按需加载的3种方式(解决网页首次加载速度慢的问题)
一.vue的异步组件加载 使用异步组件加载,打包的时候会将每个组件分开打包到不同的js文件中: {path: '/index', name: 'index', meta:{ title:'首页', r ...
- JZ-014-链表中倒数第 K 个结点
链表中倒数第 K 个结点 题目描述 输入一个链表,输出该链表中倒数第k个结点. 题目链接: 链表中倒数第 K 个结点 代码 /** * 标题:链表中倒数第 K 个结点 * 题目描述 * 输入一个链表, ...
- JZ-008-跳台阶
跳台阶 题目描述 一只青蛙一次可以跳上1级台阶,也可以跳上2级.求该青蛙跳上一个n级的台阶总共有多少种跳法(先后次序不同算不同的结果) 题目链接: 跳台阶 代码 public class Jz08 { ...
- 上传css,js文件并引用
今天在做单页面的简历,由于css样式跟js代码过多,所以就想着可不可以把css文件跟js文件上传到网上,然后引用. 一开始的想法是我上传到gitee上,但是从gitee服务器返回的Header上加了C ...
- 昇思MindSpore全场景AI框架 1.6版本,更高的开发效率,更好地服务开发者
摘要:本文带大家快速浏览昇思MindSpore全场景AI框架1.6版本的关键特性. 全新的昇思MindSpore全场景AI框架1.6版本已发布,此版本中昇思MindSpore全场景AI框架易用性不断改 ...
- laravel8 登录功能的实现
1.选择合适的框架,渲染出如上图所示的登录视图,视图有样式即可,可使用BootStrap或layUI去布局实现(10分) 2.正确显示出验证码(10分) 3.验证码要求无杂点.无干扰线,4位纯数字(1 ...
- PHP高并发商城秒杀
1.什么是秒杀 秒杀活动是一些购物平台推出的集中人气的活动,一般商品数量很少,价格很便宜,限定开始购买的时间,会在以秒为单位的时间内被购买一空.比如原价千元甚至万元的商品以一元的价格出售,但数量只有一 ...
- JVM学习总结(一)
JVM--Java虚拟机 1.类加载器 JVM虚拟机的类加载器有三个 bootstrapClassLoader 引导类加载器 是有C语言编写,在JVM虚拟机启动时 加载到内存中负责加载rt.jar夹包 ...