step into就是單步執行,遇到子函數就進入并且繼續單步執行;
step over是在單步執行時,在函數內遇到子函數時不會進入子函數內單步執行,而是將子函數整個執行完再停止,也就是把子函數整個作為一步。有一點,經過我們簡單的調試,在不存在子函數的情況下是和step into效果一樣的。
step out就是但單步執行到子函數內時,用step out就可以執行完子函數余下部分,并返回到上一層函
數。
step into:進入子函數
step over:越過子函數,但子函數會執行
step out:跳出子函數
posted @
2008-08-13 18:36 竹子 閱讀(10978) |
評論 (0) |
編輯 收藏
如何計算某一天是星期幾?
—— 蔡勒(Zeller)公式
歷史上的某一天是星期幾?未來的某一天是星期幾?關于這個問題,有很多計算公式(兩個通用計算公式和一些分段計算公式),其中最著名的是蔡勒(Zeller)公式。即w=y+[y/4]+[c/4]-2c+[26(m+1)/10]+d-1
公式中的符號含義如下,w:星期;c:世紀-1;y:年(兩位數);m:月(m大于等于3,小于等于14,即在蔡勒公式中,某年的1、2月要看作上一年的13、14月來計算,比如2003年1月1日要看作2002年的13月1日來計算);d:日;[ ]代表取整,即只要整數部分。(C是世紀數減一,y是年份后兩位,M是月份,d是日數。1月和2月要按上一年的13月和 14月來算,這時C和y均按上一年取值。)
算出來的W除以7,余數是幾就是星期幾。如果余數是0,則為星期日。
以2049年10月1日(100周年國慶)為例,用蔡勒(Zeller)公式進行計算,過程如下:
蔡勒(Zeller)公式:w=y+[y/4]+[c/4]-2c+[26(m+1)/10]+d-1
=49+[49/4]+[20/4]-2×20+[26× (10+1)/10]+1-1
=49+[12.25]+5-40+[28.6]
=49+12+5-40+28
=54 (除以7余5)
即2049年10月1日(100周年國慶)是星期5。
你的生日(出生時、今年、明年)是星期幾?不妨試一試。
不過,以上公式只適合于1582年10月15日之后的情形(當時的羅馬教皇將愷撒大帝制訂的儒略歷修改成格里歷,即今天使用的公歷)。
過程的推導:(對推理不感興趣的可略過不看)
星期制度是一種有古老傳統的制度。據說因為《圣經·創世紀》中規定上帝用了六
天時間創世紀,第七天休息,所以人們也就以七天為一個周期來安排自己的工作和生
活,而星期日是休息日。從實際的角度來講,以七天為一個周期,長短也比較合適。所
以盡管中國的傳統工作周期是十天(比如王勃《滕王閣序》中說的“十旬休暇”,即是
指官員的工作每十日為一個周期,第十日休假),但后來也采取了西方的星期制度。
在日常生活中,我們常常遇到要知道某一天是星期幾的問題。有時候,我們還想知
道歷史上某一天是星期幾。通常,解決這個方法的有效辦法是看日歷,但是我們總不會
隨時隨身帶著日歷,更不可能隨時隨身帶著幾千年的萬年歷。假如是想在計算機編程中
計算某一天是星期幾,預先把一本萬年歷存進去就更不現實了。這時候是不是有辦法通
過什么公式,從年月日推出這一天是星期幾呢?
答案是肯定的。其實我們也常常在這樣做。我們先舉一個簡單的例子。比如,知道
了2004年5月1日是星期六,那么2004年5月31日“世界無煙日”是星期幾就不難推算出
來。我們可以掰著指頭從1日數到31日,同時數星期,最后可以數出5月31日是星期一。
其實運用數學計算,可以不用掰指頭。我們知道星期是七天一輪回的,所以5月1日是星
期六,七天之后的5月8日也是星期六。在日期上,8-1=7,正是7的倍數。同樣,5月15
日、5月22日和5月29日也是星期六,它們的日期和5月1日的差值分別是14、21和28,也
都是7的倍數。那么5月31日呢?31-1=30,雖然不是7的倍數,但是31除以7,余數為2,
這就是說,5月31日的星期,是在5月1日的星期之后兩天。星期六之后兩天正是星期一。
這個簡單的計算告訴我們計算星期的一個基本思路:首先,先要知道在想算的日子
之前的一個確定的日子是星期幾,拿這一天做為推算的標準,也就是相當于一個計算的
“原點”。其次,知道想算的日子和這個確定的日子之間相差多少天,用7除這個日期
的差值,余數就表示想算的日子的星期在確定的日子的星期之后多少天。如果余數是
0,就表示這兩天的星期相同。顯然,如果把這個作為“原點”的日子選為星期日,那
么余數正好就等于星期幾,這樣計算就更方便了。
但是直接計算兩天之間的天數,還是不免繁瑣。比如1982年7月29日和2004年5月
1日之間相隔7947天,就不是一下子能算出來的。它包括三段時間:一,1982年7月29
日以后這一年的剩余天數;二,1983-2003這二十一個整年的全部天數;三,從2004年
元旦到5月1日經過的天數。第二段比較好算,它等于21*365+5=7670天,之所以要加
5,是因為這段時間內有5個閏年。第一段和第三段就比較麻煩了,比如第三段,需要把
5月之前的四個月的天數累加起來,再加上日期值,即31+29+31+30+1=122天。同理,第
一段需要把7月之后的五個月的天數累加起來,再加上7月剩下的天數,一共是155天。
所以總共的相隔天數是122+7670+155=7947天。
仔細想想,如果把“原點”日子的日期選為12月31日,那么第一段時間也就是一個
整年,這樣一來,第一段時間和第二段時間就可以合并計算,整年的總數正好相當于兩
個日子的年份差值減一。如果進一步把“原點”日子選為公元前1年12月31日(或者天文
學家所使用的公元0年12月31日),這個整年的總數就正好是想算的日子的年份減一。這
樣簡化之后,就只須計算兩段時間:一,這么多整年的總天數;二,想算的日子是這一
年的第幾天。巧的是,按照公歷的年月設置,這樣反推回去,公元前1年12月31日正好是
星期日,也就是說,這樣算出來的總天數除以7的余數正好是星期幾。那么現在的問題就
只有一個:這么多整年里面有多少閏年。這就需要了解公歷的置閏規則了。
我們知道,公歷的平年是365天,閏年是366天。置閏的方法是能被4整除的年份在
2月加一天,但能被100整除的不閏,能被400整除的又閏。因此,像1600、2000、2400
年都是閏年,而1700、1800、1900、2100年都是平年。公元前1年,按公歷也是閏年。
因此,對于從公元前1年(或公元0年)12月31日到某一日子的年份Y之間的所有整年
中的閏年數,就等于
[(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400],
[...]表示只取整數部分。第一項表示需要加上被4整除的年份數,第二項表示需要去掉
被100整除的年份數,第三項表示需要再加上被400整除的年份數。之所以Y要減一,這
樣,我們就得到了第一個計算某一天是星期幾的公式:
W = (Y-1)*365 + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] + D. (1)
其中D是這個日子在這一年中的累積天數。算出來的W就是公元前1年(或公元0年)12月
31日到這一天之間的間隔日數。把W用7除,余數是幾,這一天就是星期幾。比如我們來
算2004年5月1日:
W = (2004-1)*365 + [(2004-1)/4] - [(2004-1)/100] + [(2004-1)/400] +
(31+29+31+30+1)
= 731702,
731702 / 7 = 104528……6,余數為六,說明這一天是星期六。這和事實是符合的。
上面的公式(1)雖然很準確,但是計算出來的數字太大了,使用起來很不方便。仔
細想想,其實這個間隔天數W的用數僅僅是為了得到它除以7之后的余數。這啟發我們是
不是可以簡化這個W值,只要找一個和它余數相同的較小的數來代替,用數論上的術語
來說,就是找一個和它同余的較小的正整數,照樣可以計算出準確的星期數。
顯然,W這么大的原因是因為公式中的第一項(Y-1)*365太大了。其實,
(Y-1)*365 = (Y-1) * (364+1)
= (Y-1) * (7*52+1)
= 52 * (Y-1) * 7 + (Y-1),
這個結果的第一項是一個7的倍數,除以7余數為0,因此(Y-1)*365除以7的余數其實就
等于Y-1除以7的余數。這個關系可以表示為:
(Y-1)*365 ≡ Y-1 (mod 7).
其中,≡是數論中表示同余的符號,mod 7的意思是指在用7作模數(也就是除數)的情
況下≡號兩邊的數是同余的。因此,完全可以用(Y-1)代替(Y-1)*365,這樣我們就得到
了那個著名的、也是最常見到的計算星期幾的公式:
W = (Y-1) + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] + D. (2)
這個公式雖然好用多了,但還不是最好用的公式,因為累積天數D的計算也比較麻
煩。是不是可以用月份數和日期直接計算呢?答案也是肯定的。我們不妨來觀察一下各
個月的日數,列表如下:
月 份:1月 2月 3月 4月 5月 6月 7月 8月 9月 10月 11月 12月
--------------------------------------------------------------------------
天 數: 31 28(29) 31 30 31 30 31 31 30 31 30 31
如果把這個天數都減去28(=4*7),不影響W除以7的余數值。這樣我們就得到另一張
表:
月 份:1月 2月 3月 4月 5月 6月 7月 8月 9月 10月 11月 12月
------------------------------------------------------------------------
剩余天數: 3 0(1) 3 2 3 2 3 3 2 3 2 3
平年累積: 3 3 6 8 11 13 16 19 21 24 26 29
閏年累積: 3 4 7 9 12 14 17 20 22 25 27 30
仔細觀察的話,我們會發現除去1月和2月,3月到7月這五個月的剩余天數值是3,2,3,2,
3;8月到12月這五個月的天數值也是3,2,3,2,3,正好是一個重復。相應的累積天數中,
后一月的累積天數和前一月的累積天數之差減去28就是這個重復。正是因為這種規律的
存在,平年和閏年的累積天數可以用數學公式很方便地表達:
╭ d; ?。ó擬=1)
D = { 31 + d; (當M=2) (3)
╰ [ 13 * (M+1) / 5 ] - 7 + (M-1) * 28 + d + i. (當M≥3)
其中[...]仍表示只取整數部分;M和d分別是想算的日子的月份和日數;平年i=0,閏年
i=1。對于M≥3的表達式需要說明一下:[13*(M+1)/5]-7算出來的就是上面第二個表中的
平年累積值,再加上(M-1)*28就是想算的日子的月份之前的所有月份的總天數。這是一
個很巧妙的辦法,利用取整運算來實現3,2,3,2,3的循環。比如,對2004年5月1日,有:
D = [ 13 * (5+1) / 5 ] - 7 + (5-1) * 28 + 1 + 1
= 122,
這正是5月1日在2004年的累積天數。
假如,我們再變通一下,把1月和2月當成是上一年的“13月”和“14月”,不僅仍
然符合這個公式,而且因為這樣一來,閏日成了上一“年”(一共有14個月)的最后一
天,成了d的一部分,于是平閏年的影響也去掉了,公式就簡化成:
D = [ 13 * (M+1) / 5 ] - 7 + (M-1) * 28 + d. (3≤M≤14) (4)
上面計算星期幾的公式,也就可以進一步簡化成:
W = (Y-1) + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] + [ 13 * (M+1) / 5 ] - 7
+ (M-1) * 28 + d.
因為其中的-7和(M-1)*28兩項都可以被7整除,所以去掉這兩項,W除以7的余數不變,
公式變成:
W = (Y-1) + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] + [ 13 * (M+1) / 5 ] + d.
(5)
當然,要注意1月和2月已經被當成了上一年的13月和14月,因此在計算1月和2月的日子
的星期時,除了M要按13或14算,年份Y也要減一。比如,2004年1月1日是星期四,用這
個公式來算,有:
W = (2003-1) + [(2003-1)/4] - [(2003-1)/100] + [(2003-1)/400] + [13*(13+1)/5]
+ 1
= 2002 + 500 - 20 + 5 + 36 + 1
= 2524;
2524 / 7 = 360……4.這和實際是一致的。
公式(5)已經是從年、月、日來算星期幾的公式了,但它還不是最簡練的,對于年
份的處理還有改進的方法。我們先來用這個公式算出每個世紀第一年3月1日的星期,列
表如下:
年份: 1(401,801,…,2001) 101(501,901,…,2101)
--------------------------------------------------------------------
星期: 4 2
====================================================================
年份:201(601,1001,…,2201) 301(701,1101,…,2301)
--------------------------------------------------------------------
星期: 0 5
可以看出,每隔四個世紀,這個星期就重復一次。假如我們把301(701,1101,…,2301)
年3月1日的星期數看成是-2(按數論中對余數的定義,-2和5除以7的余數相同,所以可
以做這樣的變換),那么這個重復序列正好就是一個4,2,0,-2的等差數列。據此,我們
可以得到下面的計算每個世紀第一年3月1日的星期的公式:
W = (4 - C mod 4) * 2 - 4. (6)
式中,C是該世紀的世紀數減一,mod表示取模運算,即求余數。比如,對于2001年3月
1日,C=20,則:
W = (4 - 20 mod 4) * 2 - 4
= 8 - 4
= 4.
把公式(6)代入公式(5),經過變換,可得:
(Y-1) + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] ≡ (4 - C mod 4) * 2 - 1
(mod 7). (7)
因此,公式(5)中的(Y-1) + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400]這四項,在計算
每個世紀第一年的日期的星期時,可以用(4 - C mod 4) * 2 - 1來代替。這個公式寫
出來就是:
W = (4 - C mod 4) * 2 - 1 + [13 * (M+1) / 5] + d. (8)
有了計算每個世紀第一年的日期星期的公式,計算這個世紀其他各年的日期星期的公式
就很容易得到了。因為在一個世紀里,末尾為00的年份是最后一年,因此就用不著再考
慮“一百年不閏,四百年又閏”的規則,只須考慮“四年一閏”的規則。仿照由公式(1)
簡化為公式(2)的方法,我們很容易就可以從式(8)得到一個比公式(5)更簡單的計算任意
一天是星期幾的公式:
W = (4 - C mod 4) * 2 - 1 + (y-1) + [y/4] + [13 * (M+1) / 5] + d. (9)
式中,y是年份的后兩位數字。
如果再考慮到取模運算不是四則運算,我們還可以把(4 - C mod 4) * 2進一步改寫
成只含四則運算的表達式。因為世紀數減一C除以4的商數q和余數r之間有如下關系:
4q + r = C,
其中r即是 C mod 4,因此,有:
r = C - 4q
= C - 4 * [C/4]. (10)
則
(4 - C mod 4) * 2 = (4 - C + 4 * [C/4]) * 2
= 8 - 2C + 8 * [C/4]
≡ [C/4] - 2C + 1 (mod 7). (11)
把式(11)代入(9),得到:
W = [C/4] - 2C + y + [y/4] + [13 * (M+1) / 5] + d - 1. (12)
這個公式由世紀數減一、年份末兩位、月份和日數即可算出W,再除以7,得到的余數是
幾就表示這一天是星期幾,唯一需要變通的是要把1月和2月當成上一年的13月和14月,
C和y都按上一年的年份取值。因此,人們普遍認為這是計算任意一天是星期幾的最好的
公式。這個公式最早是由德國數學家克里斯蒂安·蔡勒(Christian Zeller, 1822-
1899)在1886年推導出的,因此通稱為蔡勒公式(Zeller’s Formula)。為方便口算,
式中的[13 * (M+1) / 5]也往往寫成[26 * (M+1) / 10]。
現在仍然讓我們來算2004年5月1日的星期,顯然C=20,y=4,M=5,d=1,代入蔡勒
公式,有:
W = [20/4] - 40 + 4 + 1 + [13 * (5+1) / 5] + 1 - 1
= -15.
注意負數不能按習慣的余數的概念求余數,只能按數論中的余數的定義求余。為了方便
計算,我們可以給它加上一個7的整數倍,使它變為一個正數,比如加上70,得到55。
再除以7,余6,說明這一天是星期六。這和實際是一致的,也和公式(2)計算所得的結
果一致。
最后需要說明的是,上面的公式都是基于公歷(格里高利歷)的置閏規則來考慮
的。對于儒略歷,蔡勒也推出了相應的公式是:
W = 5 - C + y + [y/4] + [13 * (M+1) / 5] + d - 1. (13)
這樣,我們終于一勞永逸地解決了不查日歷計算任何一天是星期幾的問題。
posted @
2008-08-13 13:46 竹子 閱讀(509) |
評論 (0) |
編輯 收藏
import java.security.MessageDigest;
/**
* MD5加密類
* @author zhang
*/
public class MD5Encoding
{
/**
*
*
*/
private MD5Encoding()
{
}
/**
* 加密算法MD5
*
* @param text 明文
* @return String 密文
*/
public final static String encoding(String text)
{
char hexDigits[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};
String encodingStr = null;
try
{
byte[] strTemp = text.getBytes();
MessageDigest mdTemp = MessageDigest.getInstance("MD5");
mdTemp.update(strTemp);
byte[] md = mdTemp.digest();
int j = md.length;
char str[] = new char[j * 2];
int k = 0;
for (int i = 0; i < j; i++)
{
byte byte0 = md[i];
str[k++] = hexDigits[byte0 >>> 4 & 0xf];
str[k++] = hexDigits[byte0 & 0xf];
}
encodingStr = new String(str);
}
catch (Exception e)
{
}
return encodingStr;
}
public static void main(String[] areg)
{
MD5Encoding md5 = new MD5Encoding();
md5.encoding("admin");
}
}
posted @
2008-08-12 22:55 竹子 閱讀(180) |
評論 (0) |
編輯 收藏
1, Tomcat內存參數配置
如果是Win與Linux系統啟動Tomcat服務器,需要在tomcat/bin/catalina.sh與/tomcat/bin/catalina.bat兩個文件:在兩個文件里面加如:
SET CATALINA_OPTS= -Xms64m -Xmx128m
2, Tomcat連接數配置
配置Tomcat連接數.需要在Server.xml文件里面加如:
maxThreads="150" 表示最多同時處理150個連接
minSpareThreads="25" 表示即使沒有人使用也開這么多空線程等待
maxSpareThreads="75" 表示如果最多可以空75個線程,例如某時刻有80人訪問,之后沒有人訪問了,則tomcat不會保留80個空線程,而是關閉5個空的。
acceptCount="100" 當同時連接的人數達到maxThreads時,還可以接收排隊的連接,超過這個連接的則直接返回拒絕連接。
connectionTimeout="20000" maxThreads="150"
minSpareThreads="25"
maxSpareThreads="75"
enableLookups="false"
acceptCount="100" debug="0"
disableUploadTimeout="true"
redirectPort="8443" URIEncoding="UTF-8"/>
URIEncoding="UTF-8" 是設定JSP編碼格式.
posted @
2008-08-12 22:46 竹子 閱讀(1332) |
評論 (0) |
編輯 收藏
我倡議的是奧運會如有幸能到奧運會現場的朋友在巴基斯坦運動員入場時能起立給他們十二分的掌聲.全世界只有他們在我們的地震災害是搬空了自己的戰略儲備帳篷來支援我們.在全世界要制裁中國時,只有巴基斯坦和古巴投了反對票,而我們用幾十萬志愿軍的生命卻只換來了朝鮮的棄權票!在那以后巴基斯坦和古巴也遭了了美國長時間瘋狂的制裁!在那一刻中國才知道什么是患難與共的好兄弟!雖然那是很早以前的事情了!但是中國人不能忘記!
許多人都向地震中的汶川伸出援手,其中也有我們的鄰邦。“動用國家全部的運輸機”、“搬光國家戰略儲備的帳篷”這種傾囊相助的國家,就是我們堅定的盟友-巴基斯坦。
巴基斯坦在汶川中的感動還不僅僅是這些。當巴基斯坦的首批醫療隊趕赴災區的時候,我方的接待人員想給醫療隊員做出妥善的生活安排的時候,他們的領隊回道:不用操心醫療隊的吃飯、接待問題,他們是去救人的,有菜葉吃就行。
奧運會在北京召開,屆時世界各國的領導和世界各國的運動員都會到場.這是中國人像外展示自己的時候!我們要讓世界知道中國是懂得感恩的民族.讓巴基斯坦的民眾知道中國的民眾也是把他們看做是自己的兄弟!我們要用這種方式像他們表示最誠摯的謝意!讓他們感覺到不管他們會遇到什么樣的困難,中國人永遠會和他們站在一起!雖然我們和他們的長相和人種不同,但我們的心在一起!也讓美國人知道什么才是和諧世界,和諧中國.讓中國和巴基斯坦的友誼成為世界的典范!也讓巴鐵的兄弟們知道在中國遇到災難時他們能搬空自己的戰略儲備來支援中國,中國也能在他們遇到災難時付出的更多!
古人云:"滴水之恩,當涌泉相報"奧運會就是中國報答巴基斯坦的最佳時機?;茧y見真情。巴基斯坦的朋友在我們需要幫助的時候,給與我們最及時的幫助,這些援助對他們而言就是傾其所有。
同時我們應像社會招集一些口號之類的東西在奧運會開幕式時展示出來.讓那些亡我中國之心不死的人,感受到中國人團結的力量.也讓那些和中國人一起的人們感受一下中國人的團結讓他們更堅定的和中國站在一起.
巴基斯坦,中國歡迎你!北京歡迎你!雖去不了北京在電視屏幕前的我,也會起立,鼓掌!
希望廣大網友轉帖!
posted @
2008-07-25 13:52 竹子 閱讀(103) |
評論 (0) |
編輯 收藏
- “一切從名字開始”。無論是開發大的項目,寫學習代碼或打patch,都要注意給項目、變量、函數起一個好名字。
- “不要COPY第二次”,當一個常量用到第二次的時候,就立刻給它做預定義。比如起一個常量名或用一個變量賦值。
- “不要以誤小而為之”。“這點小錯無所謂”的態度對自己百害而無一利。
- “不以善小而不為”。你有下面的情況嗎?“修個小BUG沒意思”,“隨便起個名字無所謂”,“懶得提煉函數,直接復制代碼”。如果有必須改。
- “工欲善其事,必先利其器”,一定要把自己的工作環境打理好。
- “精益求精”,代碼不簡陋,不冗余。html頁面要對齊,配置文件不混亂,執行效率不得過且過。
- “切喜新厭舊”。多鉆研開源代碼,多研究學習別人的代碼。
- “重構 、 測試 、調試”。這三件法寶,是你提髙開發能力的利器。
無論JAVA還是.net、erlang還是C都有前景,關鍵是要做深做透,掌握正確獲取知識的方法、處理項目的能力是相通的,具體的說就是不要把目光盯在做硬件還是做軟件上,用java還是C,要勤動手打好基礎,提高自己對系統總體設計的能力,從系統的眼光看問題。為什么都是做JAVA的有的畢業拿3000,有的5000、8000,除了運氣和關系外,重要的是你對事物的認識深度和高度。我一直都記住這句話:有前途的人做什么都有前途,沒前途的人做什么都沒前途。
posted @
2008-07-22 18:05 竹子 閱讀(126) |
評論 (0) |
編輯 收藏
做個記錄,以備使用.
0x1 鼠標左鍵
0x2 鼠標右鍵
0x3 CANCEL 鍵
0x4 鼠標中鍵
0x8 BACKSPACE 鍵
0x9 TAB 鍵
0xC CLEAR 鍵
0xD ENTER 鍵
0x10 SHIFT 鍵
0x11 CTRL 鍵
0x12 MENU 鍵
0x13 PAUSE 鍵
0x14 CAPS LOCK 鍵
0x1B ESC 鍵
0x20 SPACEBAR 鍵
0x21 PAGE UP 鍵
0x22 PAGE DOWN 鍵
0x23 END 鍵
0x24 HOME 鍵
0x25 LEFT ARROW 鍵
0x26 UP ARROW 鍵
0x27 RIGHT ARROW 鍵
0x28 DOWN ARROW 鍵
0x29 SELECT 鍵
0x2A PRINT SCREEN 鍵
0x2B EXECUTE 鍵
0x2C SNAPSHOT 鍵
0x2D INSERT 鍵
0x2E DELETE 鍵
0x2F HELP 鍵
0x90 NUM LOCK 鍵
A 至 Z 鍵與 A – Z 字母的 ASCII 碼相同:
值 描述
65 A 鍵
66 B 鍵
67 C 鍵
68 D 鍵
69 E 鍵
70 F 鍵
71 G 鍵
72 H 鍵
73 I 鍵
74 J 鍵
75 K 鍵
76 L 鍵
77 M 鍵
78 N 鍵
79 O 鍵
80 P 鍵
81 Q 鍵
82 R 鍵
83 S 鍵
84 T 鍵
85 U 鍵
86 V 鍵
87 W 鍵
88 X 鍵
89 Y 鍵
90 Z 鍵
0 至 9 鍵與數字 0 – 9 的 ASCII 碼相同:
值 描述
48 0 鍵
49 1 鍵
50 2 鍵
51 3 鍵
52 4 鍵
53 5 鍵
54 6 鍵
55 7 鍵
56 8 鍵
57 9 鍵
下列常數代表數字鍵盤上的鍵:
值 描述
0x60 0 鍵
0x61 1 鍵
0x62 2 鍵
0x63 3 鍵
0x64 4 鍵
0x65 5 鍵
0x66 6 鍵
0x67 7 鍵
0x68 8 鍵
0x69 9 鍵
0x6A MULTIPLICATION SIGN (*) 鍵
0x6B PLUS SIGN (+) 鍵
0x6C ENTER 鍵
0x6D MINUS SIGN (–) 鍵
0x6E DECIMAL POINT (.) 鍵
0x6F DIVISION SIGN (/) 鍵
下列常數代表功能鍵:
值 描述
0x70 F1 鍵
0x71 F2 鍵
0x72 F3 鍵
0x73 F4 鍵
0x74 F5 鍵
0x75 F6 鍵
0x76 F7 鍵
0x77 F8 鍵
0x78 F9 鍵
0x79 F10 鍵
0x7A F11 鍵
0x7B F12 鍵
0x7C F13 鍵
0x7D F14 鍵
0x7E F15 鍵
0x7F F16 鍵
posted @
2008-07-22 00:40 竹子 閱讀(1553) |
評論 (0) |
編輯 收藏
DAO模式在現在的開發中應用非常的廣泛,它可以幫助我們實現持久化邏輯和業務邏輯的分離,同時實現對多種持久化實現的支持。當然現在你可以通過 hibernate來實現對多種持久化的支持,由于新的技術新的方式的出現,DAO也相應的做出了一些調整,比如泛型DAO,在SpringSide中有很還得例子可以參考這方面的實現。
這里聊下傳統意義上的DAO模式(在閻宏的JAVA與模式書中有詳細的介紹),需要注意的幾個方面:
1、不要DAO中出現業務邏輯
DAO只需關注持久化部分,可以通過Facade來控制事務的邊界,從而提高DAO的復用性,在不同的事務策略中應用
2、不要過多的在 DAO層捕捉異常
在很多的開發中,會喜歡使用Checked Exception,拋到servcie層、再到action層,其實在DAO中發生的異常常常是不可恢復的(比如DB的連接問題),所以應該選擇 RuntimeException,我們所需要的只是log的記錄并通知管理員,并通過全局的異常處理畫面告之。
暫時這些,由于ORM的懶加載技術,在DAO中可能會有些調整,為了增加DAO的復用性,這方面的技術也應該剝離出來,不過暫時還沒這么處理過,所以無法總結,希望有這方面經驗的人提供一些
posted @
2008-07-20 09:41 竹子 閱讀(128) |
評論 (0) |
編輯 收藏
本節是單元測試的第三篇。我以為這是重中之重的一章。單元測試的關鍵在于代碼要可測??蓽y才能測。要做好單元測試,就必須在代碼的可測性方面努力,在設計、重構方面用心。本篇主要分享我在如何寫出可測性代碼方面的理解,與大家共勉!
單元測試(提升篇)
------編寫可測試性代碼
一、可測試性設計
1.
接口依賴
這是最重要的一點,因為這可以使得我們很容易的針對一個接口實現Mock對象,模擬/替換實際對象會變的很容易。達到使一個被測對象處于一個孤立環境的測試要求。
這里,ClassA依賴于ClassB的具體實現,TestCase根本無法獨立于ClassB對ClassA進行獨立測試。

