在Java中如何避免創(chuàng)建不必要的對(duì)象
簡(jiǎn)介
在Java開(kāi)發(fā)中,程序員要盡可能的避免創(chuàng)建相同的功能的對(duì)象,因?yàn)檫@樣既消耗內(nèi)存,又影響程序運(yùn)行速度。在這種情況下可以考慮重復(fù)利用對(duì)象。
接下來(lái)舉例幾種對(duì)象重復(fù)利用的場(chǎng)景,看看我們是不是有中招了,如果有趕緊趁著還沒(méi)被發(fā)現(xiàn)悄悄改掉,被發(fā)現(xiàn)了會(huì)被diss啦!
1、String和Boolean
如下兩種寫(xiě)法看似沒(méi)有什么區(qū)別,但是如果深入jvm底層了解,我們可以利用jvm運(yùn)行時(shí)常量池的特性,避免創(chuàng)建具有相同功能的String對(duì)象(尤其是在循環(huán)內(nèi)部創(chuàng)建)可以帶來(lái)比較可觀的性能優(yōu)化以及節(jié)約內(nèi)存。
錯(cuò)誤寫(xiě)法
// 每次都會(huì)創(chuàng)建一個(gè)新的String對(duì)象,且不會(huì)加入常量池 String name2 = new String("李子捌");
正確寫(xiě)法
// 正確寫(xiě)法 String name1 = "李子捌";
除此之外,剛寫(xiě)Java代碼的程序員們,也要正確的選擇String、StringBuilder、StringBuffer類的使用。String為不可變對(duì)象,通常用于定義不變字符串;StringBuilder、StringBuffer用于可變字符串操作場(chǎng)景,如字符串拼接;其中StringBuffer是線程安全的,它通過(guò)Synchronized關(guān)鍵字來(lái)實(shí)現(xiàn)線程同步。
// StringBuffer中的append()方法 public synchronized StringBuffer append(String str) { toStringCache = null; super.append(str); return this; } // StringBuilder中的append()方法 public StringBuilder append(String str) { super.append(str); return this; }
Boolean是常用的類型,在開(kāi)發(fā)中也應(yīng)該使用Boolean.valueof()而不是new Boolean(),從Boolean的源碼可以看出,Boolean類定義了兩個(gè)final static的屬性,而B(niǎo)oolean.valueof()直接返回的是定義的這兩個(gè)屬性,而new Boolean()卻會(huì)創(chuàng)建新的對(duì)象。
public static final Boolean TRUE = new Boolean(true); public static final Boolean FALSE = new Boolean(false);
2、自動(dòng)拆箱和裝箱
Java提供了基本數(shù)據(jù)類型的自動(dòng)拆箱和裝箱功能,那是不是意味著我們可以在代碼中隨意的使用這兩個(gè)類型呢?其實(shí)理論上在代碼層面是沒(méi)得問(wèn)題,不過(guò)在具體的性能方面還是有優(yōu)化的空間啦?。。?/p>
我們來(lái)測(cè)試下性能
long start = System.currentTimeMillis(); Integer sum = 0; for (int i = 0; i < 100000; i++) { sum += i; } System.out.println(System.currentTimeMillis() - start);
使用Integer耗時(shí)3毫秒
long start = System.currentTimeMillis(); // 修改Integer 為 int int sum = 0; for (int i = 0; i < 100000; i++) { sum += i; } System.out.println(System.currentTimeMillis() - start);
使用int耗時(shí)0毫秒
因此關(guān)于自動(dòng)拆箱裝箱的使用,我們其實(shí)也可以做適當(dāng)?shù)目紤],畢竟有時(shí)候代碼性能就是一點(diǎn)點(diǎn)擠出來(lái)的嘛?。?!
3、正則表達(dá)式
正則表達(dá)式我們經(jīng)常用于字符串是否合法的校驗(yàn),我們先來(lái)看一段簡(jiǎn)單的代碼(大家有沒(méi)有一眼看出問(wèn)題呢?我想你肯定看出來(lái)了?。。。?/p>
public static void main(String[] args) { String email = "1057301174@qq.com"; String regex = "^([a-z0-9A-Z]+[-|\.]?)+[a-z0-9A-Z]@([a-z0-9A-Z]+(-[a-z0-9A-Z]+)?\.)+[a-zA-Z]{2,}$"; long start = System.currentTimeMillis(); for (int i = 0; i < 10000; i++) { email.matches(regex); } System.out.println(System.currentTimeMillis() - start); }
執(zhí)行這段代碼的時(shí)間,一共耗時(shí)71毫秒,看似好像挺快的!
但是我們做個(gè)非常簡(jiǎn)單的優(yōu)化,優(yōu)化后的代碼如下所示:
public static void main(String[] args) { String email = "1057301174@qq.com"; String regex = "^([a-z0-9A-Z]+[-|\.]?)+[a-z0-9A-Z]@([a-z0-9A-Z]+(-[a-z0-9A-Z]+)?\.)+[a-zA-Z]{2,}$"; Pattern pattern = Pattern.compile(regex); long start = System.currentTimeMillis(); for (int i = 0; i < 10000; i++) { //email.matches(regex); pattern.matcher(email); } System.out.println(System.currentTimeMillis() - start); }
再次執(zhí)行代碼,一共耗時(shí)1毫秒,快了70倍呀?。?!
這是因?yàn)镾tring.matches()方法在循環(huán)中創(chuàng)建時(shí),每次都需要執(zhí)行Pattern.compile(regex),而創(chuàng)建Patter實(shí)例的成本很高,因?yàn)樾枰獙⒄齽t表達(dá)式編譯成一個(gè)有限狀態(tài)機(jī)( finite state machine)。這種我們經(jīng)常會(huì)因?yàn)镴ava api提供了比較方便的方法調(diào)用而忽略了性能考究,往往不容易被發(fā)現(xiàn)。這個(gè)時(shí)候就需要優(yōu)秀的你,去“咬文嚼字”啦!
總結(jié)
到此這篇關(guān)于在Java中如何避免創(chuàng)建不必要對(duì)象的文章就介紹到這了,更多相關(guān)Java避免創(chuàng)建不必要對(duì)象內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
JAVA對(duì)象JSON數(shù)據(jù)互相轉(zhuǎn)換的四種常見(jiàn)情況
這篇文章主要介紹了JAVA對(duì)象JSON數(shù)據(jù)互相轉(zhuǎn)換的四種常見(jiàn)情況,需要的朋友可以參考下2014-04-04Java關(guān)鍵字volatile知識(shí)點(diǎn)總結(jié)
在本篇文章里小編給大家整理的是一篇關(guān)于Java關(guān)鍵字volatile知識(shí)點(diǎn)總結(jié)內(nèi)容,有興趣的朋友們可以學(xué)習(xí)參考下。2021-01-01Java語(yǔ)言中finally是否一定會(huì)執(zhí)行你知道嗎
這篇文章主要為大家詳細(xì)介紹了Java finally是否一定會(huì)執(zhí)行,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-02-02基于Spring AOP @AspectJ進(jìn)階說(shuō)明
這篇文章主要介紹了基于Spring AOP @AspectJ進(jìn)階說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-01-01詳解Java實(shí)現(xiàn)JSONArray轉(zhuǎn)Map的三種實(shí)現(xiàn)方式
本文主要介紹了Java實(shí)現(xiàn)JSONArray轉(zhuǎn)Map的三種實(shí)現(xiàn)方式,本文只是自己常用的三種,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03