您好,登錄后才能下訂單哦!
先總結一下Top10吧
這個列表總結了10個Java開發人員最常犯的錯誤。
當需要把Array
轉成ArrayList
的時候,開發人員經常這樣做:
List list = Arrays.asList(arr);
Arrays.asList()
會返回一個ArrayList
,但是要特別注意,這個ArrayList
是Arrays
類的靜態內部類,并不是java.util.ArrayList
類。
java.util.Arrays.ArrayList
類實現了set()
,get()
,contains()
方法,但是并沒有實現增加元素的方法(事實上是可以調用add方法,但是沒有具體實現,僅僅拋出UnsupportedOperationException
異常),因此它的大小也是固定不變的。為了創建一個真正的java.util.ArrayList
,你應該這樣做:
ArrayList arrayList = new ArrayList(Arrays.asList(arr));
ArrayList
的構造方法可以接收一個Collection
類型,而java.util.Arrays.ArrayList
已經實現了該接口。
開發人員經常這樣做:
Set set = new HashSet(Arrays.asList(arr));
return set.contains(targetValue);
以上代碼可以正常工作,但是沒有必要將其轉換成set
集合,將一個List
轉成Set
需要額外的時間,其實我們可以簡單的使用如下方法即可:
Arrays.asList(arr).contains(targetValue);
或者
for (String s: arr){
if(s.equals(targetValue))
return true;
}
return false;
第一種方法可讀性更強。
考慮如下代碼,在迭代期間刪除元素:
ArrayList list = new ArrayList(Arrays.asList("a", "b", "c","d"));
for (int i = 0; i
list.remove(i);
}
System.out.println(list);
結果打印:[b, d]
在上面這個方法中有一系列的問題,當一個元素被刪除的時候,list
大小減小,然后原先索引指向了其它元素。所以如果你想在循環里通過索引來刪除多個元素,將不會正確工作。
你也許知道使用迭代器是在循環里刪除元素的正確方式,或許你也知道foreach
循環跟迭代器很類似,但事實情況卻不是這樣,如下代碼:
ArrayList list = new ArrayList(Arrays.asList("a", "b", "c","d"));
for (String s : list) {
if (s.equals("a"))
list.remove(s);
}
將拋出ConcurrentModificationException
異常。
然而接下來的代碼卻是OK的:
ArrayList list = new ArrayList(Arrays.asList("a", "b", "c","d"));
Iterator iter = list.iterator();
while(iter.hasNext()) {
String s = iter.next();
if (s.equals("a")) {
iter.remove();
}
}
next()
方法需要在remove()
方法之前被調用,在foreach
循環里,編譯器會在刪除元素操作化調用next
方法,這導致了ConcurrentModificationException
異常。更多詳細信息,可以查看ArrayList.iterator()
的源碼。
從算法的角度來講,HashTable
是一種數據結構名稱。但是在Java
中,這種數據結構叫做HashMap
。
HashTable
與HashMap
的一個主要的區別是HashTable
是同步的,所以,通常來說,你會使用HashMap
,而不是Hashtable
。
在Java
中,原始類型(raw type
)和×××通配符類型很容易讓人混淆。舉個Set
的例子,Set
是原始類型,而Set
是×××通配符類型。
請看如下代碼,add
方法使用了一個原始類型的List
作為入參:
public static void add(List list, Object o){
list.add(o);
}
public static void main(String[] args){
List list = new ArrayList();
add(list, 10);
String s = list.get(0);
}
運行以上代碼將會拋出異常:
Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String at ...
使用原始類型集合非常危險,因為它跳過了泛型類型檢查,是不安全的。另外,Set
, Set
和Set
這三個有很大的不同。
開發人員經常使用public
修飾類字段,雖然這很容易讓別人直接通過引用獲取該字段的值,但這是一個不好的設計。根據經驗,應該盡可能的降低成員屬性的訪問級別。
為什么開發人員經常使用ArrayList
和LinkedList
,卻不知道他們之間的區別,因為它們看起來很像。然而它們之間有著巨大的性能差異。簡單的說,如果有大量的增加刪除操作并且沒有很多的隨機訪問元素的操作,應該首選LinkedList
。
不可變對象有很多優點,如簡單、安全等。但是對于每個不同的值都需要一個單獨的對象,太多的對象會引起大量垃圾回收,因此在選擇可變與不可變的時候,需要有一個平衡。
通常,可變對象用于避免產生大量的中間對象,一個經典的例子是大量字符串的拼接。如果你使用一個不可變對象,將會馬上產生大量符合垃圾回收標準的對象,這浪費了CPU大量的時間和精力。使用可變對象是正確的解決方案(StringBuilder
);
String result="";
for (String s: arr){
result = result + s;
}
另外,在有些其它情況下也是需要使用可變對象。例如往一個方法傳入一個可變對象,然后收集多種結果,而不需要寫太多的語法。另一個例子是排序和過濾:當然,你可以寫一個方法來接收原始的集合,并且返回一個排好序的集合,但是那樣對于大的集合就太浪費了。
class Super{
string s;
public Super(String s){
this.s=s;
}
}
public class Sub extends Super{
int x=200;
public Sub(String s){
}
public Sub(){
System.out.println("Sub");
}
public static void main(String[] args){
Sub s=new Sub();
}
}
之所以出現這個編譯錯誤,是因為父類的默認構造方法未定義。在Java中,如果一個類沒有定義構造方法,編譯器會默認插入一個無參數的構造方法;但是如果一個構造方法在父類中已定義,在這種情況,編譯器是不會自動插入一個默認的無參構造方法,這正是以上demo
的情況;
對于子類來說,不管是無參構造方法還是有參構造方法,都會默認調用父類的無參構造方法;當編譯器嘗試在子類中往這兩個構造方法插入super(
)方法時,因為父類沒有一個默認的無參構造方法,所以編譯器報錯;
要修復這個錯誤,很簡單:
public Super(){
System.out.println("Super");
}
super(value)
;有兩種創建字符串的方式:
String x = "abc";
String y = new String("abc");
它們之間有什么區別呢?
以下代碼提供了一個快速回答:
String a = "abcd";
String b = "abcd";
System.out.println(a == b);
// True
System.out.println(a.equals(b));
// True
String c = new String("abcd");
String d = new String("abcd");
System.out.println(c == d);
// False
System.out.println(c.equals(d));
// True
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。