?? java常見問題.txt
字號:
myFORM = processActionFORM(myFORM);
其中myFORM是一個對象, 它在傳遞到processActionFORM中去做處理的時候, 先用另外一個otherFORM去接了一下, 這個時候otherFORM也就指向myFORM的句柄, 而在外面再用myFORM去接返回值, myFORM這個時候其實還是指向原來的句柄。 雖然結果沒錯, 但這個時候有個問題就是同時有兩個別名指向同一個句柄。 執行其中任何一個別名的方法, 另一個別名的對象也隨著改變了。 所以上面的這個例子要盡量改寫為:
private void processActionFORM(SomeActionFORM myFORM) {
myFORM…..
…;
return otherFORM;
}
調用如下:
processActionFORM(myFORM);
特別在畫面之間傳遞參數的時候, 除非你能保證你的數據不被別人篡改, 否則你就得小心, 或者你需要重新clone一個對象傳過去, 然后取返回值。
注: 如果在EJB的client與server之間傳遞參數, 則不存在句柄的傳遞, 數據在server端修改之后client端并不改變。 因為它們是通過網絡進行Object字節流傳輸的, 不存在句柄地址相同的條件。 你盡可放心的使用, 最后如果client端需要再使用這個新的Object, 則需要再回傳過去。 (EJB2.0 支持local interface, 這種情況下有可能就跟前面所說的句柄傳遞一樣了, 沒試過)。
4.2 String的特殊性
String在Java中被設計成安全的String, 對于String的任一個操作都是先重新生成一個String的拷貝, 然后對這個拷貝進行操作。 所以String在參數傳遞的時候可以看作是值傳遞。 即如果你需要修改一個String并返回修改后的String, 你得要再去接一下返回值。 如:
String str = “This is a sample”;
str = editStr(str);
System.out.println(str); // “Here is a sample”
private String editStr(String str) {
String newStr = str. substring(4);
newStr = “Here” + newStr;
return newStr;
}
如果想用句柄傳遞, 可以使用String的內部操作使用的一個類StringBuffer, 對它的操作都是同一個對象上進行的, 所以效率也自然高一些。 上面的例子用StringBuffer改寫如下:
StringBuffer str = new StringBuffer(“This is a sample”);
editStr(str);
System.out.println(str); // “Here is a sample”
private void editStr(StringBuffer str) {
str.replace(0, 4, “Here”);
}
5. 例外處理
有些新手總是習慣于使用返回值來進行錯誤處理, 如果使用異常處理這種方法將會使程序結構更合理, 效率更高。 比如在Client端需要通過EJB來進行DB操作, Client端需要知道DB處理有沒有出錯就可以通過層層的向上拋Exception的方法, 一直到Client端需要處理的地方截住, 然后進行例外處理。 如:
后臺DB處理:
public static java.sql.Timestamp getDBSysdate() throws CSCWebException {
Connection conn = null;
Timestamp sysTime = null;
try {
conn = PJEJBSvrUtil.getWLPoolConnection();
sysTime = CommonDAO.getDBSysdate(conn);
} catch (SQLException ce) {
throw new CSCWebException(ce.getMessage());
} finally {
try {
if (conn != null) {
conn.close();
}
} catch (Exception e) {
throw new CSCWebException(e.getMessage());
}
}
return sysTime;
}
在前臺:
try {
commonIntf.getDBSysdate();
} catch (CSCWebException cex) {
cat.debug(“”, cex);
return getExceptionForward(cex);
}
要防止違例被漏處理, 除非是你肯定不需要處理的, 提倡在遇到Exception的時候就要往上拋, 由最終調用處來進行處理, 當然也不能一概而論, 視情況而定。 比如我想例外在方法體內就要解決掉, 給出一個CSC中出現的bug。
public boolean checkTelFORMat(String telNo) {
boolean error = false;
if ( telNo == null || telNo.equals("") ) {
error = true;
} else {
if ( ejb.util.StringUtil.chkPhone(telNo) ) {
error = false;
} else {
error = true;
}
}
if (telNo.startsWith("184") || telNo.startsWith("186")) {
if (telNo.length() == 3) {
error = true;
}
}
if ( error ) {
ObjMngr.showError("MCSTC001E");
cmbTelNo.requestFocus();
return false;
} else {
return true;
}
}
用了一個Flag來記住每次Check的結果, 然后在最后再出Error Dialog, 這是比較典型的C的寫法, 改寫如下:
public boolean checkTelFORMat(String telNo) {
try {
if (telNo == null || telNo.equals("")) {
throw new Exception();
} else {
if (!ejb.util.StringUtil.chkPhone(telNo)) {
throw new Exception();
}
}
if (telNo.startsWith("184") || telNo.startsWith("186")) {
if (telNo.length() == 3) {
throw new Exception();
}
}
} catch (Exception e) {
ObjMngr.showError("MCSTC001E");
cmbTelNo.requestFocus();
return false;
}
return true;
}
使用了拋Exception的方法, 在方法的最后截住, 這樣一遇到Error就能馬上處理掉, 從效率上講也是最高的。
6.數據庫操作的問題
根據現有的開發經驗, 一般我們在SessionBean(或不用EJB的時候的Module Bean)中取到Connection然后調用專門操作數據庫(DAO)中的方法, 也就是在Bean中進行Connection的打開與關閉操作, 而在DAO中進行Statement和ResultSet操作, 一般在最后都需要進行關閉它們。 接上節的例子, 在CommonDAO里面有這樣的一個方法:
public static java.sql.Timestamp getDBSysdate(Connection conn)
throws CSCWebException {
Statement stmt = null;
ResultSet rs = null;
Timestamp sysTime = null;
try {
stmt = conn.createStatement();
rs = stmt.executeQuery("SELECT SYSDATE FROM DUAL");
if (rs.next()) {
sysTime = rs.getTimestamp("SYSDATE");
}
} catch (SQLException ce) {
throw new CSCWebException(ce.getMessage());
} catch (Exception e) {
throw new CSCWebException(e.getMessage());
} finally {
try {
if (rs != null) {
rs.close();
}
if (stmt != null) {
stmt.close();
}
} catch (Exception e) {
throw new CSCWebException(e.getMessage());
}
}
return sysTime;
}
在finally里面的語句總是會被執行到的, 所以即使上面拋出了SQLException也會去執行stmt與rs的close操作的。
7. index越界
index越界包括很多:
Ø String的index
String的index從0開始, 最大為它的字符長度。常用的為substring這個方法:
String str = “This is a sample”;
str.substring(5, 7); 返回 is;
str.substring(15, 16) 或 str.substring(15); 都返回最后一個字符 e;
str.substring(16, 17); 這個會拋java.lang.StringIndexOutOfBoundsException錯誤;
Ø 數組下標
數組下標也由0開始, 最大為長度-1, 例:
int[] columnLen = new int[]{20, 30, 40, 30, 25, 50, 65, 100};
columnLen[0]為20;
columnLen[7]為100;
columnLen[8]會拋java.lang.ArrayIndexOutOfBoundsException錯誤。
Ø Vector, ArrayList等Collection的size
Vector與ArrayList同屬于List, 它們都是有序的集合體, 下標也都是從0開始, 最大也是長度-1, 跟數組不同的是, 它們的元素必須都是Object, 但可以為不同類型的對象, 不過在取出之后得要進行類型轉換。 而數組得要所有元素的類型相同。 ArrayList在構造之后, 并不存在元素的情況下, 如果調用set(index, Object)就會出錯。得要先進行add(Object)才行。 例:
ArrayList list = new ArrayList(10); // 這個地方的10只是list的初始容量, 并不代表它具有了10個元素, 這個跟數組不同, 數組在這種情況下具有了10個初始值, 初始值跟具體的元素類型有關, 一般的對象為null;
list.set(0, “First”); // 會拋java.lang.IndexOutOfBoundsException
list.add(“First”);
list.set(0, “new First”); // 正確, 因為位置0已經存在元素
8.其它
具體到每個項目, 不同的API使用都還有可能遇到一些共通的問題。 需要在項目開始的時候進行必要的培訓, 往往給出一個好的sample會事半功倍。
還用一些問題在此不再詳細描述:
ü 大小寫問題(String.equals()與String.equalsIgnoreCase(), 變量名等)
ü 括號匹配問題
ü 。。。
9.有待討論的問題
9.1 方法返回地點
private int getStatus( String sFlag ){
if (CodeBook.STATUS_OK_NAME.equals( sFlag )) {
return CodeBook.STATUS_OK_value;
} else if (CodeBook.STATUS_OK1_NAME.equals( sFlag )) {
return CodeBook.STATUS_OK1_value;
} else if (CodeBook.STATUS_NG_NAME.equals( sFlag )) {
return CodeBook.STATUS_NG_value;
}
return CodeBook.STATUS_NG_value;
}
與
private int getStatus(String sFlag) {
int iStatus = CodeBook.STATUS_NG_value;
if (CodeBook.STATUS_OK_NAME.equals(sFlag)) {
iStatus = CodeBook.STATUS_OK_value;
} else if (CodeBook.STATUS_OK1_NAME.equals(sFlag)) {
iStatus = CodeBook.STATUS_OK1_value;
} else if (CodeBook.STATUS_NG_NAME.equals(sFlag)) {
iStatus = CodeBook.STATUS_NG_value;
}
return iStatus;
}
9.2 變量聲明點
例:
for (int i = 0; i < 10; i++) {
String sTmp = String.valueOf(i);
...
}
與
String sTmp = null;
for (int i = 0; i < 10; i++) {
sTmp = String.valueOf(i);
...
}
--------------------------------------------------------------------------------
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -