涉及到客戶端的系統中經常需要用到比較版本號的功能,但是比較版本號又不能完全按照字符串比較的方式去用compareTo之類的方法;
這就需要我們總結版本號的通用規則,設計一個比較算法并封裝成通用方法來使用:
通常版本號如:1.3.20.8,6.82.20160101,8.5a/8.5c等;
通用規則就是,先將版本字符串按照點號分割,然后主版本與主版本比較、此版本與此版本比較,如此按序一級一級往后比較,直到有分出大小;
值得注意的是,很多比較版本號的方法都先將字符串轉換成int或者double類型,這樣做未必通用,因為可能含有字母,如8.5c這樣的版本號;
通用的方式依然是將分割后的字符串當做字符串來比較,不過,比較字符串之前,先比較位數;
比較版本號的方法示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
/** * 比較版本號的大小,前者大則返回一個正數,后者大返回一個負數,相等則返回0 * @param version1 * @param version2 * @return */ public static int compareVersion(String version1, String version2) throws Exception { if (version1 == null || version2 == null ) { throw new Exception( "compareVersion error:illegal params." ); } String[] versionArray1 = version1.split( "\\." ); //注意此處為正則匹配,不能用"."; String[] versionArray2 = version2.split( "\\." ); int idx = 0 ; int minLength = Math.min(versionArray1.length, versionArray2.length); //取最小長度值 int diff = 0 ; while (idx < minLength && (diff = versionArray1[idx].length() - versionArray2[idx].length()) == 0 //先比較長度 && (diff = versionArray1[idx].compareTo(versionArray2[idx])) == 0 ) { //再比較字符 ++idx; } //如果已經分出大小,則直接返回,如果未分出大小,則再比較位數,有子版本的為大; diff = (diff != 0 ) ? diff : versionArray1.length - versionArray2.length; return diff; } |
注意:其中 split 方法入參為正則匹配表達式,不能用"."("."在正則表達式里匹配任何值),需要用"\\.",才算是按點號分割;
這樣,先分割成子串數組,再挨個比較子版本號,比較子版本號時,先比較位數,位數大的就大,位數一樣時再按字符串比較方式比較;
如果全部比較完(其中一個版本號比較完)之后,再看一下哪個版本號有更更多的子版本號,也就是分割后的數組長度,有子版本號的為大;
這樣就比較完善地考慮了各種情況,并比較出版本號大小;包括有字母后綴的也可以使用;
如 "9.9", "10.8.8.6" ,如果直接按字符串比較,則會前者大,后者小,而明顯是錯誤的;分割后比較第一個主版本9與10,從位數上,就已經得出結果后者大;
再如 "9.9b", "9.8a" 等也適用,如果用轉換成int或者double的方法就不適用.