我有以下通用的BigDecimal
转换器(深入审查代码是绝对多余的)。
@FacesConverter(value = "bigDecimalConverter")
public class BigDecimalConverter implements Converter {
@Inject
private CurrencyRateBean currencyRateBean; // Maintains a currency selected by a user in his/her session.
private static final int scale = 2; // Taken from an enum.
@Override
public Object getAsObject(FacesContext context, UIComponent component, String value) {
if (!StringUtils.isNotBlank(value)) {
return null;
}
try {
BigDecimal bigDecimal = new BigDecimal(value);
return bigDecimal.scale() > scale ? bigDecimal.setScale(scale, RoundingMode.HALF_UP).stripTrailingZeros() : bigDecimal.stripTrailingZeros();
} catch (NumberFormatException e) {
throw new ConverterException(new FacesMessage(FacesMessage.SEVERITY_ERROR, null, "Message"), e);
}
}
@Override
public String getAsString(FacesContext context, UIComponent component, Object value) {
if (value == null) {
return "";
}
BigDecimal newValue;
if (value instanceof Long) {
newValue = BigDecimal.valueOf((Long) value);
} else if (value instanceof Double) {
newValue = BigDecimal.valueOf((Double) value);
} else if (!(value instanceof BigDecimal)) {
throw new ConverterException("Message");
} else {
newValue = (BigDecimal) value;
}
final String variant = (String) component.getAttributes().get("variant");
if (variant != null) {
if (variant.equalsIgnoreCase("grouping")) {
DecimalFormat formatter = (DecimalFormat) NumberFormat.getNumberInstance();
formatter.setGroupingUsed(true);
formatter.setMinimumFractionDigits(scale);
formatter.setMaximumFractionDigits(scale);
return formatter.format(newValue);
} else if (variant.equalsIgnoreCase("currency")) {
String currency = currencyRateBean.getCurrency();
DecimalFormat formatter = (DecimalFormat) NumberFormat.getCurrencyInstance(new Locale("en", new String(currency.substring(0, 2))));
formatter.setDecimalFormatSymbols(formatter.getDecimalFormatSymbols());
formatter.setCurrency(Currency.getInstance(currency));
return formatter.format(newValue);
}
}
DecimalFormat formatter = (DecimalFormat) NumberFormat.getNumberInstance();
formatter.setGroupingUsed(false); // Not necessary.
formatter.setMinimumFractionDigits(scale);
formatter.setMaximumFractionDigits(scale);
return formatter.format(newValue);
}
}
可以这样使用:
<h:outputText value="#{bean.value}">
<f:converter converterId="bigDecimalConverter"/>
<f:attribute name="variant" value="grouping"/>
</h:outputText>
根据<f:attribute>
中variant
的值,它将值转换为等值货币($123)或使用组(11,111)的值。其他标准也可以在需要时定义,即不需要其他类型格式的单独转换器(如果有的话)。转换器的默认任务是将值四舍五入到两个小数点。
为了避免提及,
<f:converter converterId="bigDecimalConverter"/>
或converter="#{bigDecimalConverter}"
无处不在,转换器类需要装饰,
@FacesConverter(forClass = BigDecimal.class)
这样做时,JSF会事先获取自己的javax.faces.convert.BigDecimalConverter
。我试过扩展这个类,但是没有任何效果。
是否有可能覆盖默认的JSF javax.faces.convert.BigDecimalConverter
,以便我们自己的转换器可以通过指定forClass = BigDecimal.class
来使用,这样就不需要在任何地方乱改<f:converter converterId="bigDecimalConverter"/>
或converter="#{bigDecimalConverter}"
了?
一般来说,重写默认转换器(以及验证器、组件和呈现器)不能在同一标识符上使用注释。它真的必须显式地注册在webapp自己的faces-config.xml
。
如果你想覆盖converter="javax.faces.BigDecimal"
,那么这样做:
<converter>
<converter-id>javax.faces.BigDecimal</converter-id>
<converter-class>com.example.YourBigDecimalConverter</converter-class>
</converter>
如果您想覆盖java.math.BigDecimal
类型的隐式转换,那么这样做:
<converter>
<converter-for-class>java.math.BigDecimal</converter-for-class>
<converter-class>com.example.YourBigDecimalConverter</converter-class>
</converter>
你需要后者
替换默认转换器应遵循以下步骤:
-
扩展您计划替换的默认转换器,以避免类强制转换异常。在你的例子中,这是
javax.faces.convert.BigDecimalConverter
-
在faces-config.xml中注册您的转换器,指定与默认值对应的
<converter-id>
。这允许您覆盖默认值。你将得到的效果是:<converter> <converter-class> com.you.YourBigDecimalConverter </converter-class> <converter-id> javax.faces.BigDecimal </converter-id> </converter>