因此,我們將ClassA改為依賴于接口B_Inf。這樣,可以很容易的實現一個Mock_B替換ClassB去ClassA進行孤立測試。
2.
依賴注入
一個方法對外部類的依賴,應該是可注入的。即可以通過構造方法、get/set方法的方式在外部將依賴關系注入。事實上,這也為我們在測試用例中替換待測類的依賴對象提供了機會。不應該出現在方法內部新建對象使用的情況。
3.
降低耦合度
待測類應與最少的類耦合,即最小交互原則。特別是要減少與那些離了具體環境就不能運行的類的耦合??梢酝ㄟ^門面模式等對外部調用進行隔離。
5.AOP
面向切面編程。給我們提供的啟示是,將真正需要測的邏輯分離出來。擺脫那些無意義且簡單重復的代碼對測試的干擾。
6. 明確的契約
方法一定要有明確清晰的輸入/輸出。建議在方法的注釋描述中,分三段“描述”“前置條件”“后置條件”。
二、可測試性重構
1.
可惡的靜態方法
在我們的代碼中有大量的調用靜態方法的地方。用起來我們很爽,但這對測試來說卻是災難。因為我們除了通過改變代碼創建stub來改變這些方法的行為外,我們沒有任何途徑。更要命的是,寫這些stub的代價非常的巨大,常常令人望而卻步。
解決方案:
將方法內部的這些調用提取成protected方法。在外部創建待測類的子類,重寫該protected方法。
最佳實踐:

這些靜態方法由單態類提供,單態類由工廠方法獲取,具體類使用這些單態類的接口。
我們在方法中通過接口使用對外部模塊的調用。一方面,隔離了外部模塊改變對我們產生的沖擊,另一方面,也使我們使用Mock替換實際的外部組件,創建孤立測試環境成為可能。
2.
待測類方法中,new出另一個對象并使用其方法
兩種方案:1)將new 出對象的過程封裝成protected方法。同重構1。2)將該對象提取成類屬性,即由使用關系變成關聯關系。
3.
分離不可測/不必測代碼
在不影響的情況下,將不可測部分分離到一些不需要測的簡單方法中去?;蛘邔⒖蓽y的部分提取到一個私有方法中去。然后針對這個私有方法進行測試。
通常這種做法使用范圍有限,但有些時候還是值的一試。
4.
單一職責
職責太多,肯定不好測。針對于這一點“不好測的方法必然不好用”。當方法過大,承擔責任過多時,拆分是應該的。
5.
為類提供一個空構造方法。
我們的各個測試方法都依賴于對象處于一個特定的狀態,滿足一定的前置條件。而要將對象置為我們希望的狀態,我們必須首先擁有一個對象。然而,很多時候,在一個隔離的單元測試環境下,構造函數由于各種原因不能正常初始化。
此時,可以為類提供一個的空的構造方法。在外部構造一個“裸”對象,而后根據前置條件將各個屬性設置成需要的Mock對象。
-----------------------------------------------------------------------------------------------------------------------------------------------------------
事實上,要想寫出具有可測性的代碼,最佳的辦法就是測試驅動開發。先寫測試代碼把功能體現出來,再寫功能代碼讓測試通過。這樣寫出的代碼顯而易見會更具有測試性。
原地址:http://www.aygfsteel.com/wukaichun600/archive/2008/07/18/214125.html
posted @
2008-07-20 09:32 竹子 閱讀(144) |
評論 (0) |
編輯 收藏
本節是單元測試系列的第二篇。重點講解如何使用Mock/Stub和依賴注入技術進行單元測試。關于工具JUnit等則不做累贅介紹。 希望通過本章能夠幫助大家開始單元測試的有益實踐,與大家共勉!
單元測試(技能篇)
一、Stub技術
這是最為古老的一種測試技能。通過類層次上的替換實現了對待測環境的模擬。
實現的時候有兩種途徑:
1、重寫實際類,在測試時,先于實際類加載,即覆蓋。如:我們在unittest/stub文件夾下針對于每一個重寫類都有相同的包結構和類名:

