Java JSF货币在数据库上映射为数字
我使用的是Seam 2.2.2、JSF Mojarra 1.2_13-b01-FCS、JBoss 5.1.0服务器 我有一个Oracle表,其中有三列指定为type NUMBER。生成的hibernate映射已将实体类中的这些列映射为java.math.BigDecimal。这些是货币值,BigDecimal似乎是在Java中存储和计算这些值的首选方法 我正在尝试使用数字转换器将屏幕上的值用作h:inputText和h:outputText中的货币。在我尝试更新其中一个值之前工作正常,当java抛出异常时: EquipItemEdit.xhtml value=“#{equipItemHome.instance.cost}”:java.lang.IllegalArgumentException:参数类型不匹配 我的实体映射如下:Java JSF货币在数据库上映射为数字,java,jsf,seam,Java,Jsf,Seam,我使用的是Seam 2.2.2、JSF Mojarra 1.2_13-b01-FCS、JBoss 5.1.0服务器 我有一个Oracle表,其中有三列指定为type NUMBER。生成的hibernate映射已将实体类中的这些列映射为java.math.BigDecimal。这些是货币值,BigDecimal似乎是在Java中存储和计算这些值的首选方法 我正在尝试使用数字转换器将屏幕上的值用作h:inputText和h:outputText中的货币。在我尝试更新其中一个值之前工作正常,当java
@Entity
@Table(name = "EQUIP_ITEM")
public class EquipItem implements java.io.Serializable {
...
private BigDecimal cost;
...
@Column(name = "COST", precision = 22, scale = 0)
public BigDecimal getCost() {
return this.cost;
}
public void setCost(BigDecimal cost) {
this.cost = cost;
}
...
}
数据库中的表定义
CREATE TABLE EQUIP_ITEM
(
EQUIP_ID NUMBER,
COUNTER NUMBER,
ITEM VARCHAR2(6 BYTE),
ACQADC VARCHAR2(1 BYTE),
COG VARCHAR2(2 BYTE),
COST NUMBER,
NOMENC VARCHAR2(40 BYTE),
QUA_AUTH NUMBER,
...
)
以及字段在JSP上的映射方式:
...
<h:inputText id="cost"
styleClass="value"
value="#{equipItemHome.instance.cost}"
size="15">
<f:convertNumber type="currency" currencySymbol="$" locale="en_US"/>
</h:inputText>
...
尽管名称不同,
将不会转换为BigDecimal
。。。如果您检查Javadoc中的javax.faces.convert.NumberConverter
,它告诉您它将生成Long
(对于整数)或Double
。这是因为,要使用NumberFormat解析BigDecimal,您需要设置一个标志setParseBigDecimal(true)
,否则NumberFormat对象只会产生双精度
JSF确实为BigDecimal提供了一个单独的转换器(恰当地命名为javax.faces.BigDecimal
),而不是
使用:
而且转换也会起作用。但是,此转换器不如
灵活,您不能设置模式或使用不同的地区。因此,大多数人只编写自己的转换器实现
import java.io.Serializable;
import java.math.BigDecimal;
import java.text.NumberFormat;
import java.text.ParseException;
import java.util.Locale;
import javax.faces.component.UIComponent;
import javax.faces.context.FacesContext;
import org.jboss.seam.annotations.Name;
import org.jboss.seam.annotations.intercept.BypassInterceptors;
@Name("usDollarsConverter")
@BypassInterceptors
@org.jboss.seam.annotations.faces.Converter
public class UsDollarsConverter implements javax.faces.convert.Converter, Serializable {
public Object getAsObject(FacesContext context, UIComponent component, String string) {
if(string == null) {
return null;
}
try {
NumberFormat nf = NumberFormat.getCurrencyInstance(Locale.US);
nf.setMinimumFractionDigits(2);
nf.setMaximumFractionDigits(2);
Object value = nf.parse(string);
if(value instanceof Double) {
value = new BigDecimal((Double)value);
} else if(value instanceof Long) {
value = new BigDecimal((Long) value);
}
return value;
} catch (ParseException e) {
return null;
}
}
public String getAsString(FacesContext context, UIComponent component, Object value) {
if(value == null) {
return null;
}
if(value instanceof Double) {
value = new BigDecimal((Double)value);
} else if(value instanceof Long) {
value = new BigDecimal((Long) value);
}
if(value instanceof BigDecimal) {
NumberFormat nf = NumberFormat.getCurrencyInstance(Locale.US);
nf.setMinimumFractionDigits(2);
nf.setMaximumFractionDigits(2);
return nf.format(((BigDecimal)value).doubleValue());
}
throw new java.lang.IllegalArgumentException("UsDollarsConverter requires a type java.math.BigDecimal, will not work with " + value.getClass().getCanonicalName());
}
}