有时我们需要控制输出的数字的格式,如何使用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:cherami@163.net
原文: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);
}
public static final String[] DECIMAL_PATTERN = {"0", "#,##0"};
public static final String[] DECIMAL_DECIMAL_PATTERN = {"0.0", "#,##0.0"};
public static final String[] DECIMAL_PATTERN = {"0", "#,##0"};
public static final String[] DECIMAL_DECIMAL_PATTERN = {"0.0", "#,##0.0"};
public static void main(String[]args){
BigDecimal value = toBigDecimal(DECIMAL_PATTERN , src);
}