第55章
大B:“在多線程模式下,惰性初始化會使多個線程同時初始化該單體,造成一個JVM中多個單例類型的實例,如果這個單例類型的成員變量在運行過程中發生變化,會造成多個單例類型實例的不一致。”
小A:“那應該怎麽辦?”
大B:“加個同步修飾符:publicstaticsynchronizedSingletongetInstance()。這樣就保證了線程的安全性。這種處理方式雖然引入了同步代碼,但是因為這段同步代碼隻會在最開始的時候執行一次或多次,所以對整個係統的性能不會有影響。”
小A:“在更新屬性的時候,會造成屬性的讀寫不一致。那應該怎麽處理?”
大B:“1、讀者/寫者的處理方式。設置一個讀計數器,每次讀取信息前,將計數器加1,讀完後將計數器減1。使用notifyAll()解除在該對象上調用wait的線程阻塞狀態。隻有在讀計數器為0時,才能更新數據,同時調用wait()方法要阻塞所有讀屬性的調用。2、采用‘影子實例’的辦法。具體說,就是在更新屬性時,直接生成另一個單例對象實例,這個新生成的單例對象實例將從數據庫,文件或程序中讀取最新的信息;然後將這些信息直接賦值給舊單例對象的屬性。”
小A:“嘿嘿!師兄,能不能舉例來看一下啊?”
大B:“好的。”
例子:
publicclassGlobalConfig{
privatestaticGlobalConfiginstance;
privateVectorproperties=null;
privatebooleanisUpdating=false;
privateintreadCount=0;
privateGlobalConfig(){
//LoadconfigurationinformationfromDBorfile
//Setvaluesforproperties
}
privatestaticsynchronizedvoidsyncInit(){
if(instance……null){
instance=newGlobalConfig();
}
}
publicstaticGlobalConfiggetInstance(){
if(instance……null){
syncInit();
}
returninstance;
}
publicsynchronizedvoidupdate(Stringpdata){
syncUpdateIn();
//Updateproperties
}
privatesynchronizedvoidsyncUpdateIn(){
while(readCount>0){
try{
wait();
}catch(Exceptione){
}
}
}
privatesynchronizedvoidsyncReadIn(){
readCount++;
}
privatesynchronizedvoidsyncReadOut(){
readCount——;
notifyAll();
}
publicVectorgetProperties(){
syncReadIn();
//Processdata
syncReadOut();
returnproperties;
}
}
publicclassGlobalConfig{
privatestaticGlobalConfiginstance;
privateVectorproperties=null;
privatebooleanisUpdating=false;
privateintreadCount=0;
privateGlobalConfig(){
//LoadconfigurationinformationfromDBorfile
//Setvaluesforproperties
}
privatestaticsynchronizedvoidsyncInit(){
if(instance……null){
instance=newGlobalConfig();
}
}
publicstaticGlobalConfiggetInstance(){
if(instance……null){
syncInit();
}
returninstance;
}
publicsynchronizedvoidupdate(Stringpdata){
syncUpdateIn();
//Updateproperties
}
privatesynchronizedvoidsyncUpdateIn(){
while(readCount>0){
try{
wait();
}catch(Exceptione){
}
}
}
privatesynchronizedvoidsyncReadIn(){
readCount++;
}
privatesynchronizedvoidsyncReadOut(){
readCount——;
notifyAll();
}
publicVectorgetProperties(){
syncReadIn();
//Processdata
syncReadOut();
returnproperties;
}
}