国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Java > 正文

Java經典用法總結

2019-11-26 14:37:04
字體:
來源:轉載
供稿:網友

在Java編程中,有些知識并不能僅通過語言規范或者標準API文檔就能學到的,本文為大家羅列。

一、實現

1、現equals() 

class Person { String name; int birthYear; byte[] raw;  public boolean equals(Object obj) {  if (!obj instanceof Person)   return false;   Person other = (Person)obj;  return name.equals(other.name)    && birthYear == other.birthYear    && Arrays.equals(raw, other.raw); }  public int hashCode() { ... }}
  • 參數必須是Object類型,不能是外圍類。
  • foo.equals(null) 必須返回false,不能拋NullPointerException。(注意,null instanceof 任意類 總是返回false,因此上面的代碼可以運行。)
  • 基本類型域(比如,int)的比較使用 == ,基本類型數組域的比較使用Arrays.equals()。
  • 覆蓋equals()時,記得要相應地覆蓋 hashCode(),與 equals() 保持一致。

2、現hashCode()

class Person { String a; Object b; byte c; int[] d;  public int hashCode() {  return a.hashCode() + b.hashCode() + c + Arrays.hashCode(d); }  public boolean equals(Object o) { ... }}
  • 當x和y兩個對象具有x.equals(y) == true ,你必須要確保x.hashCode() == y.hashCode()。
  • 根據逆反命題,如果x.hashCode() != y.hashCode(),那么x.equals(y) == false 必定成立。
  • 你不需要保證,當x.equals(y) == false時,x.hashCode() != y.hashCode()。但是,如果你可以盡可能地使它成立的話,這會提高哈希表的性能。
  • hashCode()最簡單的合法實現就是簡單地return 0;雖然這個實現是正確的,但是這會導致HashMap這些數據結構運行得很慢。

3、實現compareTo() 

class Person implements Comparable<Person> { String firstName; String lastName; int birthdate;  // Compare by firstName, break ties by lastName, finally break ties by birthdate public int compareTo(Person other) {  if (firstName.compareTo(other.firstName) != 0)   return firstName.compareTo(other.firstName);  else if (lastName.compareTo(other.lastName) != 0)   return lastName.compareTo(other.lastName);  else if (birthdate < other.birthdate)   return -1;  else if (birthdate > other.birthdate)   return 1;  else   return 0; }}

總是實現泛型版本 Comparable 而不是實現原始類型 Comparable 。因為這樣可以節省代碼量和減少不必要的麻煩。
只關心返回結果的正負號(負/零/正),它們的大小不重要。
Comparator.compare()的實現與這個類似。

4、實現clone()

