Java JSF货币在数据库上映射为数字

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

我使用的是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:参数类型不匹配

我的实体映射如下:

@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());
    }


}