【问题标题】:Error when a form with a select field is submitted提交带有选择字段的表单时出错
【发布时间】:2014-10-29 21:25:39
【问题描述】:

在我当前的 spring 项目中,我的表单是用这样的结构实现的:

<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),并在选择中选择此项,则表单提交时不会出错。

任何人都可以告诉我如何解决这个问题,以允许我提交带有或不带有来自 &lt;select&gt; 的选定数据的表单。

更新

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

【问题讨论】:

  • 这是一个 ORM 问题。猜测休眠?如果您为正在使用的 ORM 添加标签,您将更有可能找到一些帮助。另请查看此问题/遮阳篷stackoverflow.com/questions/2302802/…
  • 你能告诉我们dao.persist(e)的代码吗?您如何坚持到底,因为我可能认为我知道这里的问题是什么。
  • @ug_ 我看到了链接,但如果我尝试从那里遵循建议(基本上,它使用 cascade 选项),应用程序尝试插入新的 Pagina 或 Produto 并与 Pagina 关联我正在插入。
  • @Aeseir 方法 persist(E transientInstance) 来自 Dao&lt;E&gt; 类的代码是这样的:sessionFactory.getCurrentSession().persist(transientInstance);(仅此行)。
  • 太好了,请问您也可以为您尝试提交的对象提供代码

标签: spring hibernate spring-mvc orm


【解决方案1】:

在 Spring MVC 中选择是棘手的。

我认为您的问题是,当您的主要实体到达要持久化的数据层时,关系不存在。

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

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

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

公司sn-p。

// 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

}

联系sn-p

// 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 sn-p。 我们假设模型中有一个“联系人”对象和一个客户列表。

<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 上下文中注册。

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

这里发生的情况是,每当 Spring 找到 Customer 类型的 Object 时,它将使用 PropertyEditor 将(在这种情况下)Id 转换为对象,并通过联系人(在这种情况下)获取数据的类型层(休眠),正确的客户实体将在那里等待,像拉里一样快乐。

这是自动执行此操作的方法。

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

我希望我能正确理解你的问题,因为我花了好几分钟才写完这篇文章...... :)

【讨论】:

  • 好的,我了解您的回答的基础知识,但我有几个问题:1)StringUtils 组件的库是什么?当我在我的应用程序中添加它时,eclipse 提供了几个选项。 2) 我应该将派生自PropertyEditorSupport 的类放在哪里? 3)我应该把方法initBinder放在哪里?
  • @KleberMota StringUtils 来自 apache commons,即commons.apache.org/proper/commons-lang/apidocs/org/apache/…
  • 你必须在你的控制器中放置 intiBiner
  • @KleberMota 在验证器中放置派生类
  • @HarshalPatil 我只是添加了所有这些,但我仍然遇到同样的错误。
猜你喜欢
  • 1970-01-01
  • 2015-11-04
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-11-07
  • 1970-01-01
  • 2014-11-11
  • 1970-01-01
相关资源
最近更新 更多