簡體   English   中英

例外:生成不完整的 JSON

[英]Exception: Generating incomplete JSON

我有以下方法,它使用給定的 arguments 生成 json 文件,如果文件存在,它會讀取它以獲取數據,然后 append 將新信息傳遞給它。 基本上,我在沒有創建 json 文件的情況下收到此錯誤。

public void cargarVentasDiaMesaJson(Mesas m, double total) {
    FileInputStream fI = null;
    JsonReader jRe = null;
    FileOutputStream f = null;
    JsonWriter jR = null;
    try {
        f = new FileOutputStream("C:\\archivos\\VentasDiaMesa.json");
        jR = Json.createWriter(f);
        JsonObjectBuilder obj = Json.createObjectBuilder();
        JsonArrayBuilder arrayB = Json.createArrayBuilder();
        File file = new File("C:\\archivos\\VentasDiaMesa.json");
        if(file.exists()) {
            fI = new FileInputStream("C:\\archivos\\VentasDiaMesa.json");
            jRe = Json.createReader(fI);
            JsonObject jsonR = jRe.readObject();
            JsonArray array = jsonR.getJsonArray("ventasDiaMesa");
            for(int i=0; i<array.size(); i++) {
                JsonObject vDiaMesa = array.getJsonObject(i);
                String fecha = vDiaMesa.getString("fecha");
                int numero = vDiaMesa.getInt("numero");
                double tot = Double.parseDouble(vDiaMesa.getString("total"));
                JsonObjectBuilder diaMesa = Json.createObjectBuilder();
                diaMesa.add("fecha", fecha);
                diaMesa.add("numero", numero);
                diaMesa.add("total", tot);
                JsonObject objLinea = diaMesa.build();
                arrayB.add(objLinea);
            }
        }
        JsonObjectBuilder diaMesa = Json.createObjectBuilder();
        Calendar fecha = Calendar.getInstance();
        String mes = (fecha.get(Calendar.MONTH)+1)<10?"0"+(fecha.get(Calendar.MONTH)+1):""+(fecha.get(Calendar.MONTH)+1);
        String dia = fecha.get(Calendar.DATE)<10?"0"+fecha.get(Calendar.DATE):""+fecha.get(Calendar.DATE);
        String date = "" + fecha.get(Calendar.YEAR) + mes + dia;
        diaMesa.add("fecha", date);
        String numero = (m.getNumero()<10)?"0"+m.getNumero():""+m.getNumero();
        diaMesa.add("numero", numero);
        String tot = Double.toString(total);
        while(tot.length() != 10) {
            tot = "0" + tot;
        }
        diaMesa.add("total", tot);
        JsonObject objLinea = diaMesa.build();
        arrayB.add(objLinea);
        JsonArray arr = arrayB.build();
        obj.add("ventasDiaMesa", arr);
        JsonObject obbbj = obj.build();
        jR.writeObject(obbbj);
    } catch (IOException e) {
        // TODO: handle exception
        e.printStackTrace();
    } finally {
        jR.close();
    }
}

我收到以下錯誤消息

Exception in thread "AWT-EventQueue-0" javax.json.stream.JsonGenerationException: Generating incomplete JSON
at org.glassfish.json.JsonGeneratorImpl.close(JsonGeneratorImpl.java:509)
at org.glassfish.json.JsonWriterImpl.close(JsonWriterImpl.java:150)
at Modelo.DAO.VentasDiaMesaJSON.cargarVentasDiaMesaJson(VentasDiaMesaJSON.java:77)
at Controlador.ControladorPedidos.totalMesa(ControladorPedidos.java:94)
at Controlador.ControladorPedidos.cerrarDia(ControladorPedidos.java:119)
at Vista.PedidosGUI$1.actionPerformed(PedidosGUI.java:50)
at javax.swing.AbstractButton.fireActionPerformed(Unknown Source)
at javax.swing.AbstractButton$Handler.actionPerformed(Unknown Source)
at javax.swing.DefaultButtonModel.fireActionPerformed(Unknown Source)
at javax.swing.DefaultButtonModel.setPressed(Unknown Source)
at javax.swing.AbstractButton.doClick(Unknown Source)
at javax.swing.plaf.basic.BasicMenuItemUI.doClick(Unknown Source)
at javax.swing.plaf.basic.BasicMenuItemUI$Handler.mouseReleased(Unknown Source)
at java.awt.Component.processMouseEvent(Unknown Source)
at javax.swing.JComponent.processMouseEvent(Unknown Source)
at java.awt.Component.processEvent(Unknown Source)
at java.awt.Container.processEvent(Unknown Source)
at java.awt.Component.dispatchEventImpl(Unknown Source)
at java.awt.Container.dispatchEventImpl(Unknown Source)
at java.awt.Component.dispatchEvent(Unknown Source)
at java.awt.LightweightDispatcher.retargetMouseEvent(Unknown Source)
at java.awt.LightweightDispatcher.processMouseEvent(Unknown Source)
at java.awt.LightweightDispatcher.dispatchEvent(Unknown Source)
at java.awt.Container.dispatchEventImpl(Unknown Source)
at java.awt.Window.dispatchEventImpl(Unknown Source)
at java.awt.Component.dispatchEvent(Unknown Source)
at java.awt.EventQueue.dispatchEventImpl(Unknown Source)
at java.awt.EventQueue.access$500(Unknown Source)
at java.awt.EventQueue$3.run(Unknown Source)
at java.awt.EventQueue$3.run(Unknown Source)
at java.security.AccessController.doPrivileged(Native Method)
at java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(Unknown Source)
at java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(Unknown Source)
at java.awt.EventQueue$4.run(Unknown Source)
at java.awt.EventQueue$4.run(Unknown Source)
at java.security.AccessController.doPrivileged(Native Method)
at java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(Unknown Source)
at java.awt.EventQueue.dispatchEvent(Unknown Source)
at java.awt.EventDispatchThread.pumpOneEventForFilters(Unknown Source)
at java.awt.EventDispatchThread.pumpEventsForFilter(Unknown Source)
at java.awt.EventDispatchThread.pumpEventsForHierarchy(Unknown Source)
at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
at java.awt.EventDispatchThread.run(Unknown Source)

我正在創建 3 個 object 構建器,然后我繼續構建它們,最后在關閉文件之前編寫主要的 object。 我在這里看不到錯誤。 任何想法可能在哪里?

謝謝

我相信您正在同時讀取和寫入同一個文件 當您碰巧再次寫入文件時,輸入流也不會關閉。 這可能會使作者陷入奇怪的 state。 請您嘗試以下調整。

public void cargarVentasDiaMesaJson(Mesas m, double total) {
    String fileName = "C:\\archivos\\VentasDiaMesa.json";

    try {
        JsonObjectBuilder obj = Json.createObjectBuilder();
        JsonArrayBuilder arrayB = Json.createArrayBuilder();
        File file = new File(fileName);
        if (file.exists()) {
            // Try-with-resources: will auto-close JsonReader and inputStream
            //  >> close() of this stream was missing in original code
            try (JsonReader jRe = Json.createReader(new FileInputStream(file))) {
                JsonObject jsonR = jRe.readObject();
                JsonArray array = jsonR.getJsonArray("ventasDiaMesa");
                for (int i = 0; i < array.size(); i++) {
                    JsonObject vDiaMesa = array.getJsonObject(i);
                    String fecha = vDiaMesa.getString("fecha");
                    int numero = vDiaMesa.getInt("numero");
                    double tot = Double.parseDouble(vDiaMesa.getString("total"));
                    JsonObjectBuilder diaMesa = Json.createObjectBuilder();
                    diaMesa.add("fecha", fecha);
                    diaMesa.add("numero", numero);
                    diaMesa.add("total", tot);
                    JsonObject objLinea = diaMesa.build();
                    arrayB.add(objLinea);
                }
            }
        }
        JsonObjectBuilder diaMesa = Json.createObjectBuilder();
        Calendar fecha = Calendar.getInstance();
        String mes = (fecha.get(Calendar.MONTH) + 1) < 10 ? "0" + (fecha.get(Calendar.MONTH) + 1) : "" + (fecha.get(Calendar.MONTH) + 1);
        String dia = fecha.get(Calendar.DATE) < 10 ? "0" + fecha.get(Calendar.DATE) : "" + fecha.get(Calendar.DATE);
        String date = "" + fecha.get(Calendar.YEAR) + mes + dia;
        diaMesa.add("fecha", date);
        String numero = (m.getNumero() < 10) ? "0" + m.getNumero() : "" + m.getNumero();
        diaMesa.add("numero", numero);
        String tot = Double.toString(total);
        while (tot.length() != 10) {
            tot = "0" + tot;
        }
        diaMesa.add("total", tot);
        JsonObject objLinea = diaMesa.build();
        arrayB.add(objLinea);
        JsonArray arr = arrayB.build();
        obj.add("ventasDiaMesa", arr);
        JsonObject obbbj = obj.build();

        // (Re)Write object
        //   >> Opening of this stream was made too early in original code.
        JsonWriter jR;
        try {
            jR = Json.createWriter(new FileOutputStream(fileName));
            jR.writeObject(obbbj);
        } catch (IOException ex) {
            ex.printStackTrace();
        } finally {
            jR.close();
        }
    } catch (Exception e) {
        // TODO: handle exception
        e.printStackTrace();
    }
}

非常感謝您在發布代碼之前清理代碼。 它幾乎不可讀。 我建議至少:

  • 通過為變量和 CamelCase 提供人類可讀的名稱來使用正確的變量命名。 盡量避免使用單字母和雙字母名稱。
  • 創建更多方法
  • 幫助你的代碼呼吸
  • 創建更少無用(一次性)的變量或引用

例如:

public void cargarVentasDiaMesaJson(Mesas m, double total) {
    String fileName = "C:\\archivos\\VentasDiaMesa.json";

    try {
        JsonArrayBuilder updReadArrayB = Json.createArrayBuilder();

        File file = new File(fileName);
        if (file.exists()) {
            // Try-with-resources: will auto-close JsonReader and inputStream
            //  >> close() of this stream was missing in original code
            try (JsonReader jReader = Json.createReader(new FileInputStream(file))) {
                readPrepareObject(jReader, updReadArrayB);
            } catch (IOException ex) {
                // Cannot read file
                ex.printStackTrace();
            }

            JsonObject outJsonObj = rebuildObject(m, total, updReadArrayB, Json.createObjectBuilder());

            // (Re)Write object
            //   >> Opening of this stream was made too early in original code.
            JsonWriter jWriter;
            try {
                jWriter = Json.createWriter(new FileOutputStream(fileName));
                jWriter.writeObject(outJsonObj);
            } catch (IOException ex) {
                ex.printStackTrace();
            } finally {
                jWriter.close();
            }
        }
    } catch (Exception e) {
        // TODO: handle exception
        e.printStackTrace();
    }
}

private void readPrepareObject(final JsonReader jRe, JsonArrayBuilder arrayB) throws NumberFormatException {
    JsonObject jsonR = jRe.readObject();
    JsonArray array = jsonR.getJsonArray("ventasDiaMesa");

    for (int i = 0; i < array.size(); i++) {
        JsonObject vDiaMesa = array.getJsonObject(i);
        String fecha = vDiaMesa.getString("fecha");
        int numero = vDiaMesa.getInt("numero");
        double tot = Double.parseDouble(vDiaMesa.getString("total"));
        JsonObjectBuilder diaMesa = Json.createObjectBuilder();
        diaMesa.add("fecha", fecha);
        diaMesa.add("numero", numero);
        diaMesa.add("total", tot);

        arrayB.add(diaMesa.build());
    }
}

private JsonObject rebuildObject(Mesas m, double total, JsonArrayBuilder arrayB, JsonObjectBuilder obj) {
    JsonObjectBuilder diaMesa = Json.createObjectBuilder();

    Calendar fecha = Calendar.getInstance();
    String mes = (fecha.get(Calendar.MONTH) + 1) < 10 ? "0" + (fecha.get(Calendar.MONTH) + 1) : "" + (fecha.get(Calendar.MONTH) + 1);
    String dia = fecha.get(Calendar.DATE) < 10 ? "0" + fecha.get(Calendar.DATE) : "" + fecha.get(Calendar.DATE);
    String date = "" + fecha.get(Calendar.YEAR) + mes + dia;
    diaMesa.add("fecha", date);
    String numero = (m.getNumero() < 10) ? "0" + m.getNumero() : "" + m.getNumero();
    diaMesa.add("numero", numero);
    String tot = Double.toString(total);
    while (tot.length() != 10) {
        tot = "0" + tot;
    }
    diaMesa.add("total", tot);
    JsonObject objLinea = diaMesa.build();
    arrayB.add(objLinea);
    JsonArray arr = arrayB.build();
    obj.add("ventasDiaMesa", arr);

    return obj.build();
}

請注意,我沒有檢查(並且幾乎沒有修改)您的Json 操作說明。

最后但同樣重要的是:不要在 AWT 事件線程中執行長的、阻塞的或與 IO 相關的代碼。 您應該使用 ExecutorService 在 AWT 事件隊列之外執行大部分邏輯。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM