国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

Java快速掃盲指南

Tony_Zby / 679人閱讀

摘要:不相等的對象要具有不相等的哈希碼為了哈希表的操作效率,這一點很重要,但不是強(qiáng)制要求,最低要求是不相等的對象不能共用一個哈希碼。方法和方法協(xié)同工作,返回對象的哈希碼。這個哈希碼基于對象的身份生成,而不是對象的相等性。

本文面向

剛學(xué)完Java的新手們。這篇文章不講語法,而是一些除了語法必須了解的概念。

將要去面試的初級工程師們。查漏補缺,以免遭遇不測。

目前由于篇幅而被挪出本文的知識點:

淺析JVM之內(nèi)存管理

Java多線程筆記

Java反射學(xué)習(xí)小記

簡談Java String

JDK,JRE和 JVM 的區(qū)別

JVM:java 虛擬機(jī),負(fù)責(zé)將編譯產(chǎn)生的字節(jié)碼轉(zhuǎn)換為特定機(jī)器代碼,實現(xiàn)一次編譯多處執(zhí)行;

JRE:java運行時環(huán)境,包含了java虛擬機(jī)jvm,java基礎(chǔ)類庫。是使用java語言編寫的程序運行所需要的軟件環(huán)境;

JDK:java開發(fā)工具包,是編寫java程序所需的開發(fā)工具。JDK包含了JRE,同時還包含了編譯器javac,調(diào)試和分析工具,JavaDoc。

Java是如何編譯和執(zhí)行的?

上圖表示了Java代碼是怎么編譯和加載的

整個流程從 Java 源碼開始,經(jīng)過 javac 程序處理后得到類文件,這個文件中保存的是編譯源碼后得到的 Java 字節(jié)碼。類文件是 Java 平臺能處理的最小功能單位,也是把新代碼傳給運行中程序的唯一方式。

新的類文件通過類加載機(jī)制載入虛擬機(jī),從而把新類型提供給解釋器執(zhí)行。

Object的重要方法

所有類都直接或間接擴(kuò)展 java.lang.Object 類。這個類定義了很多有用的方法,而且你可以根據(jù)需求來重寫這些方法。

toString( )方法

toString( ) 方法的作用是返回對象的文本表示形式。連接字符串或使用 System.out.println( ) 等方法時,會自動在對象上調(diào)用這個方法。給對象提供文本表示形式,十分利于調(diào)試或記錄日志,而且精心編寫的 toString( ) 方法還能給報告生成等任務(wù)提供幫助。

Object 類中的 toString( ) 方法返回的字符串由對象所屬的類名和對象的十六進(jìn)制形式哈希碼(由 hashCode( ) 方法計算得到,本章節(jié)稍后會介紹)組成。這個默認(rèn)的實現(xiàn)方式提供了對象的類型和標(biāo)識兩個基本信息,但一般并沒什么用。

equals( )方法

== 運算符測試兩個引用是否指向同一個對象(比較兩個內(nèi)存單元的內(nèi)容是否一樣)。如果要測試兩個不同的對象是否相等,必須使用 equals( ) 方法。任何類都能覆蓋 equals( ) 方法,定義專用的相等比較方式。Object.equals( ) 方法直接使用 == 運算符,只有兩個對象是同一個對象時,才判定二者相等。

很多類以及自定義類的equals方法都需要重寫,是需要根據(jù)場景與需求來定制的。JDK自帶的許多類往往都是:

對比一些簡單的屬性值

再對比復(fù)雜的屬性值or對比業(yè)務(wù)上最快能區(qū)分對象的值

再對比其他的值or對比地址、長度

主要為了將那些不匹配的情況盡快排除

hashCode( )方法

Java中的hashCode方法就是根據(jù)一定的規(guī)則將與對象相關(guān)的信息(比如對象的存儲地址,對象的字段等)映射成一個數(shù)值,這個數(shù)值稱作為散列值。 如果集合中已經(jīng)存在一萬條數(shù)據(jù)或者更多的數(shù)據(jù),如果采用equals方法去逐一比較,效率必然是一個問題。此時hashCode方法的作用就體現(xiàn)出來了,當(dāng)集合要添加新的對象時,先調(diào)用這個對象的hashCode方法,得到對應(yīng)的hashcode值,實際上在HashMap的具體實現(xiàn)中會用一個table保存已經(jīng)存進(jìn)去的對象的hashcode值,如果table中沒有該hashcode值,它就可以直接存進(jìn)去,不用再進(jìn)行任何比較了;如果存在該hashcode值,就調(diào)用它的equals方法與新元素進(jìn)行比較,相同的話就不存了,不相同就散列其它的地址,所以這里存在一個沖突解決的問題,這樣一來實際調(diào)用equals方法的次數(shù)就大大降低了。

另外注意,默認(rèn)的hashCode會發(fā)起native調(diào)用,如果用hashCode對兩個對象對比,會導(dǎo)致開銷增大。
hashcode方法的作用

只要覆蓋了 equals( ) 方法,就必須覆蓋 hashCode( ) 方法。hashCode( ) 方法返回一個整數(shù),用于哈希表數(shù)據(jù)結(jié)構(gòu)。如果兩個對象經(jīng) equals( ) 方法測試是相等的,它們就要具有相同的哈希碼。不相等的對象要具有不相等的哈希碼(為了哈希表的操作效率),這一點很重要,但不是強(qiáng)制要求,最低要求是不相等的對象不能共用一個哈希碼。為了滿足最低要求,hashCode( ) 方法要使用稍微復(fù)雜的算法或位操作。

Object.hashCode( ) 方法和 Object.equals( ) 方法協(xié)同工作,返回對象的哈希碼。這個哈希碼基于對象的身份生成,而不是對象的相等性。(如果需要使用基于身份的哈希碼,可以通過靜態(tài)方法 System.identityHashCode( ) 獲取 Object.hashCode( ) 方法的返回值。)

hashCode和equal方法

hashCode的存在主要是用于查找的快捷性,如Hashtable,HashMap等,hashCode是用來在散列存儲結(jié)構(gòu)中確定對象的存儲地址的;

如果兩個對象相同,就是適用于equals(java.lang.Object) 方法,那么這兩個對象的hashCode一定要相同;

如果對象的equals方法被重寫,那么對象的hashCode也盡量重寫,并且產(chǎn)生hashCode使用的對象,一定要和equals方法中使用的一致,否則就會違反上面提到的第2點;

兩個對象的hashCode相同,并不一定表示兩個對象就相同,也就是不一定適用于equals(java.lang.Object)方法,只能夠說明這兩個對象在散列存儲結(jié)構(gòu)中,如Hashtable,他們"存放在同一個籃子里"。

HashCode和equal方法

Comparable::compareTo( )方法

如果一個類實現(xiàn)了 Comparable 接口,就可以比較一個實例是小于、大于還是等于另一個實例。這也表明,實現(xiàn) Comparable 接口的類可以排序。

因為 compareTo( ) 方法不在 Object 類中聲明,所以由每個類自行決定實例能否排序。如果能排序就定義 compareTo( ) 方法,實現(xiàn)實例排序的方式。

compareTo( ) 方法返回一個 int 類型的值,這個值需要進(jìn)一步說明。如果當(dāng)前對象(this)小于傳入的對象,compareTo( ) 方法應(yīng)該返回一個負(fù)數(shù);如果兩個對象相等,應(yīng)該返回 0;如果當(dāng)前對象大于傳入的對象,應(yīng)該返回一個正數(shù)。

clone( )方法

Object 類定義了一個名為 clone( ) 的方法,這個方法的作用是返回一個對象,并把這個對象的字段設(shè)為和當(dāng)前對象一樣。clone( ) 方法不常用,原因有兩個。其一,只有類實現(xiàn)了 java.lang.Cloneable 接口,這個方法才有用。Cloneable 接口沒有定義任何方法(是個標(biāo)記接口),因此若想實現(xiàn)這個接口,只需在類簽名的 implements 子句中列出這個接口即可。其二,clone( ) 方法聲明為 protected,因此,如果想讓其他類復(fù)制你的對象,你的類必須實現(xiàn) Cloneable 接口,并覆蓋 clone( ) 方法,而且要把 clone( ) 方法聲明為 public。

clone( ) 方法很難正確實現(xiàn),而副本構(gòu)造方法實現(xiàn)起來更容易也更安全。

finalize( )方法

一種古老的資源管理技術(shù)叫終結(jié)(finalization),開發(fā)者應(yīng)該知道有這么一種技術(shù)。然而,這種技術(shù)幾乎完全廢棄了,任何情況下,大多數(shù) Java 開發(fā)者都不應(yīng)該直接使用。

只有少數(shù)應(yīng)用場景適合使用終結(jié),而且只有少數(shù) Java 開發(fā)者會遇到這種場景。如果有任何疑問,就不要使用終結(jié),處理資源的 try 語句往往是正確的替代品。

終結(jié)機(jī)制的作用是自動釋放不再使用的資源。垃圾回收自動釋放的是對象使用的內(nèi)存資源,不過對象可能會保存其他類型的資源,例如打開的文件和網(wǎng)絡(luò)連接。垃圾回收程序不會為你釋放這些額外的資源,因此,終結(jié)機(jī)制的作用是讓開發(fā)者執(zhí)行清理任務(wù),例如關(guān)閉文件、中斷網(wǎng)絡(luò)連接、刪除臨時文件,等等。

終結(jié)機(jī)制的工作方式是這樣的:如果對象有 finalize( ) 方法(一般叫作終結(jié)方法),那么不再使用這個對象(或?qū)ο蟛豢蛇_(dá))后的某個時間會調(diào)用這個方法,但要在垃圾回收程序回收分配給這個對象的空間之前調(diào)用。終結(jié)方法用于清理對象使用的資源。

另外注意,這是一個實例方法。而在類上,沒有等效的機(jī)制。

引用類型與基本類型比較
type which
基礎(chǔ) byte short int long float double char boolean
引用 數(shù)組 對象

8種基本類型對應(yīng)的包裝類也是被final修飾。另外,String類和StringBuffer類也是被final修飾的。

引用類型和對象與基本類型和基本值有本質(zhì)的區(qū)別。

八種基本類型由 Java 語言定義,程序員不能定義新基本類型。引用類型由用戶定義,因此有無限多個。例如,程序可以定義一個名為 Point 的類,然后使用這個新定義類型的對象存儲和處理笛卡兒坐標(biāo)系中的 (x, y) 點。

基本類型表示單個值。引用類型是聚合類型(aggregate type),可以保存零個或多個基本值或?qū)ο蟆@纾覀兗僭O(shè)的 Point 類可能存儲了兩個 double 類型的值,表示點的 x 和 y 坐標(biāo)。char[ ] 和 Point[ ] 數(shù)組類型是聚合類型,因為它們保存一些 char 類型的基本值或 Point 對象。

基本類型需要一到八個字節(jié)的內(nèi)存空間。把基本值存儲到變量中,或者傳入方法時,計算機(jī)會復(fù)制表示這個值的字節(jié)。而對象基本上需要更多的內(nèi)存。創(chuàng)建對象時會在堆(heap)中動態(tài)分配內(nèi)存,存儲這個對象;如果不再需要使用這個對象了,存儲它的內(nèi)存會被自動垃圾回收。

把對象賦值給變量或傳入方法時,不會復(fù)制表示這個對象的內(nèi)存,而是把這個內(nèi)存的引用存儲在變量中或傳入方法。

在 Java 中,引用完全不透明,引用的表示方式由 Java 運行時的實現(xiàn)細(xì)節(jié)決定。如果你是 C 程序員的話,完全可以把引用看作指針或內(nèi)存地址。不過要記住,Java 程序無法使用任何方式處理引用。

似乎看的有點暈?來點兒代碼吧!

下述代碼處理 int 類型基本值:

int x = 42;
int y = x;

執(zhí)行這兩行代碼后,變量 y 中保存了變量 x 中所存值的一個副本。在 Java 虛擬機(jī)內(nèi)部,這個 32 位整數(shù) 42 有兩個獨立的副本。

現(xiàn)在,想象一下把這段代碼中的基本類型換成引用類型后再運行會發(fā)生什么:

Point p = new Point(1.0, 2.0);
Point q = p;

運行這段代碼后,變量 q 中保存了一份變量 p 中所存引用的一個副本。在虛擬機(jī)中,仍然只有一個 Point 對象的副本,但是這個對象的引用有兩個副本----這一點有重要的含義。假設(shè)上面兩行代碼的后面是下述代碼:

System.out.println(p.x);  // 打印p的x坐標(biāo):1.0
q.x = 13.0;               // 現(xiàn)在,修改q的x坐標(biāo)
System.out.println(p.x);  // 再次打印p.x,這次得到的值是13.0

因為變量 p 和 q 保存的引用指向同一個對象,所以兩個變量都可以用來修改這個對象,而且一個變量中的改動在另一個變量中可見。數(shù)組也是一種對象,所以對數(shù)組來說也會發(fā)生同樣的事,如下面的代碼所示:

// greet保存一個數(shù)組的引用
char[ ] greet = { "h","e","l","l","o" };
char[ ] cuss = greet;             // cuss保存的是同一個數(shù)組的引用
cuss[4] = "!";                   // 使用引用修改一個元素
System.out.println(greet);       // 打印“hell!”

把基本類型和引用類型的參數(shù)傳入方法時也有類似的區(qū)別。假如有下面的方法:

void changePrimitive(int x) {
    while(x > 0) {
        System.out.println(x--);
    }
}

調(diào)用這個方法時,會把實參的副本傳給形參 x。在這個方法的代碼中,x 是循環(huán)計數(shù)器,向零遞減。因為 x 是基本類型,所以這個方法有這個值的私有副本——這是完全合理的做法。

可是,如果把這個方法的參數(shù)改為引用類型,會發(fā)生什么呢?

void changeReference(Point p) {
    while(p.x > 0) {
        System.out.println(p.x--);
    }
}

調(diào)用這個方法時,傳入的是一個 Point 對象引用的私有副本,然后使用這個引用修改對應(yīng)的 Point 對象。例如,有下述代碼:

Point q = new Point(3.0, 4.5); // 一個x坐標(biāo)為3的點
changeReference(q);            // 打印3,2,1,而且修改了這個Point對象
System.out.println(q.x);       // 現(xiàn)在,q的x坐標(biāo)是0!

調(diào)用 changeReference( ) 方法時,傳入的是變量 q 中所存引用的副本。現(xiàn)在,變量 q 和方法的形參 p 保存的引用指向同一個對象。這個方法可以使用它的引用修改對象的內(nèi)容。但是要注意,這個方法不能修改變量 q 的內(nèi)容。也就是說,這個方法可以隨意修改引用的 Point 對象,但不能改變變量 q 引用這個對象這一事實。

那么在用運算符:==時,也會有差別。

相等運算符(==)比較基本值時,只測試兩個值是否一樣(即每一位的值都完全相同)。而 == 比較引用類型時,比較的是引用而不是真正的對象。也就是說,== 測試兩個引用是否指向同一個對象,而不測試兩個對象的內(nèi)容是否相同。

Java 的四種引用

在 JDK1.2 后,Java 對引用概念擴(kuò)充,分為強(qiáng)引用、軟引用、弱引用、虛引用。強(qiáng)度漸弱。

在開始了解前,最好先稍微了解一下Java Memory Model。我的這篇文章中簡單的講了一下JMM

強(qiáng)引用

就是值在程序代碼之中普遍存在的,類似 Object obj = new Object() 這類的引用,只要強(qiáng)引用還在,垃圾收集器永遠(yuǎn)不會回收掉被引用的對象。