在類路徑中優先加載:

2、在實際代碼中添加判斷。比如,如果當前是測試環境if(isUT)執行XX操作,截斷真正需要做的事。
publicvoid sendCommand(int cmdCode)
{
if(isUT())
{
//...
}
else
{
//...
}
}
Stub技術的問題就在于我們在重寫這些類的時候,不僅僅要關注接口,還要關注其內部邏輯。如果你只是簡單的返回一個固定的響應,會很簡單。但是對于每一次運行需要根據不同的輸入返回不同的輸出時方法內部的處理就會復雜的多。
由于實現的難度,所以,使用時就要注意:有高價值、重用度高、數量少。這就是說,重寫一個類,就可以有一大批類可以用。
二、Mock技術
Mock是目前單元測試中最常用的。用來在對象層次上實現細類度替換十分方便。
當我們在測試中,需要其它類/接口的一個方法時,我們可以通過繼承/實現其一個子類對象來替換實際對象。在Mock子類中將需要的方法直接返回需要的結果就行了。
privateclass Mock_QueryCtrl extends QueryCtrl
{
public List queryNEList()
{
List neList = new ArrayList();
//直接填充并返回你需要的數據...
return neList;
}
}
同樣,我們也可以通過測試待測類的子類來測試待測類。這對于被測方法使用了自身類的方法時很適用。
三、依賴注入
單元測試的一個關鍵就是替換。類層次上的替換,通過在類路徑中提前加載就可以實現。而在對象層次上,java的反射機制提供了很好的幫助。
1).獲取/注入私有屬性
2).執行私有方法
附:注入私有屬性的實現:
publicvoid setFieldObject(Object instance, String fieldName, Object
value)
throws IllegalArgumentException, IllegalAccessException,
NoSuchFieldException {
Field field = null;
Class c = instance.getClass();
do {
try
{
field = c.getDeclaredField(fieldName);
} catch (SecurityException e)
{
e.printStackTrace();
} catch (NoSuchFieldException e)
{
c = c.getSuperclass();
}
}
while (c.getName() != "java.lang.Object" && field
== null);
if (field != null)
{
field.setAccessible(true);
field.set(instance, value);
}
else
{
thrownew NoSuchFieldException(fieldName);
}
}
注:這是一個簡單實現,實際中需要優化。
四、實例:
下例演示了如何測試類NEListTable的ShowNETable()方法。其中注意的是,方法中調用了類QueryCtrl的queryNEList()方法。
待測類:
publicclass NEListTable
{
QueryCtrl ctrl = null;
publicvoid ShowNETable()
{
List neList =
ctrl.queryNEList();
for(int i =
0;i<neList.size();i++)
{
//將neList轉換為表格行
}
//顯示表格...
}
}
publicclass QueryCtrl {
public List queryNEList()
{
returnnull;
}
}
測試類:
public class TestNEListTable extends TestCase
{
private
NEListTable table = null;
private
TestHelper helper = null;
public
void testShowNETable()
{
Mock_QueryCtrl ctrl = new
Mock_QueryCtrl();
helper.setObjectField(table,"ctrl",ctrl);//將Mock對象注入table
table.ShowNETable();
assertTrue(table.getRowCount()>0);
}
private
class Mock_QueryCtrl extends
QueryCtrl
{
public
List queryNEList()
{
List neList = new
ArrayList();
//返回你需要的數據...
return neList;
}
}
}
原地址:http://www.aygfsteel.com/wukaichun600/archive/2008/07/10/213790.html
posted @
2008-07-20 09:31 竹子 閱讀(162) |
評論 (0) |
編輯 收藏