單例模式也是我最喜歡的模式,因為不管你建立多少個對象,當前對象在記憶體中只存在一份執行個體,而且像Android中對資料庫的操作,對HTTP的請求都可以用單例模式來實現,而且效率也會提升不少。這裡我們就對《西遊記》裡的主人公唐僧來實現單例模式呢:
public class Tangseng {private static Tangseng tangseng;Tangseng() {// TODO Auto-generated constructor stub}public Tangseng getInstance() {if (tangseng == null) {tangseng = new Tangseng();}return tangseng;}}
就簡單的幾行代碼,我們就實現了單例模式,以後調用的時候不管你new多少個對象,記憶體中只有一個。
在Android中調用一下看一下呢,
public class XiyoujiActivity extends Activity {private ListView listView;/** Called when the activity is first created. */@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main);listView = (ListView) findViewById(R.id.listView1);ArrayList<String> lists = new ArrayList<String>();lists.add(new Tangseng().getInstance().toString());lists.add(new Tangseng().getInstance().toString());ArrayAdapter<String> adapter = new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1, lists);listView.setAdapter(adapter);}}
顯示結果:
大家從圖上應該能看出來,不管我new了多少個唐僧的類,它的記憶體位址是同一個,這就是單例模式的特性,因為一般我們都將實現單例模式的類靜態化,靜態化是什麼意思呢?靜態化就是將方法變成靜態方法,JAVA中對於靜態方法是可以直接調用的,不需要new出來的。接下來我們再重構一下:
public class Tangseng {private static Tangseng tangseng;private Tangseng() {// TODO Auto-generated constructor stub}public static Tangseng getInstance() {if (tangseng == null) {tangseng = new Tangseng();}return tangseng;}}
其實也沒多大改動,只是在getInstance方法前面加了個static 名稱,這樣這個方法就實現了靜態方法,靜態方法有什麼好處呢?在Android中對於每個對象的生命週期都有很嚴格的規定,例如當前的Activity被另一個Activity覆蓋,如果Android運行記憶體不夠的話,前面的Activity就會被回收掉,但如果這個方法實現了static,一般是不會被回收的,這樣我們就提高的代碼的運行效率。接下來就我們再改一下調用方式呢:
public class XiyoujiActivity extends Activity {private ListView listView;/** Called when the activity is first created. */@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main);listView = (ListView) findViewById(R.id.listView1);ArrayList<String> lists = new ArrayList<String>();lists.add(Tangseng.getInstance().toString());lists.add(Tangseng.getInstance().toString());ArrayAdapter<String> adapter = new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1, lists);listView.setAdapter(adapter);}}
這裡我們就不用new對像了,直接用類名來調用。運行後的結果應該跟是一樣的,兩個記憶體位址是一樣的。這一章就講完了,謝謝大家關注。