六种单例的创建方式
1.饿汉式
public class Singleton {
private static Singleton instance = new Singleton();
private Singleton() {}
public static Singleton getInstance() {
return instance;
}
}
优点: 基于类的加载机制,避免了多线程同步问题,加载速度快。
缺点: 在类加载的时候就完成初始化,没有懒加载,如果没有使用这个实例,会造成内存浪费。
2.懒汉式-线程不安全版
public class Singleton {
private static Singleton instance;
private Singleton() {}
public static Singleton getInstance() {
if(instance == null){
instance = new Singleton();
}
return instance;
}
}
优点: 第一次调用是才初始化对象,避免浪费资源
缺点: 加载速度慢,线程不安全
3.懒汉式-线程安全版(synchronized加锁)
public class Singleton {
private static Singleton instance;
private Singleton() {}
public static synchronized Singleton getInstance() {
if(instance == null){
instance = new Singleton();
}
return instance;
}
}
优点: 多线程中保证线程安全 缺点: 每次获取对象实例,都需要进行同步,造成不必要的同步开销。
4.双重校验锁
public class Singleton {
private static volatile Singleton instance;
private Singleton() {}
public static Singleton getInstance() {
if(instance == null){
synchronized(Singleton.class) {
if(instance == null) {
instance = new Singleton();
}
}
}
return instance;
}
}
优点: 线程安全,懒加载,减少同步开销
缺点: 第一个获取对象速度稍慢,但其在某些情况下也会出现失效的情况,并不是完美的方式。
这里面使用了两次判空:第一次为了不必要的加锁同步,第二次是确保在instance为null的情况下才创建实例,避免多次创建。
方法中还是用了关键字volatile对变量进行修饰,有如下几个作用:
1.在Java内存模型中volatile可以保证可见性,及防止程序指令重排序。
2.对象的创建分为如下几个步骤:
instance = new Singleton();
-
1.为instance分配内存空间
-
2.初始化instance
-
3.将instance指向内存地址
如果不加volatile的话,程序的执行顺序就可能变成1->3->2,多线程中就会导致线程获取一个没有初始化的实例。例如线程a 执行了1,3, 此时线程b调用getInstance()发现instance不为空,返回instance,但此时instance还未初始化。
单例模式
单例设计模式:保证一个类仅有一个实例,并且提供一个访问它的全局访问点。有些对象只需要一个,这时可用单例模式。
-
传统的单例模式和new 创建对象的调用不一样
-
调用者要调用xxx.getInstance才能获得该单例
function Singleton(name) {
this.name = name;
}
Singleton.getInstance = function (name) {
if(this.instace){
return this.instace;
}else {
this.instace = new Singleton(name);
return this.instace;
}
};
var a = Singleton.getInstance(‘a’);
var b = Singleton.getInstance(‘b’);
console.log(a===b); //true
5."透明"的单例模式
-
透明”的单例类,用户从这个类中创建对象的时候,可以像使用其他任何普通类一样
-
直接 new 一个对象
-
不能new 多个对象,扩展性不好
var instace;
function Person(name) {
this.name = name;
if (!instace) {
instace = this;
}
return instace;
}
Person.prototype.getName = function () {
console.log(this.name);
};
var a = new Person(‘a’);
var b = new Person(‘b’);
console.log(a===b);
6.代理模式创建单例模式
-
代理模式:自己不去做,委托中间人做
-
Person是一个普通类,通过new Person可以创建一个对象
-
用代理模式创建CreateSinglePerson方法,通过new CreateSinglePerson可以创建一个单例
function Person(name) {
this.name = name;
}
Person.prototype.getName = function () {
console.log(this.name);
};
var CreateSinglePerson = (function (name) {
var instance;
return function () {
if (!instance) {
instance = new Person(name);
}
return instance;
};
})();
var a = new CreateSinglePerson(‘a’);
var b = new CreateSinglePerson(‘b’);
console.log(a === b);
var c = new Person(‘c’);
var d = new Person(‘d’);
console.log(c === d);
JavaScript中的单例模式
-
单例模式的核心是确保只有一个实例,并提供全局访问
-
在JavaScript可以通过直接创建一个对象来实现单例模式
-
可以用闭包的方式实现私有变量
let MyApp = {
name:‘app’,
getName:function() {
console.log(this.name);
}
};
let MyApp2 = (function(){
var _name = ‘app’;
return {
getName:function() {
console.log(_name);
}
}
})();
#####惰性单例
总结
最后对于程序员来说,要学习的知识内容、技术有太多太多,要想不被环境淘汰就只有不断提升自己,从来都是我们去适应环境,而不是环境来适应我们!
这里附上上述的技术体系图相关的几十套腾讯、头条、阿里、美团等公司20年的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。
相信它会给大家带来很多收获:
当程序员容易,当一个优秀的程序员是需要不断学习的,从初级程序员到高级程序员,从初级架构师到资深架构师,或者走向管理,从技术经理到技术总监,每个阶段都需要掌握不同的能力。早早确定自己的职业方向,才能在工作和能力提升中甩开同龄人。
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
Uc7ztDrF-1714522185820)]
[外链图片转存中…(img-oaYACJRi-1714522185821)]
当程序员容易,当一个优秀的程序员是需要不断学习的,从初级程序员到高级程序员,从初级架构师到资深架构师,或者走向管理,从技术经理到技术总监,每个阶段都需要掌握不同的能力。早早确定自己的职业方向,才能在工作和能力提升中甩开同龄人。
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!