JAVA 數字格式化處理方法; 國際化
有時我們需要控制輸出的數字的格式,如何使用java的類庫做到這個呢?例如數字“1234.56”如何以“1234.560”、“1,234.56”格式輸出,在此你可以找到答案例子:例如數字:
1、1234.56以1234.560格式輸出
DecimalFormat df1 = new DecimalFormat("####.000");
df1.setGroupingUsed(false);
System.out.println(df1.format(1234.56));
2、1234.56以123,4.560輸出
DecimalFormat df1 = new DecimalFormat("###.000");
df1.setGroupingUsed(true);//或者不寫
System.out.println(df1.format(1234.56));
3、0.47以百分數輸出
NumberFormat nf = NumberFormat.getPercentInstance();
System.out.println(nf.format(0.47));
輸出:47%
4、以當地格式輸出1234.56
NumberFormat nf1 = NumberFormat.getInstance();
System.out.println(nf1.format(1234.56));
如果你在美國,執行程式後輸出:
1,234.56
1.234,56
--------------如果大家想了解得更仔細些,下面這篇文章可供參考:---------------------
翻譯:Cherami
email:[email protected]
原文:http://developer.java.sun.com/developer/TechTips/2000/tt0411.html
有時我們需要控制輸出的數字的格式,如何使用java的類庫做到這個呢?
也許你不關心格式,但是你需要關心你的程式可以在全世界通用,像下面的這樣一個簡單的語句是依賴地區的:
System.out.println(1234.56);
在美國,"." 是小數點,但在其它地方就不一定了。如何處理這個呢?
java.text 包中的一些包可以處理這類問題。下面的簡單範例使用那些類解決上面提出的問題:
import java.text.NumberFormat;
import java.util.Locale;
public class DecimalFormat1 {
public static void main(String args[]) {
// 得到本地的預設格式
NumberFormat nf1 = NumberFormat.getInstance();
System.out.println(nf1.format(1234.56));
// 得到德國的格式
NumberFormat nf2 =
NumberFormat.getInstance(Locale.GERMAN);
System.out.println(nf2.format(1234.56));
}
}
如果你在美國,執行程式後輸出:
1,234.56
1.234,56
換句話說,在不同的地方使用不同的習慣表示數字。
NumberFormat.getInstance()方法返回NumberFormat的一個例項(實際上是NumberFormat具體的一個子類,例如DecimalFormat), 這適合根據本地設定格式化一個數字。你也可以使用非預設的地區設定,例如德國。然後格式化方法根據特定的地區規則格式化數字。這個程式也可以使用一個簡單的形式:
NumberFormat.getInstance().format(1234.56)
但是儲存一個格式然後重用更加有效。國際化是格式化數字時的一個大問題。
另一個是對格式的有效控制,例如指定小數部分的位數,下面是解決這個問題的一個簡單例子:
import java.text.DecimalFormat;
import java.util.Locale;
public class DecimalFormat2 {
public static void main(String args[]) {
// 得到本地的預設格式
DecimalFormat df1 = new DecimalFormat("####.000");
System.out.println(df1.format(1234.56));
// 得到德國的格式
Locale.setDefault(Locale.GERMAN);
DecimalFormat df2 = new DecimalFormat("####.000");
System.out.println(df2.format(1234.56));
}
}
在這個例子中設定了數字的格式,使用像"####.000"的符號。這個模式意味著在小數點前有四個數字,如果不夠就空著,小數點後有三位數字,不足用0補齊。程式的輸出:
1234.560
1234,560
相似的,也可以控制指數形式的格式,例如:
import java.text.DecimalFormat;
public class DecimalFormat3 {
public static void main(String args[]) {
DecimalFormat df = new DecimalFormat("0.000E0000");
System.out.println(df.format(1234.56));
}
}
輸出:
1.235E0003
對於百分數:
import java.text.NumberFormat;
public class DecimalFormat4 {
public static void main(String args[]) {
NumberFormat nf = NumberFormat.getPercentInstance();
System.out.println(nf.format(0.47));
}
}
輸出:
47%
至此,你已經看到了格式化數字的幾個不同的技術。另一方面,如何讀取並解析包含格式化的數字的字串?解析支援包含在NumberFormat中。例如:
import java.util.Locale;
import java.text.NumberFormat;
import java.text.ParseException;
public class DecimalFormat5 {
public static void main(String args[]) {
// 本地格式
NumberFormat nf1 = NumberFormat.getInstance();
Object obj1 = null;
// 基於格式的解析
try {
obj1 = nf1.parse("1234,56");
}
catch (ParseException e1) {
System.err.println(e1);
}
System.out.println(obj1);
// 德國格式
NumberFormat nf2 =
NumberFormat.getInstance(Locale.GERMAN);
Object obj2 = null;
// 基於格式的解析
try {
obj2 = nf2.parse("1234,56");
}
catch (ParseException e2) {
System.err.println(e2);
}
System.out.println(obj2);
}
}
這個例子分兩部分,都是解析一個字串:"1234,56"。第一部分使用本地格式解析,第二部分使用德國格式解析。當程式在美國執行,結果是:
123456
1234.56
換句話說,"1234,56"在美國被認為是一個巨大的整數"123456"而在德國被認為是一個小數"1234.56"。
還有格式化討論的最後一個問題。在上面的例子中, DecimalFormat 和 NumberFormat 都被使用了。DecimalFormat 常用於獲得很好的格式控制,而NumberFormat 常用於指定不同於本地的地區。如何結合兩個類呢?
答案圍繞著這樣的事實:DecimalFormat是NumberFormat的一個子類,其例項被指定為特定的地區。因此,你可以使用NumberFormat.getInstance 指定一個地區,然後將結構強制轉換為一個DecimalFormat物件。文件中提到這個技術可以在大多情況下適用,但是你需要用try/catch 塊包圍強制轉換以防轉換不能正常工作 (大概在非常不明顯得情況下使用一個奇異的地區)。下面是一個這樣的例子:
import java.text.DecimalFormat;
import java.text.NumberFormat;
import java.util.Locale;
public class DecimalFormat6 {
public static void main(String args[]) {
DecimalFormat df = null;
// 得到一個NumberFormat 物件並
// 強制轉換為一個 DecimalFormat 物件
try {
df = (DecimalFormat)
NumberFormat.getInstance(Locale.GERMAN);
}
catch (ClassCastException e) {
System.err.println(e);
}
// 設定格式模式
df.applyPattern("####.00000");
// format a number
System.out.println(df.format(1234.56));
}
}
getInstance() 方法獲得格式,然後呼叫applyPattern()方法設定格式模式,輸出:
1234,56000
如果你不關心國際化,可以直接使用DecimalFormat=========================================================
我們通常都很喜歡用SimpleDateFormat來做一些日期和字串之間的轉換,就是所謂的format()和parse()了,具體用法看程式或者是JAVA的API文件,這裡不累述了。但是往往我們忘記了,JAVA的國際化這個讓人歡喜讓人愁的東西。下面出現一個問題:以下程式段
String dateStr = "17/Mar/2003 11:30:51";
SimpleDateFormat frm = new SimpleDateFormat("dd/MMM/yyyy HH:mm:ss");
Date date = frm.parse(dateStr);
SimpleDateFormat frm1 = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
System.out.println("reformat : " +frm1.format(date));
這執行時會丟擲異常java.text.ParseException: Unparseable date: "17/Mar/2003 11:30:51",顯然程式不能parse到“Mar”這個英文月的縮寫。但是按照java上面的文件顯示,以上程式是沒有問題的。一個典型的執行時異常。不用說考慮一下我們的環境。當前程式編寫的環境是中文winxp,jdk1.4.2,IDEA 4.0,呵呵,我們是中國人當然都喜歡用中文環境啦。問題出來了吧?看一下一段代碼:
Date date = new Date();
SimpleDateFormat frm1 = new SimpleDateFormat("yyyy/MMM/dd HH:mm:ss");
System.out.println("now : " +frm1.format(date));
輸出結果是:now : 2004/二月/24 11:57:00
看到了,預設狀態底下,我們用SimpleDateFormat是按照我們當前系統的Locale(請恕我一直都搞不清楚他的中文翻譯)的,也就是中文的Locale,但是我們要進行分析的日期字串是英文的,當然就認不出來了。
ok,知道了之後,我們修改一下加多個locale進去,讓SimpleDateFormat在構造的時候指定Locale(Java是很笨的,我們不說他怎麼知道呢?)告訴他我們現在要處理的是英文的格式串
Locale locale = Locale.US;
String dateStr = "17/Mar/2003 11:30:51";
SimpleDateFormat frm = new SimpleDateFormat("dd/MMM/yyyy HH:mm:ss", locale);
Date date = frm.parse(dateStr);
SimpleDateFormat frm1 = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss", locale);
System.out.println("reformat : " +frm1.format(date));
程式正常了……
問題主要就是出在我們在不同環境底下對一些國際化處理上面的不小心或者容易忽略的地方,僅當作教訓供大家一笑。
問題主要就是出在我們在不同環境底下對一些國際化處理上面的不小心或者容易忽略的地方,僅當作教訓供大家一笑。
----------------------------
public static final BigDecimal toBigDecimal(String[] patterns, String src) throws EsqSystemException {
if (ValueUtils.isBlankOrNull(src)) {
return null;
}
BigDecimal value = null;
int srcLength = src.length();
ParsePosition pos = new ParsePosition(0);
DecimalFormat formatter = new DecimalFormat();
for (int i = 0; i < patterns.length; i++) {
pos.setErrorIndex(0);
pos.setIndex(0);
String pattern = patterns[i];
formatter.applyPattern(pattern);
Number num = formatter.parse(src, pos);
if (num != null && srcLength == pos.getIndex()) {
// 曄姺惉岟
if (num instanceof BigDecimal) {
value = (BigDecimal) num;
} else {
double doubleValue = num.doubleValue();
value = new BigDecimal(doubleValue);
}
return value;
}
}
throw new EsqSystemException("parse error = " + src);<