軟引用

它關(guān)聯(lián)著的對象,在系統(tǒng)將要發(fā)生內(nèi)存溢出異常之前,將會把這些對象列進(jìn)回收范圍內(nèi)進(jìn)行第二次回收。提供 SoftReference 類來實現(xiàn)軟引用。

弱引用

強(qiáng)度比軟引用更弱一些,被弱引用關(guān)聯(lián)的對象只能生存到下一次垃圾收集發(fā)生之前。提供 WeakReference 類來實現(xiàn)軟引用。

虛引用

一個對象是否有虛引用的存在,完全不會對其生存時間構(gòu)成影響,也無法通過虛引用來去的一個對象實例。為一個對象設(shè)置虛引用關(guān)聯(lián)的唯一目的就是能在這個對象被收集器回收時收到一個系統(tǒng)通知。提供 PhantomReference 類來實現(xiàn)軟引用。

Java 7之基礎(chǔ) - 強(qiáng)引用、弱引用、軟引用、虛引用

Java垃圾回收機(jī)制與引用類型

神奇的數(shù)組

數(shù)組類型不是類,但數(shù)組實例是對象。這意味著,數(shù)組從 java.lang.Object 類繼承了方法。數(shù)組實現(xiàn)了 Cloneable 接口,而且覆蓋了 clone( ) 方法,確保數(shù)組始終能被復(fù)制,而且 clone( ) 方法從不拋出 CloneNotSupportedException 異常。數(shù)組還實現(xiàn)了 Serializable 接口,所以只要數(shù)組中元素的類型能被序列化,數(shù)組就能被序列化。而且,所有數(shù)組都有一個名為 length 的字段,這個字段的修飾符是 public final int,表示數(shù)組中元素的數(shù)量。

因為數(shù)組擴(kuò)展自 Object 類,而且實現(xiàn)了 Cloneable 和 Serializable 接口,所以任何數(shù)組類型都能放大轉(zhuǎn)換成這三種類型中的任何一種。而且,特定的數(shù)組類型還能放大轉(zhuǎn)換成其他數(shù)組類型。如果數(shù)組中的元素類型是引用類型 T,而且 T 能指定給類型 S,那么數(shù)組類型 T[ ] 就能指定給數(shù)組類型 S[ ]。注意,基本類型的數(shù)組不能放大轉(zhuǎn)換。例如,下述代碼展示了合法的數(shù)組放大轉(zhuǎn)換:

String[ ] arrayOfStrings;      // 創(chuàng)建字符串?dāng)?shù)組
int[ ][ ] arrayOfArraysOfInt;   // 創(chuàng)建int二維數(shù)組
Object[ ] oa = arrayOfStrings;// String可以指定給Object,因此String[ ]可以指定給Object[ ]
Comparable[ ] ca = arrayOfStrings;// String實現(xiàn)了Comparable接口,因此String[ ]可以視作Comparable[ ]
Object[ ] oa2 = arrayOfArraysOfInt;// int[ ]是Object類的對象,因此int[ ][ ]可以指定給Object[ ]
// 所有數(shù)組都是可以復(fù)制和序列化的對象
Object o = arrayOfStrings;
Cloneable c = arrayOfArraysOfInt;
Serializable s = arrayOfArraysOfInt[0];

因為數(shù)組類型可以放大轉(zhuǎn)換成另一種數(shù)組類型,所以編譯時和運行時數(shù)組的類型并不總是一樣。這種放大轉(zhuǎn)換叫作"數(shù)組協(xié)變"(array covariance)。

所以在某種意義上,集合框架比數(shù)組好用:

Object [] objectArray = new Long[1];
objectArray[0] = "I dont fit in"; //Throws ArrayStoreException
List ol = new ArrayList(); //Incompatible types
ol.add("I dont fit in");

一個只有在運行時才能拋出異常,一個在編譯期便可以發(fā)現(xiàn)錯誤。

封裝類 Java中為什么要為基本類型提供封裝類呢?

是為了在各種類型間轉(zhuǎn)化,通過各種方法的調(diào)用。否則你無法直接通過變量轉(zhuǎn)化。

比如,現(xiàn)在int要轉(zhuǎn)為String

int a=0;
String result=Integer.toString(a);

比如現(xiàn)在要用泛型

List nums;

這里< >里面就需要指定一個類。如果用int,則報錯。

自動裝箱(autoboxing)與拆箱(unboxing)

自動裝箱是 Java 編譯器在基本數(shù)據(jù)類型和對應(yīng)的對象包裝類型之間做的一個轉(zhuǎn)化。

基本類型和引用類型的表現(xiàn)完全不同。有時需要把基本值當(dāng)成對象,為此,Java 平臺為每一種基本類型都提供了包裝類。Boolean、Byte、Short、Character、Integer、Long、Float 和 Double 是不可變的最終類,每個實例只保存一個基本值。包裝類一般在把基本值存儲在集合中時使用。 例如

java.util.List:
List numbers =newArrayList( );// 創(chuàng)建一個List集合
numbers.add(newInteger(-1));// 存儲一個包裝類表示的基本值
int i =((Integer)numbers.get(0)).intValue( );// 取出這個基本值

把 int 轉(zhuǎn)化成 Integer,double 轉(zhuǎn)化成 Double等,反之就是自動拆箱。

Integer  a=1;//這就是一個自動裝箱,如果沒有自動裝箱的話,需要這樣Integer  a=new Integer(1)  
int b=a;//這就是一個自動拆箱,如果沒有自動拆箱的話,需要這樣:int b=a.intValue( )

這樣就能看出自動裝箱和自動拆箱是簡化了基本數(shù)據(jù)類型和相對應(yīng)對象的轉(zhuǎn)化步驟。

自動拆裝箱將會導(dǎo)致性能問題,因為有些數(shù)字不屬于緩存范圍——意味著會產(chǎn)生新的對象,尤其是在集合框架中會嚴(yán)重導(dǎo)致性能下降。

請運行一下下面的代碼,并探究一下:

public static void main(String []args){
    Integer a = 1;
    Integer b = 1;
    Integer c = 200;
    Integer d = 200;
    System.out.println(a==b);
    System.out.println(c==d);
}

Java中的自動裝箱與拆箱

關(guān)于異常

圖是我自己做的。如果覺得子類父類傻傻分不清,可以按照“紅橙黃綠”這個順序來,最高父類是紅。

遇上Error就是跪了,你就別想拯救了。

Exception一般由編碼、環(huán)境、用戶操作輸入出現(xiàn)問題,我們要可以捕捉的也處于這一塊兒。

運行時異常由java虛擬機(jī)由自己捕獲自己拋出。

檢查異常則由自己捕獲自己拋出多重catch,順序是從子類到父類。

異常拋出

throw:將產(chǎn)生的異常拋出。交給上層去處理。異常鏈----A方法拋出異常,B方法嘗試捕獲。main中調(diào)用B,B捕獲的異常中會有A異常的信息。

throws:聲明將要拋出何種類型的異常。

下面是異常類族譜

捕捉的異常時,不要僅僅調(diào)用printStackTreace( )去打印輸出,應(yīng)該添加事務(wù)回滾等操作。catch(Exception)可以捕捉遺漏的異常。最后在finally語句里記得釋放資源。

Java異常處理的10個最佳實踐

這里是我收集的 Java 編程中異常處理的 10 個最佳實踐。大家對 Java 中的受檢異常(checked Exception)褒貶不一,這種語言特性要求該異常必須被處理。在本文中,我們盡可能少使用受檢異常,同時也要學(xué)會在 Java 編程中,區(qū)別使用受檢和非受檢異常。

1 為可恢復(fù)的錯誤使用受檢異常,為編程錯誤使用非受檢異常。

對 Java 開發(fā)者來說,選擇受檢還是非受檢異常總是讓人感到困惑。受檢異常保證你會針對錯誤情況提供異常處理代碼,這是一種從語言層面上強(qiáng)制你編寫健壯代碼的一種方式,但同時也引入大量雜亂的代碼并導(dǎo)致其可讀性變差。當(dāng)然,如果你有可替代方式或恢復(fù)策略的話,捕獲異常并做處理看起來似乎也合情合理。在 Java 編程中選擇受檢異常還是運行時異常的更多信息,請參考 checked vs unchecked exceptions。

2 在 finally 程序塊中關(guān)閉或者釋放資源

這是 Java 編程中一個廣為人知的最佳實踐和一個事實上的標(biāo)準(zhǔn),尤其是在處理網(wǎng)絡(luò)和 IO 操作的時候。在 finally 塊中關(guān)閉資源能保證無論是處于正常還是異常執(zhí)行的情況下,資源文件都能被合理釋放,這由 finally 語句塊保證。從 Java7 開始,新增加了一項更有趣的功能:自動資源管理,或者稱之為ARM塊。盡管如此,我們?nèi)匀灰涀≡?finally 塊中關(guān)閉資源,這對于釋放像 FileDescriptors 這類資源至關(guān)重要,因為它在 socket 和文件操作中都會被用到。

3 在堆棧信息中包含引起異常的原因

Java 庫和開源代碼在很多情況下會將一種異常包裝成另一種異常。這樣記錄和打印根異常就變得非常重要。Java 異常類提供了 getCause() 方法來獲取導(dǎo)致異常的原因,這可以提供更多有關(guān)異常發(fā)生的根本原因的信息。這條實踐對調(diào)試或排除故障大有幫助。在把一個異常包裝成另一種異常時,記住需要把源異常傳遞給新異常的構(gòu)造器。

4 始終提供異常的有意義的完整信息

異常信息是最重要的,在其中,你能找到問題產(chǎn)生的原因,因為這是出問題后程序員最先看到的地方。記得始終提供精確的真實的信息。例如,對比下面兩條 IllegalArgumentException 的異常信息:

message 1: “Incorrect argument for method” message 2: “Illegal value for ${argument}: ${value}

第一條消息僅說明了參數(shù)是非法的或不正確的,但第二條消息包括了參數(shù)名和非法值,這對找到錯誤原因很重要。在編寫異常處理代碼的時候,應(yīng)當(dāng)始終遵循該 Java 最佳實踐。

5 避免過度使用受檢異常

受檢異常的強(qiáng)制性在某種程度上具有一定的優(yōu)勢,但同時它也使得代碼可讀性變差,混淆了正常的業(yè)務(wù)邏輯代碼。你可以通過適度使用受檢異常來最大限度地減少這類情況的發(fā)生,這樣可以得到更簡潔的代碼。你同樣可以使用 Java7 的新功能,比如在一個catch語句中捕獲多個異常,以及自動管理資源,以此來移除一些冗余的代碼。