 class Values implements Cloneable { String abc; double foo; int[] bars; Date hired;  public Values clone() {  try {   Values result = (Values)super.clone();   result.bars = result.bars.clone();   result.hired = result.hired.clone();   return result;  } catch (CloneNotSupportedException e) { // Impossible   throw new AssertionError(e);  } }}
  • 使用 super.clone() 讓Object類負責創建新的對象。
  • 基本類型域都已經被正確地復制了。同樣,我們不需要去克隆String和BigInteger等不可變類型。
  • 手動對所有的非基本類型域(對象和數組)進行深度復制(deep copy)。
  • 實現了Cloneable的類,clone()方法永遠不要拋CloneNotSupportedException。因此,需要捕獲這個異常并忽略它,或者使用不受檢異常(unchecked exception)包裝它。
  • 不使用Object.clone()方法而是手動地實現clone()方法是可以的也是合法的。

二、預防性檢測

1、預防性檢測(Defensive checking)數值

int factorial(int n) { if (n < 0)  throw new IllegalArgumentException("Undefined"); else if (n >= 13)  throw new ArithmeticException("Result overflow"); else if (n == 0)  return 1; else  return n * factorial(n - 1);}
  • 不要認為輸入的數值都是正數、足夠小的數等等。要顯式地檢測這些條件。
  • 一個設計良好的函數應該對所有可能性的輸入值都能夠正確地執行。要確保所有的情況都考慮到了并且不會產生錯誤的輸出(比如溢出)。

2、預防性檢測對象

int findIndex(List<String> list, String target) { if (list == null || target == null)  throw new NullPointerException(); ...}
  • 不要認為對象參數不會為空(null)。要顯式地檢測這個條件。

3、預防性檢測數組索引 

void frob(byte[] b, int index) { if (b == null)  throw new NullPointerException(); if (index < 0 || index >= b.length)  throw new IndexOutOfBoundsException(); ...}

不要認為所以給的數組索引不會越界。要顯式地檢測它。

4、預防性檢測數組區間

void frob(byte[] b, int off, int len) { if (b == null)  throw new NullPointerException(); if (off < 0 || off > b.length  || len < 0 || b.length - off < len)  throw new IndexOutOfBoundsException(); ...}

不要認為所給的數組區間(比如,從off開始,讀取len個元素)是不會越界。要顯式地檢測它。

三、數組

1、填充數組元素
使用循環: 

// Fill each element of array 'a' with 123byte[] a = (...);for (int i = 0; i < a.length; i++) a[i] = 123;(優先)使用標準庫的方法:Arrays.fill(a, (byte)123);

2、復制一個范圍內的數組元素
使用循環:

// Copy 8 elements from array 'a' starting at offset 3// to array 'b' starting at offset 6,// assuming 'a' and 'b' are distinct arraysbyte[] a = (...);byte[] b = (...);for (int i = 0; i < 8; i++) b[6 + i] = a[3 + i];(優先)使用標準庫的方法:System.arraycopy(a, 3, b, 6, 8);

3、調整數組大小
使用循環(擴大規模): 

// Make array 'a' larger to newLenbyte[] a = (...);byte[] b = new byte[newLen];for (int i = 0; i < a.length; i++) // Goes up to length of A b[i] = a[i];a = b;

使用循環(減小規模):

// Make array 'a' smaller to newLenbyte[] a = (...);byte[] b = new byte[newLen];for (int i = 0; i < b.length; i++) // Goes up to length of B b[i] = a[i];a = b;

(優先)使用標準庫的方法:

1a = Arrays.copyOf(a, newLen);

4、把4個字節包裝(packing)成一個int

int packBigEndian(byte[] b) { return (b[0] & 0xFF) << 24    | (b[1] & 0xFF) << 16    | (b[2] & 0xFF) << 8    | (b[3] & 0xFF) << 0;} int packLittleEndian(byte[] b) { return (b[0] & 0xFF) << 0    | (b[1] & 0xFF) << 8    | (b[2] & 0xFF) << 16    | (b[3] & 0xFF) << 24;}

5、把int分解(Unpacking)成4個字節 

byte[] unpackBigEndian(int x) { return new byte[] {  (byte)(x >>> 24),  (byte)(x >>> 16),  (byte)(x >>> 8),  (byte)(x >>> 0) };} byte[] unpackLittleEndian(int x) { return new byte[] {  (byte)(x >>> 0),  (byte)(x >>> 8),  (byte)(x >>> 16),  (byte)(x >>> 24) };}

總是使用無符號右移操作符(>>>)對位進行包裝(packing),不要使用算術右移操作符(>>)。

以上就是本文的全部內容,希望對大家的學習有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 都江堰市| 于田县| 青神县| 长沙县| 唐山市| 婺源县| 贵南县| 孝感市| 宽甸| 乐亭县| 饶河县| 桂阳县| 大新县| 乡宁县| 宝山区| 武乡县| 冕宁县| 建德市| 抚宁县| 安岳县| 鹤岗市| 新乐市| 醴陵市| 庆城县| 庆元县| 通海县| 晋州市| 长顺县| 沁水县| 辽源市| 利津县| 随州市| 增城市| 横峰县| 新乐市| 五常市| 娄底市| 昌吉市| 道孚县| 旬阳县| 隆尧县|