提交带有选择字段的表单时出错

时间:2014-09-05 00:57:18

标签: spring hibernate spring-mvc orm

在我目前的春季项目中,我的表单实现了这样的结构:

<form class="form" id="Pagina" role="form" method="POST" action="/loja/Pagina/cadastra" enctype="multipart/form-data">
...
</form>

并且它在服务器中由此方法处理:

控制器

@RequestMapping(value="cadastra", method=RequestMethod.POST)
@ResponseBody
@ResponseStatus(HttpStatus.CREATED)
public E cadastra(@ModelAttribute("object") E object, BindingResult result, @RequestParam(value="file", required=false) MultipartFile file, @RequestParam(value="icone", required=false) MultipartFile icone, @RequestParam(value="screenshot", required=false) MultipartFile screenshot[]) throws Exception {
    E ret = serv.cadastra(object, file, icone, screenshot);
    if (ret != null)
        return ret;
    else
        throw new Exception();
}

服务

@PreAuthorize("hasPermission(#user, 'cadastra_'+#this.this.name)")
@Transactional
public E cadastra(E e, MultipartFile file, MultipartFile icone, MultipartFile[] screenshot) {
    return dao.persist(e);
}

我的问题是表单有这样的字段:

<label>pagina</label>
<select name="pagina.id" class="form-control select" data-lista="/loja/Pagina/listagem.json">
...
</select>

<label>produto</label>
<select name="produto.id" class="form-control select" data-lista="/loja/Produto/listagem.json">
...
</select>

在entiy类中映射这样的属性:

@OneToOne(fetch = FetchType.EAGER)
@JoinColumn(name="pagina_mae", nullable = true)
@Order(value=5)
@Select(name="pagina", ordem = 5)
@Sidebar
private Pagina pagina;

@OneToOne(fetch = FetchType.EAGER)
@JoinColumn(name="produto_mae", nullable = true)
@Order(value=6)
@Select(name="produto", ordem = 6)
@Sidebar
private Produto produto;

里面的选项如下所示:

<option value="">.</option>
<option value="...">...</option>

如果我提交选择了空白选项的表单,我会收到此错误:

object references an unsaved transient instance - save the transient instance before flushing: com.spring.loja.model.pagina.persistence.model.Pagina; nested exception is org.hibernate.TransientObjectException: object references an unsaved transient instance - save the transient instance before flushing: com.spring.loja.model.pagina.persistence.model.Pagina

但是,例如,如果在数据库中手动插入记录(在我的情况下,使用pgAdmin3),并在选择中选择此项,则表单提交没有错误。

任何人都可以告诉我如何解决这个问题,允许我提交包含或不包含<select>中所选数据的表单。

更新

Pagina的代码:

@Entity
@Table(name="pagina")
@MainForm(grupo = 2, icone = "file")
public class Pagina extends ModelEntity {

    @Id
    @Column(name = "id")
    @GeneratedValue(strategy=GenerationType.IDENTITY)
    private Integer id;

    @Column(name = "nome", unique = true)
    @Order(value=1)
    @Input(type="hidden", name="nome", ordem = 1)
    private String nome;

    @Column(name = "titulo", nullable = false)
    @Order(value=2)
    @Input(name="titulo", ordem = 2)
    private String titulo;

    @Column(name = "descricao", length=65535)
    @Order(value=4)
    @Textarea(name="descricao", ordem = 4)
    private String descricao;

    @OneToOne(fetch = FetchType.EAGER)
    @JoinColumn(name="pagina_mae", nullable = true)
    @Order(value=5)
    @Select(name="pagina", ordem = 5)
    @Sidebar
    private Pagina pagina;

    @OneToOne(fetch = FetchType.EAGER)
    @JoinColumn(name="produto_mae", nullable = true)
    @Order(value=6)
    @Select(name="produto", ordem = 6)
    @Sidebar
    private Produto produto;
}

更新2

PaginaEditor.java

@Component
public class PaginaEditor extends PropertyEditorSupport {

    @Inject
    private PaginaService paginaService;

    @Override
    public void setAsText(String text) {
        if (!text.isEmpty()) {
            Pagina pagina = paginaService.getObject(text);
            setValue(pagina);
        }
    }

}

添加到我的控制器的方法:

@InitBinder
public void initBinder(WebDataBinder binder) {
    binder.registerCustomEditor(Pagina.class, new PaginaEditor());
}

1 个答案:

答案 0 :(得分:5)

在Spring MVC中选择是棘手的。

我认为你的问题是,当你的主要实体要保持数据层时,关系就不存在了。

尝试调试并检查上面的确认是否属实。

有两种方法可以对此进行排序。

让我们在联系人系统中假设公司/联系人关系。 公司有很多联系人,联系人有一家公司。

公司摘要。

// package declaration imports and all
@Entity
@Table(name = "company")
public class Company {

    private String name;

    @OneToMany(mappedBy = "company")
    private List<Contact> contacts = new ArrayList<Contact>();

    // getter and setters and any extra stuff you fancy putting here

}

联系代码段

// package declaration imports and all
@Entity
@Table(name = "contact")
public class Contact {

    private String name;

    @ManyToOne
    private Company company;

    // getter and setters and any extra stuff you fancy putting here

}

带有select的jsp片段。 我们假设有一个&#34;联系人&#34;对象和模型中的客户列表。

<form:form modelAttribute="contact">
    <form:input path="name" />
    <form:select path="customer" items="${customers}" itemValue="id" itemLabel="name" />
</form:form>

使用此代码,您可以像这样使用PropertyEditor。

@Component
public class CustomerEditor extends PropertyEditorSupport {

    @Inject
    private CustomerService customerService;

    @Override
    public void setAsText(String text) {
        if (StringUtils.isNotBlank(text)) {
            Customer customer = this.customerService.findById(Integer
                    .valueOf(text));
            this.setValue(customer);
        }
    }

}

并在你的Spring Context中注册。

@InitBinder
public void initBinder(WebDataBinder binder) {
    binder.registerCustomEditor(Customer.class, this.customerEditor);
}

这里发生的是,只要Spring找到一个Customer类型的Object,它就会使用PropertyEditor将(在这种情况下)Id转换为对象,并按类型将联系人(在本例中)转换为数据层(Hibernate),正确的客户实体将在那里等待拉里的快乐。

这是一种自动化的方式。

另一种方法是创建表单/ DTO或您想要调用的任何内容,并添加包含customerId字段的字段(在本例中)并在保存实体之前转换自己。

我希望我理解你的问题是正确的,因为我花了几分钟来写这个......:)