6 將受檢異常轉(zhuǎn)為運行時異常

這是在諸如 Spring 之類的框架中用來減少使用受檢異常的方式之一,大部分 JDBC 的受檢異常都被包裝進(jìn) DataAccessException 中,DataAccessException異常是一種非受檢異常。這個最佳實踐帶來的好處是可以將特定的異常限制到特定的模塊中,比如把 SQLException 拋到 DAO 層,把有意義的運行時異常拋到客戶端層。

7 記住異常的性能代價高昂

需要記住的一件事是異常代價高昂,同時讓代碼運行緩慢。假如你有一個方法從 ResultSet 中進(jìn)行讀取,它經(jīng)常會拋出 SQLException 而不是將 cursor 移到下一元素,這將會比不拋出異常的正常代碼執(zhí)行的慢的多。因此最大限度的減少不必要的異常捕捉,去修復(fù)真正的根本問題。不要僅僅是拋出和捕捉異常,如果你能使用 boolean 變量去表示執(zhí)行結(jié)果,可能會得到更整潔、更高性能的解決方案。修正錯誤的根源,避免不必要的異常捕捉。

8 避免空的 catch 塊

沒有什么比空的 catch 塊更糟糕的了,因為它不僅隱藏了錯誤和異常,同時可能導(dǎo)致你的對象處于不可用狀態(tài)或者臟狀態(tài)。空的 catch 塊沒有意義,除非你非常肯定異常不會以任何方式影響對象的狀態(tài),但在程序執(zhí)行期間,用日志記錄錯誤依然是最好的方法。這在 Java 異常處理中不僅僅是一個最佳實踐,而且是一個最通用的實踐。

9 使用標(biāo)準(zhǔn)異常

第九條最佳實踐是建議使用標(biāo)準(zhǔn)和內(nèi)置的 Java 異常。使用標(biāo)準(zhǔn)異常而不是每次創(chuàng)建我們自己的異常,這對于目前和以后代碼的可維護(hù)性和一致性,都是最好的選擇。重用標(biāo)準(zhǔn)異常使代碼可讀性更好,因為大部分 Java 開發(fā)人員對標(biāo)準(zhǔn)的異常更加熟悉,比如 JDK 中的RuntimeException,IllegalStateException,IllegalArgumentException,NullPointerException,他們能立馬知道每種異常的目的,而不是在代碼或文檔里查找用戶自定義異常的目的。

10 為方法拋出的異常編寫文檔

Java 提供了 throw 和 throws 關(guān)鍵字來拋出異常,在 javadoc 中可以用@throw 為任何可能被拋出的異常編寫文檔。如果你編寫 API 或者公共接口,這就變得非常重要。當(dāng)任何方法拋出的異常都有相應(yīng)的文檔記錄時,就能潛在的提醒任何調(diào)用該方法的開發(fā)者。

Java 創(chuàng)建對象的幾種方式

用new 語句創(chuàng)建對象,這是最常見的創(chuàng)建對象的方法

運用反射手段,調(diào)用 java.lang.Class 或者 java.lang.reflect.Constructor 類的 newInstance( ) 實例方法

調(diào)用對象的 clone( ) 方法

運用反序列化手段,調(diào)用 java.io.ObjectInputStream 對象的 readObject( ) 方法

(1)和(2)都會明確的顯式的調(diào)用構(gòu)造函數(shù);(3)是在內(nèi)存上對已有對象的影印,所以不會調(diào)用構(gòu)造函數(shù) (4)是從文件中還原類的對象,也不會調(diào)用構(gòu)造函數(shù)。

序列化(Serializable )與反序列化(Deserialize)

對象序列化(Serializable)是指將對象轉(zhuǎn)換為字節(jié)序列的過程,而反序列化則是根據(jù)字節(jié)序列恢復(fù)對象的過程。

簡單的來說就是從object變成了byte,用于傳輸。

序列化一般用于以下場景:

永久性保存對象,保存對象的字節(jié)序列到本地文件中;

通過序列化對象在網(wǎng)絡(luò)中傳遞對象;

通過序列化在進(jìn)程間傳遞對象。

只有實現(xiàn)了Serializable和Externalizable接口的類的對象才能被序列化。

小Tips:對子類對象進(jìn)行反序列化操作時,如果其父類沒有實現(xiàn)序列化接口,那么其父類的構(gòu)造函數(shù)會被顯式的調(diào)用。

java.io.ObjectOutputStream代表對象輸出流,它的writeObject(Objectobj)方法可對參數(shù)指定的obj對象進(jìn)行序列化,把得到的字節(jié)序列寫到一個目標(biāo)輸出流中。

java.io.ObjectInputStream代表對象輸入流,它的readObject( )方法從一個源輸入流中讀取字節(jié)序列,再把它們反序列化為一個對象,并將其返回。

覆蓋 (Override) 和重載 (Overload)

Override:方法覆蓋是說子類重新定義了父類的方法,方法覆蓋必須有相同的方法名,參數(shù)列表和返回類型。一般會有個@Override注解。

Overload:Java中的方法重載發(fā)生在同一個類里面兩個或者是多個方法的方法名相同但是參數(shù)不同的情況

集合框架

對象存入集合時會變成Object類型,取出時需要類型轉(zhuǎn)換。所以會有泛型(這樣也不用考慮取出時的類型轉(zhuǎn)換了)。另外集合里存儲的是引用,所以泛型不能使用基本類型。

常見集合

集合概覽

集合家族一覽

Set 是一種 Collection,不過其中沒有重復(fù)的對象;List 也是一種 Collection,其中的元素按順序排列(不過可能有重復(fù))。

SortedSet 和 SortedMap 是特殊的集和映射,其中的元素按順序排列。

Collection、Set、List、Map、SortedSet 和 SortedMap 都是接口,不過 java.util 包定義了多個具體實現(xiàn),例如基于數(shù)組和鏈表的列表,基于哈希表或二叉樹的映射和集。除此之外,還有兩個重要的接口:Iterator 和 Iterable,用于遍歷集合中的對象。

Collection接口

Collection 接口是參數(shù)化接口,表示由泛型 E 對象組成的集合。這個接口定義了很多方法,用來把對象添加到集合中,把對象從集合中移除,測試對象是否在集合中,以及遍歷集合中的所有元素。還有一些方法可以把集合中的元素轉(zhuǎn)換成數(shù)組,以及返回集合的大小。

Set接口

集(set)是無重復(fù)對象組成的集合:不能有兩個引用指向同一個對象,或兩個指向 null 的引用,如果對象 a 和 b 的引用滿足條件 a.equals(b),那么這兩個對象也不能同時出現(xiàn)在集中。多數(shù)通用的 Set 實現(xiàn)都不會對元素排序,但并不禁止使用有序集(SortedSet 和 LinkedHashSet 就有順序)。而且集與列表等有序集合不同,一般認(rèn)為,集的 contains 方法,不論以常數(shù)時間還是以對數(shù)時間都為1,運行效率都高。

List接口

List 是一組有序的對象集合。列表中的每個元素都有特定的位置,而且 List 接口定義了一些方法,用于查詢或設(shè)定特定位置(或叫索引)的元素。從這個角度來看,List 對象和數(shù)組類似,不過列表的大小能按需變化,以適應(yīng)其中元素的數(shù)量。和集不同,列表允許出現(xiàn)重復(fù)的元素。

除了基于索引的 get( ) 和 set( ) 方法之外,List 接口還定義了一些方法,用于把元素添加到特定的索引,把元素從特定的索引移除,或者返回指定值在列表中首次出現(xiàn)或最后出現(xiàn)的索引。從 Collection 接口繼承的 add( ) 和 remove( ) 方法,前者把元素添加到列表末尾,后者把指定值從列表中首次出現(xiàn)的位置移除。繼承的 addAll( ) 方法把指定集合中的所有元素添加到列表的末尾,或者插入指定的索引。retainAll( ) 和 removeAll( ) 方法的表現(xiàn)與其他 Collection 對象一樣,如果需要,會保留或刪除多個相同的值。

List 接口沒有定義操作索引范圍的方法,但是定義了一個 subList( ) 方法。這個方法返回一個 List 對象,表示原列表指定范圍內(nèi)的元素。子列表會回饋父列表,只要修改了子列表,父列表立即就能察覺到變化。

Map接口

映射(map)是一系列鍵值對,一個鍵對應(yīng)一個值。Map 接口定義了用于定義和查詢映射的 API。Map 接口屬于 Java 集合框架,但沒有擴(kuò)展 Collection 接口,因此 Map 只是一種集合,而不是 Collection 類型。Map 是參數(shù)化類型,有兩個類型變量。類型變量 K 表示映射中鍵的類型,類型變量 V 表示鍵對應(yīng)的值的類型。例如,如果有個映射,其鍵是 String 類型,對應(yīng)的值是 Integer 類型,那么這個映射可以表示為 Map

Map 接口定義了幾個最有用的方法:put( ) 方法定義映射中的一個鍵值對,get( ) 方法查詢指定鍵對應(yīng)的值,remove( ) 方法把指定的鍵及對應(yīng)的值從映射中刪除。一般來說,實現(xiàn) Map 接口的類都要能高效執(zhí)行這三個基本方法:一般應(yīng)該運行在常數(shù)時間中,而且絕不能比在對數(shù)時間中運行的性能差。

Map 的重要特性之一是,可以視作集合。雖然 Map 對象不是 Collection 類型,但映射的鍵可以看成 Set 對象,映射的值可以看成 Collection 對象,而映射的鍵值對可以看成由 Map.Entry 對象組成的 Set 對象。(Map.Entry 是 Map 接口中定義的嵌套接口,表示一個鍵值對。)

Queue接口和BlockingQueue接口

隊列(queue)是一組有序的元素,提取元素時按順序從隊頭讀取。隊列一般按照插入元素的順序?qū)崿F(xiàn),因此分成兩類:先進(jìn)先出(first-in, first-out,F(xiàn)IFO)隊列和后進(jìn)先出(last-in, first-out,LIFO)隊列。

LIFO 隊列也叫棧(stack),Java 提供了 Stack 類,但強(qiáng)烈不建議使用,應(yīng)該使用實現(xiàn) Deque 接口的類。

隊列也可以使用其他順序:優(yōu)先隊列(priority queue)根據(jù)外部 Comparator 對象或 Comparable 類型元素的自然順序排序元素。與 Set 不同的是,Queue 的實現(xiàn)往往允許出現(xiàn)重復(fù)的元素。而與 List 不同的是,Queue 接口沒有定義處理任意索引位元素的方法,只有隊列的頭一個元素能訪問。Queue 的所有實現(xiàn)都要具有一個固定的容量:隊列已滿時,不能再添加元素。類似地,隊列為空時,不能再刪除元素。很多基于隊列的算法都會用到滿和空這兩個狀態(tài),所以 Queue 接口定義的方法通過返回值表明這兩個狀態(tài),而不會拋出異常。具體而言,peek( ) 和 poll( ) 方法返回 null 表示隊列為空。因此,多數(shù) Queue 接口的實現(xiàn)不允許用 null 作元素。

阻塞式隊列(blocking queue)是一種定義了阻塞式 put( ) 和 take( ) 方法的隊列。put( ) 方法的作用是把元素添加到隊列中,如果需要,這個方法會一直等待,直到隊列中有存儲元素的空間為止。而 take( ) 方法的作用是從隊頭移除元素,如果需要,這個方法會一直等待,直到隊列中有元素可供移除為止。阻塞式隊列是很多多線程算法的重要組成部分,因此 BlockingQueue 接口(擴(kuò)展 Queue 接口)在 java.util.concurrent 包中定義。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/65681.html

相關(guān)文章

  • 2019年2月份Github上收獲最多Star的10個Java項目

    摘要:特點高性能且易于使用,旨在實現(xiàn)簡單并快速的事務(wù)提交與回滾。大部分項目只需要少量的配置即可地址介紹是開源的診斷工具。當(dāng)然,它們的重點是,和。 該文已加入筆主的開源項目——JavaGuide(一份涵蓋大部分Java程序員所需要掌握的核心知識的文檔類項目),地址:https://github.com/Snailclimb... 。覺得不錯的話,記得點個Star。 1. JavaGuide ...

    xzavier 評論0 收藏0
  • 盤點一下Github上開源的Java面試/學(xué)習(xí)相關(guān)的倉庫

    摘要:最近瀏覽,收藏了一些還算不錯的面試學(xué)習(xí)相關(guān)的倉庫,分享給大家,希望對你有幫助。除了這九個倉庫,再推薦幾個不錯的學(xué)習(xí)方向的倉庫給大家。數(shù)高達(dá)的筆記后端尤其是程序員的學(xué)習(xí)倉庫兩個算法相關(guān)的倉庫,刷的小伙伴必備 最近瀏覽 Github ,收藏了一些還算不錯的 Java面試/學(xué)習(xí)相關(guān)的倉庫,分享給大家,希望對你有幫助。我暫且按照目前的 Star 數(shù)量來排序。 本文由 SnailClimb 整理...

    luxixing 評論0 收藏0
  • Android防護(hù)掃盲

    摘要:為了防止這種現(xiàn)象,我們可以對字節(jié)碼進(jìn)行混淆。動態(tài)鏈接庫是目標(biāo)文件的集合,目標(biāo)文件在動態(tài)鏈接庫中的組織方式是按照特殊方式形成的。 一、已知防護(hù)策略 1.不可或缺的混淆 Java 是一種跨平臺、解釋型語言,Java 源代碼編譯成的class文件中有大量包含語義的變量名、方法名的信息,很容易被反編譯為Java 源代碼。為了防止這種現(xiàn)象,我們可以對Java字節(jié)碼進(jìn)行混淆。混淆不僅能將代碼中的類...

    CastlePeaK 評論0 收藏0
  • 關(guān)于前端上傳文件全面基礎(chǔ)掃盲貼(五) ----- H5拖拽事件

    摘要:在中,拖放是標(biāo)準(zhǔn)的一部分,任何元素都能夠拖放。如果需要設(shè)置允許放置,我們必須阻止對元素的默認(rèn)處理方式方法。 系列文章 關(guān)于前端上傳文件全面基礎(chǔ)掃盲貼(零)關(guān)于前端上傳文件全面基礎(chǔ)掃盲貼(一) ----- XMLHttpRequest關(guān)于前端上傳文件全面基礎(chǔ)掃盲貼(二) ----- File關(guān)于前端上傳文件全面基礎(chǔ)掃盲貼(三) ----- FormData關(guān)于前端上傳文件全面基礎(chǔ)掃盲貼(...

    iKcamp 評論0 收藏0
  • 關(guān)于前端上傳文件全面基礎(chǔ)掃盲貼(二) ----- File

    摘要:為了解救上面說到的問題是向提交的一個草案,旨在推出一套標(biāo)準(zhǔn)的,其基本功能是實現(xiàn)用對本地文件進(jìn)行操作。出于安全性的考慮,該只對本地文件提供有限的訪問。 系列文章 關(guān)于前端上傳文件全面基礎(chǔ)掃盲貼(零)關(guān)于前端上傳文件全面基礎(chǔ)掃盲貼(一) ----- XMLHttpRequest關(guān)于前端上傳文件全面基礎(chǔ)掃盲貼(二) ----- File關(guān)于前端上傳文件全面基礎(chǔ)掃盲貼(三) ----- For...

    scola666 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<