8

我是 JSF 的新手,我一直在尝试从使用 ah:selectOneMenu 获取产品类别的表单中存储数据。h:selectOneMenu 是从数据库中填充的,但是,当尝试将产品存储在数据库中时,我收到一个错误:“null Converter”的转换错误设置值“52”。我已经在 StackOverflow 和在线教程中查看了类似的问题,但我仍然收到错误消息。

这是xhtml:

<h:selectOneMenu id="category_fk" value="#{productController.product.category_fk}" 
                                 converter="#{categoryConverter}" title="Category_fk" >
                    <!-- DONE: update below reference to list of available items-->
                    <f:selectItems value="#{productController.categoryList}" var="prodCat"
                                   itemValue="#{prodCat}" itemLabel="#{prodCat.name}"/>
                </h:selectOneMenu>

这是产品控制器:

@Named
@RequestScoped
public class ProductController {

    @EJB
    private ProductEJB productEjb;
    @EJB
    private CategoryEJB categoryEjb;

    private Product product = new Product();
    private List<Product> productList = new ArrayList<Product>();

    private Category category;
    private List<Category> categoryList = new ArrayList<Category>();

    public String doCreateProduct()
    {
        product = productEjb.createProduct(product);
        productList = productEjb.findAllProducts();
        return "listProduct";
    }

    @PostConstruct
    public void init()
    {
        categoryList = categoryEjb.findAllCategory();
        productList = productEjb.findAllProducts();
    }        

    // Getters/Setters and other methods omitted for simplicity

这是为简单起见而简化的 EJB:

   @Stateless
    public class ProductEJB{

        @PersistenceContext(unitName = "luavipuPU")
        private EntityManager em;

        public List<Product> findAllProducts()
        {
            TypedQuery<Product> query = em.createNamedQuery("findAllProducts", Product.class);
            return query.getResultList();
        }

        public Product createProduct(Product product)
        {
            em.persist(product);
            return product;
        }    

    }

这是为简单起见简化的产品实体:

@Entity
@NamedQueries({
    @NamedQuery(name="findAllProducts", query = "SELECT p from Product p")
})
public class Product implements Serializable
{
    private static final long serialVersionUID = 1L;

    @Id @GeneratedValue(strategy= GenerationType.AUTO)
    private int product_id;
    private String name;
    private String description;
    protected byte[] imageFile;
    private Float price;
    @Temporal(TemporalType.TIMESTAMP)
    private Date dateAdded;
    @ManyToOne    
    private Category category_fk;
    @ManyToOne
    private SaleDetails saleDetails_fk;

这是我正在使用的更新转换器:

@ManagedBean
@FacesConverter(value="categoryConverter")
public class CategoryConverter implements Converter{

    @PersistenceContext
    private transient EntityManager em;

    @Override
    public Object getAsObject(FacesContext context, UIComponent component, String value) {
        return em.find(Category.class, new Integer(value));
    }

    @Override
    public String getAsString(FacesContext context, UIComponent component, Object value) {

        Category category;
        category = (Category) value;
        return String.valueOf(category.getCategory_id());

    }

}

代码已从原始问题更新,代码现在完美运行。

4

2 回答 2

11

如果不创建 JSF Converter<h:selectOneMenu/> ,就不能在组件(或任何)上使用自定义类型。JSF 中存在转换器以帮助将本质上自定义或服务器端项目/结构转换为网页友好/人类可读格式,并且还能够将选择从客户端保存回服务器端。因此,错误消息本质上告诉您的是,它无法充分理解从客户端提交的值,无法将其作为对象/类型保存到服务器端<h:selectXXXX/>52Category

因此,您需要做的是创建一个 JSF 的实现,converter它基本上可以帮助 JSF 理解您的Category对象。

以下是您需要做的粗略估计:

  1. 为您的类型实现转换器Category

    // You must annotate the converter as a managed bean, if you want to inject
    // anything into it, like your persistence unit for example.
    @ManagedBean(name = "categoryConverterBean") 
    @FacesConverter(value = "categoryConverter")
    public class CategoryConverter implements Converter {
    
        @PersistenceContext(unitName = "luavipuPU")
        // I include this because you will need to 
        // lookup  your entities based on submitted values
        private transient EntityManager em;  
    
        @Override
        public Object getAsObject(FacesContext ctx, UIComponent component,
                String value) {
          // This will return the actual object representation
          // of your Category using the value (in your case 52) 
          // returned from the client side
          return em.find(Category.class, new BigInteger(value)); 
        }
    
        @Override
        public String getAsString(FacesContext fc, UIComponent uic, Object o) {
            //This will return view-friendly output for the dropdown menu
            return ((Category) o).getId().toString(); 
        }
    }
    
  2. 参考您的新转换器<h:selectOneMenu/>

    <h:selectOneMenu id="category_fk" 
      converter="#{categoryConverterBean}"
      value="#productController.product.category_fk}" 
      title="Category_fk" >
    <!-- DONE: update below reference to list of available items-->
        <f:selectItems value="#{productController.categoryList}" 
          var="prodCat" itemValue="#{prodCat.category_id}" 
          itemLabel="#{prodCat.name}"/>
    </h:selectOneMenu>
    

我们使用该名称categoryConverterBean是因为我们想利用您在转换器中提取的实体管理器技巧。在任何其他情况下,您将使用您在转换器注释上设置的名称。

于 2012-11-02T04:28:50.460 回答
2

Product.category_fk 似乎是Category类型,我怀疑 prodCat.category_id 是Integer或其他东西......

在您的 xhtml 中,您似乎将 selectItems 的值设置为:itemValue="#{prodCat.category_id}"

当 selectOneMenu 需要一个值时value="#{productController.product.category_fk}"

很可能不是同一类型。

于 2012-11-01T18:10:00.957 回答