Получение java .lang.StackOverflowError при использовании Spring + Thymeleaf - PullRequest
0 голосов
/ 04 марта 2020

Я разрабатываю систему фактурирования, используя Java Spring Boot + Thymeleaf. Я получаю эту ошибку:

>Whitelabel Error Page
>This application has no explicit mapping for /error, so you are seeing this as a fallback.

>Wed Mar 04 10:58:26 ART 2020
>There was an unexpected error (type=Internal Server Error, status=500).
>No message available
>java.lang.StackOverflowError

Я использую два контроллера для обработки заполнения счета, который разделен на 3 таблицы: элементы Encabezado (заголовок) и класс P ie (фут) Encabezado:

@EqualsAndHashCode
@Entity
@Table(name="facturas_encabezado")
public class Encabezado implements Serializable {

    private static final long serialVersionUID = -3479505724865821556L;
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)

    private Long id;
    @NotNull
    private Timestamp fecha;
    @NotNull
    @Size(min = 1,max = 11)
    private String numero;
    @Nullable
    private String letra;
    @NotNull
    @OneToOne
    private Cliente cliente;
    @OneToMany(mappedBy = "encabezado")
    private List<Item> items;
    @NotNull
    private Boolean anulado=false;

Элемент класса

@Entity
@Table(name="facturas_items")
public class Item implements Serializable {

    private static final long serialVersionUID = -3560761383569266746L;
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)

    private Long id;
    @OneToOne
    @JoinColumn(name = "productos_id")
    private Producto producto;
    @ManyToOne
    @JoinColumn(name = "facturas_encabezado_id")
    private Encabezado encabezado;
    private BigDecimal cantidad;
    private BigDecimal precioUnitario;  
    private BigDecimal subTotal;

HeaderController

 @Controller
@RequestMapping("/factura")
public class EncabezadoControl {
    @Autowired
    private ClienteRepositorio clienterepository;
    @Autowired
    private ProductoRepositorio productorepository;
    @Autowired
    private EncabezadoRepositorio encabezadorepository;

    @GetMapping("/creacion_factura")
    public String creacionFactura(Encabezado encabezado,Errors errors,Model model ) {
        List<Cliente> clientes;
        clientes=clienterepository.findByVisibilidad();
        model.addAttribute("titulo", "Listado de Clientes");
        model.addAttribute("data", clientes);
        model.addAttribute("encabezado",new Encabezado());

        return "facturas/creacion_factura";
    }

    @PostMapping(value = "/creacion_factura")
    public String guardarFactura(Encabezado encabezado,Errors errors,Model model) {
        Timestamp timestamp = new Timestamp(System.currentTimeMillis()); //Instanciamos fecha actual para el encabezado de factura.
        encabezado.setFecha(timestamp);
        model.addAttribute("facturaInfo",encabezado);
        System.out.println(encabezado);

        if(encabezado.getId()==null){
            encabezadorepository.save(encabezado);
        }
        Long id=encabezado.getId();
        return "redirect:/factura/establecer_productos/"+id;
    }

}

ItemController

  @Controller
@RequestMapping("/factura")
public class ItemControl {
    @Autowired
    private ItemRepositorio itemrepository;
    @Autowired
    private ProductoRepositorio productorepository;
    @Autowired
    private EncabezadoRepositorio encabezadorepository;

    @GetMapping(value = "/establecer_productos/{id}")
    public String indicarProductos(Model model, @PathVariable Long id){

        List<Producto> listaproductos=null;
        listaproductos=productorepository.findByEstado();

        model.addAttribute("titulo", "Selección de Productos");
        model.addAttribute("productos",listaproductos);

        model.addAttribute("item", new Item());


        return "facturas/seleccion_productos";
    }


    @PostMapping(value = "/establecer_productos/{ided}")
    public String procesarProductos(@PathVariable Long ided, Item item,Model model){

        if(item.getId()==null) {
            BigDecimal cantidad = item.getCantidad();
            BigDecimal precio = item.getPrecioUnitario();
            item.setSubTotal(cantidad.multiply(precio));
            item.setEncabezado(encabezadorepository.getOne(ided));
            itemrepository.save(item);
        }
        System.out.println("ITEM\n\n\n"+item+"\n\n\n");
        return "redirect:/factura/establecer_productos/"+ided;
    }

}

Представление первого шага: создание заголовка

<div class="container">

        <form method="post" th:action="@{/factura/creacion_factura}" th:object="${encabezado}">
            <br/>
            <h2>Introduzca los datos de la Factura</h2>
            <hr/>
            <div class="row">
                <div class="col-md-3">
                    <h3>Cliente</h3>
                </div>
                <div class="col-md-6">
                    <select name="cliente" class="form-control form-control-lg" th:field="*{cliente}">
                        <option th:each="cliente : ${data}" th:value="${cliente.id}" th:text="${cliente.nombre}"></option>

                    </select>
                </div>
                <div class="col-md-3"></div>
            </div>

            <div class="row" th:class="${#fields.hasErrors('numero')} ? 'form-group row has-danger' : (*{numero == null} ? 'form-group row' : 'form-group row has-success')">
                <div class="col-md-3">
                    <h3>Numero Factura</h3>
                </div>
                <div class="col-md-6">
                    <div class="input-group">
                        <input th:field="*{numero}" type="text" th:class="${#fields.hasErrors('numero')} ? 'form-control form-control-danger' : (*{numero == null} ? 'form-control' : 'form-control form-control-success')" placeholder="Ingrese el numero de factura" required="required"/>
                    </div>
                    <div th:if="${#fields.hasErrors('numero')}" class="form-control-feedback">El número debe tener entre 1 y 11 caracteres</div>
                </div>
                <div class="col-md-3"></div>
            </div>

            <div class="row" th:class="${#fields.hasErrors('letra')} ? 'form-group row has-danger' : (*{letra == null} ? 'form-group row' : 'form-group row has-success')">
                <div class="col-md-3">
                    <h3>Categoria Factura</h3>
                </div>
                <div class="col-md-6">
                    <select class="form-control form-control-lg" th:field="*{letra}" >
                        <option th:value="'A'"th:text="A"></option>
                        <option th:value="'B'"th:text="B"></option>
                        <option th:value="'C'"th:text="C"></option>
                    </select>
                </div>
                <div class="col-md-3"></div>
            </div>


            <div class="form-group row">
                <div class="col-sm-2"></div>
                <div class="col-sm-6">
                    <button type="submit" class="btn btn-primary btn-md btn-block">Siguiente Paso</button>
                </div>
            </div>

        </form>
    </div>

Второй шаг: Добавить элементы зрения

<div class="container">

    <form method="post" th:action="@{/factura/establecer_productos/{id}(id=${id})}" th:object="${item}">
        <br/>
        <h2>Seleccione los Productos de la Factura</h2>
        <hr/>

        <div class="row">
            <div class="col-md-3">
                <h3>Productos</h3>
            </div>
            <div class="col-md-6">
                <select class="form-control form-control-lg" name="producto" th:field="*{producto}">
                    <option th:each="producto : ${productos}"  th:value="${producto.id}" th:text="${producto.nombre}"></option>
                z</select>
            </div>
            <div class="col-md-3"></div>
        </div>

        <div class="row">
                <div class="col-md-3">
                    <h4>Cantidad</h4>
                </div>
                <div class="col-md-6">
                    <div class="input-group">
                        <input class="form-control text-center" th:field="*{cantidad}" type="text" th:class="${#fields.hasErrors('cantidad')} ? 'form-control form-control-danger' : (*{cantidad == null} ? 'form-control' : 'form-control form-control-success')" placeholder="introduzca cantidad a comprar" required="required"/>
                    </div>
                    <div th:if="${#fields.hasErrors('cantidad')}" class="form-control-feedback">La cantidad es obligatoria</div>
                </div>
                <div class="col-md-3"></div>
        </div>

        <div class="row">

            <div class="col-md-3">
                <h4>Precio Unitario</h4>
            </div>
            <div class="col-md-6">
                <div class="input-group">
                    <input class="form-control text-center" th:field="*{precioUnitario}" type="text" th:class="${#fields.hasErrors('precioUnitario')} ? 'form-control form-control-danger' : (*{precioUnitario == null} ? 'form-control' : 'form-control form-control-success')" placeholder="introduzca precio unitario" required="required"/>
                </div>
                <div th:if="${#fields.hasErrors('precioUnitario')}" class="form-control-feedback">El precio es obligatoria</div>
            </div>
            <div class="col-md-3"></div>

        </div>

        <div class="form-group row">
            <div class="col-sm-2"></div>
            <div class="col-sm-6">
                <button type="submit" class="btn btn-primary btn-md btn block">Siguiente</button>
            </div>
        </div>

    </form>
</div>

Компиляция выполнения идет хорошо. Заголовок счета правильно сохраняется в БД, но проблема возникает, когда мы переходим ко второму шагу: элемент сохраняется после полного выполнения. Любые решения?

1 Ответ

1 голос
/ 04 марта 2020

Не используйте @Data, @ToString, @EqualsAndHashCode с JPA, они могут привести к StackOverflowErrors и другим проблемам.

Пожалуйста, прочитайте: https://mdeinum.github.io/2019-02-13-Lombok-Data-Ojects-Arent-Entities/

А вот как эти методы следует реализовать: https://vladmihalcea.com/the-best-way-to-implement-equals-hashcode-and-tostring-with-jpa-and-hibernate/

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...