我正在使用 Seam 2.2.2、JSF Mojarra 1.2_13-b01-FCS、JBoss 5.1.0 服务器。
我有一个 Oracle 表,其中三列指定为 NUMBER 类型。生成的休眠映射已将实体类中的这些列映射为 java.math.BigDecimal。这些是货币值,BigDecimal 似乎是在 Java 中存储和计算这些值的首选方式。
我正在尝试使用数字转换器将屏幕上的值用作 h:inputText 和 h:outputText 中的货币。工作正常,直到我尝试更新其中一个值,当java抛出异常时:
EquipItemEdit.xhtml value="#{equipItemHome.instance.cost}": java.lang.IllegalArgumentException: 参数类型不匹配
我的实体映射如下:
@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>
...
有什么建议,或者有人可以指出我处理这种数据类型的正确方法吗?
现在为了让事情正常工作,我编写了自己的转换器。还没有做太多的测试,希望有人能告诉我如何让 JSF 内置转换器为此工作。
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());
}
}