單例(singleton)模式额嘿,本質(zhì)上瘸恼,就是給一個靜態(tài)變量賦值,而靜態(tài)變量在程序的整個內(nèi)存空間有且僅有1個副本册养,并且是對外可見的东帅;所以其他的class都可以調(diào)用這個副本。
比如下面球拦,在任何地方調(diào)用:S10.getInstance() 都會得到同一個S10
方法一
public class S10{
private SingletonLazy() {
}
public static S10 getInstance(){
return GetS10.instance;
}
static class GetS10{
private static S10 instance=new S10();
}
}
方法二
public class S10{
public static S10 mS10=new S10()
private SingletonLazy() {
}
public static S10 getInstance(){
return mS10
}
}
方法三靠闭,多線程并發(fā)時,容易創(chuàng)建多個不同對象坎炼;當(dāng)然最后一個對象會覆蓋前面的
public class S10{
public static S10 mS10=null;
private SingletonLazy() {
}
public static S10 getInstance(){
if(mS10==null){
mS10=new S10()
}
return mS10
}
}
下面詳細(xì)說一下static及相關(guān)的執(zhí)行順序
先看下這個例子愧膀,代碼邏輯:在main里面,調(diào)用s1和s2
public class A1 {
A1(){
System.out.printf("\n 構(gòu)造函數(shù)");
System.out.printf("\n 構(gòu)造函數(shù) "+str1);
}
public static String strStatic1="\n靜態(tài)成員變量1";
public String str1=" 000";
public static void main(String[] str){
System.out.printf("\n 開始執(zhí)行main");
// System.out.printf(strStatic1);
// System.out.println(str1);
// s1();
// new A1().s1();
s2();
A1 aaa1=null;
System.out.printf("\n 第1次new A");
aaa1=new A1();
aaa1.s1();
System.out.printf("\n----------");
System.out.printf("\n 第2次new 1");
A1 aaa2=null;
System.out.printf("\n 第2次new 2");
aaa2=new A1();
System.out.printf("\n 第2次new 3");
aaa2.s1();
System.out.printf("\n 分割線----------");
S8 ss=null;
System.out.println(S8.sss8);
S9 ss9=null;
}
public void s1(){
System.out.println("\n 開始執(zhí)行s1");
String str="\n s1 局部變量";
System.out.printf(str);
System.out.printf(str1);
}
public static void s2(){
System.out.println("\n 開始執(zhí)行s2");
// String strStatic="";
// String strStatic1=strStatic1;
String strStatic="\n s2 局部變量";
System.out.printf(strStatic);
}
static class S8 {
public static String sss8="\n 靜態(tài)變量sss8";
{
System.out.println("\n靜態(tài)類S8 普通代碼塊");
}
static {
System.out.println("\n靜態(tài)類S8 靜態(tài)代碼塊");
}
}
class S9 {
{
System.out.println("普通類S9 普通代碼塊");
}
public String sss8="\n 普通變量sss9";
}
{
System.out.println("\n 普通代碼塊 "+str1);
str1=" 001";
System.out.println("\n 普通代碼塊 "+str1);
}
static {
System.out.println("\n 靜態(tài)代碼塊");
}
}
//日志 如下
靜態(tài)代碼塊
開始執(zhí)行main
開始執(zhí)行s2
s2 局部變量
第1次new A
普通代碼塊 000
普通代碼塊 001
構(gòu)造函數(shù)
構(gòu)造函數(shù) 001
開始執(zhí)行s1
s1 局部變量 001
----------
第2次new 1
第2次new 2
普通代碼塊 000
普通代碼塊 001
構(gòu)造函數(shù)
構(gòu)造函數(shù) 001
第2次new 3
開始執(zhí)行s1
s1 局部變量 001
分割線----------
靜態(tài)類S8 靜態(tài)代碼塊
靜態(tài)變量sss8
可以看到:
1谣光、靜態(tài)代碼塊是在main執(zhí)行之前執(zhí)行檩淋,并且只執(zhí)行了這一次。
2抢肛、初始化狼钮,必然會先執(zhí)行普通代碼塊,然后才是構(gòu)造函數(shù)
3捡絮,new 100次,就會執(zhí)行100次普通代碼塊莲镣,與構(gòu)造函數(shù)
4福稳,定義一個類的變量,并不會執(zhí)行這個類里面的任何代碼塊
5, 調(diào)用內(nèi)部類的靜態(tài)成員變量瑞侮,會先執(zhí)行靜態(tài)代碼塊
6的圆、定義一個內(nèi)部類的變量鼓拧,不會執(zhí)行其任何代碼塊
7、非靜態(tài)內(nèi)部類不能存在靜態(tài)成員變量越妈,和靜態(tài)成員方法
我們稍微描述一下季俩,一些相關(guān)的變量有什么意思。
一梅掠、靜態(tài)方法
??一個類里面的靜態(tài)方法酌住,不能訪問非靜態(tài)成員變量和非靜態(tài)成員方法,只能訪問靜態(tài)成員變量和靜態(tài)成員方法
在static方法中阎抒,沒有this這個概念酪我。
1、靜態(tài)方法里面不能調(diào)用this且叁,如下面S1的寫法是錯誤的都哭,S2才是正確的
public String str="string 1";
//
public static void s1(){
String str=this.str;
}
public void s2(){
String str=this.str;
}
二、靜態(tài)變量
靜態(tài)變量和非靜態(tài)變量的區(qū)別是:靜態(tài)變量被所有的對象所共享逞带,在內(nèi)存中只有一個副本欺矫,它當(dāng)且僅當(dāng)在類初次加載時會被初始化。而非靜態(tài)變量是對象所擁有的展氓,在創(chuàng)建對象的時候被初始化汇陆,存在多個副本,各個對象擁有的副本互不影響带饱。
靜態(tài)成員變量初始化的順序:按照定義的順序毡代,順序初始化
三、靜態(tài)代碼塊
全局勺疼,不管new多少次教寂,只會初始化1次
四、普通代碼塊
全局执庐,new 多少次酪耕,就會初始化多少次
五、默認(rèn)構(gòu)造函數(shù)
只是一個普通的function轨淌,new 一個Class迂烁,它會執(zhí)行在普通代碼塊之后
拓展: