首页 > 解决方案 > JAXBException: SaveButton 及其任何超类对此上下文都是已知的

问题描述

我已经阅读了关于这个主题的几个主题,但似乎没有一个能回答我的具体问题。在包“设置”中,我有一个使用 jaxbSettingModel.java从文件加载默认设置的类。default.xml这完美无缺。然后,在同一个包下,在一个名为 的子包内SettingsWidgets,我有一个SaveButton.java类,它有一个setOnMouseClickeventListener,触发一个方法“saveSettings”。代码部分工作:当前文件被覆盖(尽管有 void)或在所需的包中创建了一个新的(但为空的)xml 文件,我得到了错误

javax.xml.bind.JAXBException: class settings.settingsWidgets.SaveButton nor any of its super class is known to this context.

我尝试过:
按照 Stack O 上的几个线程中提出的不同方式创建 JAXBContext。(将包作为字符串传递并添加 jaxb.in​​dex 和 ObjectFactory.java,传递完整的类路径而不仅仅是类,... )无济于事,在我看来(阅读错误)问题不在于我如何将 SettingsModel 类传递给上下文......经常提到@XmlSeealso
解决方法,但我没有立即看到任何使用在我的情况下(不是我试图编组一个子类或smth) 也试图从子包中重构,所以它会在同一级别上,但这并没有改变任何东西...... 我很迷失在如何以及为什么我应该包含在
SaveButton.javaSettingsModel
SaveButtonjaxbContext当我尝试时,我得到了很多错误。


我的代码:

// SettingsModel.java

@XmlRootElement(name = "robot")
@XmlAccessorType(XmlAccessType.FIELD)
public class SettingsModel implements Observable {

    private boolean valid = true;

    //registered views List
    //annotate as Type Object.Class because JAXB cannot handle interfaces
    @XmlElement(type = Object.class)
    private  ArrayList<InvalidationListener> listenersList;

    //list all variable robot/vehicle specs
    //filename is needed for saving to current or new file
    @XmlElement(name = "filename")
    private String fileName;

    @XmlElement(name = "vehicle-width")
    private double vehicleWidth;

    @XmlElement(name = "work-width")
    private double workWidth;

    // other getters and setters without logic are removed

    public void setValid(boolean valid) {
        if (valid != this.valid) {
            this.valid = valid;

            fireInvalidationEvent();
        }
    }

    public void setVehicleWidth(double vehicleWidth) {
        this.vehicleWidth = vehicleWidth;

        Field.getInstance().getRobot().setVehicleWidth(vehicleWidth);
        fireInvalidationEvent();
    }

    public void setWorkWidth(double workWidth) {
        this.workWidth = workWidth;

        // Field.getInstance().getRobot().setVehicleWidth(vehicleWidth);
        // fireInvalidationEvent();
    }



    //METHODS AND FUNCTIONS
    @Override
    public void addListener(InvalidationListener invalidationListener) {
        listenersList.add(invalidationListener);
    }

    @Override
    public void removeListener(InvalidationListener invalidationListener) {
        listenersList.remove(invalidationListener);
    }

    public void fireInvalidationEvent() {
        for (InvalidationListener listener : listenersList) {
            listener.invalidated(this);
        }
    }
}
// SaveButton.java

public class SaveButton extends Button implements InvalidationListener {

    private SettingsModel model;

    public SaveButton(SettingsModel model) {
        this.model = model;
        model.addListener(this);
        setText("SAVE SETTINGS");
        setOnMouseClicked((ev)->{
            try {
                saveSettings();
            } catch (JAXBException e) {
                e.printStackTrace();
            } catch (FileNotFoundException e) {
                e.printStackTrace();
            }
        });
    }

    private void saveSettings() throws JAXBException, FileNotFoundException {
        String filename = model.getFileName();

        Object[] options = {"Save", "Save As"};

        //choose Save or Save As
        int choice = JOptionPane.showOptionDialog(null,"Overwrite current file or Save As New file?",
                "Save Settings",
                JOptionPane.WARNING_MESSAGE,
                JOptionPane.YES_NO_OPTION,
                null,
                options,
                options[0]
                );

        if(choice != JOptionPane.OK_OPTION){
            filename = JOptionPane.showInputDialog("Enter new file name");
            model.setFileName(filename);
        }

        System.out.println(model.getFileName());

        //Marshal
        JAXBContext jaxbContext = JAXBContext.newInstance(SettingsModel.class);

        Marshaller jaxbMarshaller = jaxbContext.createMarshaller();

        //if filename stays same, file is overwritten. Test only SAVE AS until code works to not lose default file
        File file = new File("src/resources/properties/"+ filename + ".xml");

        OutputStream os = new FileOutputStream(file);

        jaxbMarshaller.marshal(model,os);
    }

    @Override
    public void invalidated(Observable observable) {
        System.out.println("button invalidated: "+ model.isValid());

        setDisable(!model.isValid());
    }
}

堆栈跟踪:

javax.xml.bind.MarshalException
 - with linked exception:
[com.sun.istack.SAXException2: javax.xml.bind.JAXBException: class settings.settingsWidgets.SaveButton nor any of its super class is known to this context.
javax.xml.bind.JAXBException: class settings.settingsWidgets.SaveButton nor any of its super class is known to this context.]
    at com.sun.xml.bind.v2.runtime.MarshallerImpl.write(MarshallerImpl.java:301)
    at com.sun.xml.bind.v2.runtime.MarshallerImpl.marshal(MarshallerImpl.java:226)
    at javax.xml.bind.helpers.AbstractMarshallerImpl.marshal(AbstractMarshallerImpl.java:80)
    at settings.settingsWidgets.SaveButton.saveSettings(SaveButton.java:59)
    at settings.settingsWidgets.SaveButton.lambda$new$0(SaveButton.java:27)
    at javafx.base/com.sun.javafx.event.CompositeEventHandler.dispatchBubblingEvent(CompositeEventHandler.java:86)
    at javafx.base/com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:238)
    at javafx.base/com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:191)
    at javafx.base/com.sun.javafx.event.CompositeEventDispatcher.dispatchBubblingEvent(CompositeEventDispatcher.java:59)
    at javafx.base/com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:58)
    at javafx.base/com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
    at javafx.base/com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
    at javafx.base/com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
    at javafx.base/com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
    at javafx.base/com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
    at javafx.base/com.sun.javafx.event.EventUtil.fireEventImpl(EventUtil.java:74)
    at javafx.base/com.sun.javafx.event.EventUtil.fireEvent(EventUtil.java:54)
    at javafx.base/javafx.event.Event.fireEvent(Event.java:198)
    at javafx.graphics/javafx.scene.Scene$ClickGenerator.postProcess(Scene.java:3564)
    at javafx.graphics/javafx.scene.Scene$ClickGenerator.access$8200(Scene.java:3492)
    at javafx.graphics/javafx.scene.Scene$MouseHandler.process(Scene.java:3860)
    at javafx.graphics/javafx.scene.Scene$MouseHandler.access$1200(Scene.java:3579)
    at javafx.graphics/javafx.scene.Scene.processMouseEvent(Scene.java:1849)
    at javafx.graphics/javafx.scene.Scene$ScenePeerListener.mouseEvent(Scene.java:2588)
    at javafx.graphics/com.sun.javafx.tk.quantum.GlassViewEventHandler$MouseEventNotification.run(GlassViewEventHandler.java:397)
    at javafx.graphics/com.sun.javafx.tk.quantum.GlassViewEventHandler$MouseEventNotification.run(GlassViewEventHandler.java:295)
    at java.base/java.security.AccessController.doPrivileged(Native Method)
    at javafx.graphics/com.sun.javafx.tk.quantum.GlassViewEventHandler.lambda$handleMouseEvent$2(GlassViewEventHandler.java:434)
    at javafx.graphics/com.sun.javafx.tk.quantum.QuantumToolkit.runWithoutRenderLock(QuantumToolkit.java:390)
    at javafx.graphics/com.sun.javafx.tk.quantum.GlassViewEventHandler.handleMouseEvent(GlassViewEventHandler.java:433)
    at javafx.graphics/com.sun.glass.ui.View.handleMouseEvent(View.java:556)
    at javafx.graphics/com.sun.glass.ui.View.notifyMouse(View.java:942)
    at javafx.graphics/com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
    at javafx.graphics/com.sun.glass.ui.win.WinApplication.lambda$runLoop$3(WinApplication.java:174)
    at java.base/java.lang.Thread.run(Thread.java:834)
Caused by: com.sun.istack.SAXException2: javax.xml.bind.JAXBException: class settings.settingsWidgets.SaveButton nor any of its super class is known to this context.
javax.xml.bind.JAXBException: class settings.settingsWidgets.SaveButton nor any of its super class is known to this context.
    at com.sun.xml.bind.v2.runtime.XMLSerializer.reportError(XMLSerializer.java:217)
    at com.sun.xml.bind.v2.runtime.XMLSerializer.reportError(XMLSerializer.java:232)
    at com.sun.xml.bind.v2.runtime.XMLSerializer.childAsXsiType(XMLSerializer.java:623)
    at com.sun.xml.bind.v2.runtime.property.ArrayElementNodeProperty.serializeItem(ArrayElementNodeProperty.java:39)
    at com.sun.xml.bind.v2.runtime.property.ArrayElementProperty.serializeListBody(ArrayElementProperty.java:142)
    at com.sun.xml.bind.v2.runtime.property.ArrayERProperty.serializeBody(ArrayERProperty.java:129)
    at com.sun.xml.bind.v2.runtime.ClassBeanInfoImpl.serializeBody(ClassBeanInfoImpl.java:329)
    at com.sun.xml.bind.v2.runtime.XMLSerializer.childAsSoleContent(XMLSerializer.java:563)
    at com.sun.xml.bind.v2.runtime.ClassBeanInfoImpl.serializeRoot(ClassBeanInfoImpl.java:310)
    at com.sun.xml.bind.v2.runtime.XMLSerializer.childAsRoot(XMLSerializer.java:464)
    at com.sun.xml.bind.v2.runtime.MarshallerImpl.write(MarshallerImpl.java:298)
    ... 34 more
Caused by: javax.xml.bind.JAXBException: class settings.settingsWidgets.SaveButton nor any of its super class is known to this context.
    at com.sun.xml.bind.v2.runtime.JAXBContextImpl.getBeanInfo(JAXBContextImpl.java:563)
    at com.sun.xml.bind.v2.runtime.XMLSerializer.childAsXsiType(XMLSerializer.java:618)
    ... 42 more

Process finished with exit code 0

任何见解将不胜感激。

标签: javajaxb

解决方案


您的 SettingsModel 有一个 listenerList 标记为 xmlelement。您可以使用以下行将 SaveButton 添加到此列表中: model.addListener(this);

因此,在编组 SettingsModel 时,JAXB 处理 listenerList 并尝试编组 SaveButton,但它不知道该怎么做。

向 SaveButton 类添加 @XmlRootElement 注释就足够了。

此外,我看到每次单击按钮时都会创建一个 JAXBContext。JAXB 规范说:

为了避免创建 JAXBContext 实例所涉及的开销,鼓励 JAXB 应用程序重用 JAXBContext 实例。抽象类 JAXBContext 的实现要求是线程安全的,因此,应用程序中的多个线程可以共享同一个 JAXBContext 实例。

您应该将 JAXBContext 移动到静态字段,因为可以重用该实例。


推荐阅读