Как заставить абстрактный класс работать с JAXB

Дорогие товарищи по программированию java, я использовал пример из http://www.vogella.com/articles/JAXB/article.html

для использования JAXB XML для моих 3 классов: UserStorage, User и UserTest

он работает нормально, но это просто демаркизация

JAXBContext context = JAXBContext.newInstance(UserStorage.class);
                Marshaller m = context.createMarshaller();
                m.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, Boolean.TRUE);

the User class is ABSTRACT!, so it throws an 

Есть ли какое-либо решение для этого, чтобы я мог привязать класс User к XML, поскольку я могу сохранить файл XML с данными пользователя, но когда я хочу его получить, он утверждает, что класс User является абстрактным, у меня есть администратор , брокер, подклассы акционеров, но пока что в моем пользовательском тестовом классе я создал только 4 админа для тестирования, спасибо и надеюсь, что вы можете помочь.

// частный статический экземпляр UserStorage; // конец кодов JAXB james.wyche.

import platform.UserStorage;
import platform.User;



public class UserTest {

private static final String USER_XML = "user2.xml";

public static void main(String[] args) throws JAXBException, IOException {

    ArrayList<User> userList = new ArrayList<User>();

    // create test users
            User user1 = new Admin();
            user1.setName("Dave");
            user1.setPass("1234");
            user1.setDeleted(true);
            user1.setBan(false);
            userList.add(user1);

            User user2 = new Admin();
            user2.setName("James");
            user2.setPass("1234");
            user2.setDeleted(true);
            user2.setBan(false);
            userList.add(user2);

            User user3 = new Admin();
            user3.setName("Mike");
            user3.setPass("1234");
            user3.setDeleted(true);
            user3.setBan(false);
            userList.add(user3);


            // create bookstore, assigning book
            UserStorage userstore = new UserStorage();
            userstore.setListName("Test List");
            userstore.setUserList(userList);

            // create JAXB context and instantiate marshaller
            JAXBContext context = JAXBContext.newInstance(UserStorage.class);
            Marshaller m = context.createMarshaller();
            m.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, Boolean.TRUE);
            m.marshal(userstore, System.out);

            Writer w = null;
            try {
                w = new FileWriter(USER_XML);
                m.marshal(userstore, w);
            } finally {
                try {
                    w.close();
                } catch (Exception e) {
                }
            }

            // get variables from our xml file, created before
            System.out.println();
            System.out.println("Output from our XML File: ");
            Unmarshaller um = context.createUnmarshaller();
            UserStorage userstore2 = (UserStorage) um.unmarshal(new FileReader(
                    USER_XML));

            for (int i = 0; i < userstore2.getUsersList().toArray().length; i++) {
                System.out.println("User " + (i + 1) + ": "
                        + userstore2.getUsersList().get(i).getName() + " Pass "
                        + userstore2.getUsersList().get(i).getPass());
            }}  }



package platform;


import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.bind.annotation.XmlType;


//If you want you can define the order in which the fields are written
//Optional

/**
 * @author dinesh.kaushish, james.wyche //updated XML properties.
 * 
 */
@XmlRootElement(name = "user")
@XmlType(propOrder = { "name", "pass", "deleted", "ban" })

..

public abstract class User implements UserInterface {

private String name;
private String pass;
private boolean deleted;
private boolean ban;

/**
 * @ author dinesh.kaushish
 * @param String username
 * return void
 */

public void setName(String name)
{
    this.name = name;
}

// If you like the variable name, e.g. "name", you can easily change this
    // name for your XML-Output:

/**
 * @author dinesh.kaushish
 * @param null
 * @return String user;
 */
@XmlElement(name = "user")
public String getName()
{
    return this.name;
}

/**
 * @author dinesh.kaushish
 * @param String pwd
 * @return void
 */
public void setPass(String pass)
{
    this.pass=pass;
}

/**
 * @author dinesh.kaushish
 * @param void
 * @return String password
 */
@XmlElement(name = "pass")
public String getPass()
{
    return pass;
}

/**
 * @author dinesh.kaushish
 * @param dFlag
 * @return void
 */
public void setDeleted(boolean deleted)
{
    this.deleted = deleted;
}

/**
 * @author dinesh.kaushish
 * @return boolean isDeleted
 */
@XmlElement(name = "deleted")
public boolean getDeleted()
{
    return deleted;
}


/**
 * @author dinesh.kaushish
 * @param bFlag
 */
public void setBan(boolean ban)
{
    this.ban = ban;
}

/**
 * @author dinesh.kaushish
 * @return Boolean isBanned
 */
@XmlElement(name = "ban")
public Boolean getBan()
{
    return ban;
}


public abstract void addUser();
public abstract void removeUser();
public abstract void verifyUser();
public abstract void passReset();
public abstract void faultReport();
public abstract void RequestsForAccess();
public abstract void UpdateDetails();
public abstract void BanUser();
public abstract void ChangePermissions();


    }



package platform;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.util.HashMap;

import javax.xml.stream.XMLEventReader;
import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLStreamException;
import javax.xml.stream.events.XMLEvent;

import java.util.ArrayList;

import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlElementWrapper;
import javax.xml.bind.annotation.XmlRootElement;





/**
 * @author michael.wambeek, james.wyche //added JAXB support.
 *
 */
//This statement means that class "Bookstore.java" is the root-element of our example
@XmlRootElement(namespace = "platform")
public class UserStorage {

// XmLElementWrapper generates a wrapper element around XML representation
    @XmlElementWrapper(name = "userList")
    // XmlElement sets the name of the entities
    @XmlElement(name = "user")
private ArrayList<User> userList;
private String listName = "";

Это не удается, потому что Jaxb попытается создать экземпляр User. что абстрактно и, следовательно, провал.

public UserStorage(){

}

/**
 * @author michael.wambeek
 * 
 * Searches for a username and returns the password.
 * 
 * @param username The username to search for
 * @return The password of the correct username or null
 * @throws Exception
 */
public String findUser(String username) throws Exception{
    return search(username);
}

public boolean storeUser(String username, String password, UserType type){
    return true;
}

/**
 * @author james.wyche
 * @param userList
 */
public void setUserList(ArrayList<User> userList) {
    this.userList = userList;
}

/**
 * 
 * @return UserList
 */
public ArrayList<User> getUsersList() {
    return userList;
}

person wyche5000    schedule 22.04.2012    source источник


Ответы (4)


В вашем абстрактном классе добавьте аннотации

см. javadoc для каждого (XmlTransient, XmlSeeAlso < / а>)

@XmlTransient //Prevents the mapping of a JavaBean property/type to XML representation
@XmlSeeAlso({Admin.class, <other class>}) //Instructs JAXB to also bind other classes when binding this class

Это предотвратит попытку jaxb инициализировать ваш абстрактный класс.

Единственным недостатком этого метода, который я обнаружил, является добавление дополнительной информации о пространстве имен к создаваемому xml.

Вам необходимо добавить XmlSeeAlso аннотация к классу User с атрибутами Admin и всеми другими конкретными классами, являющимися подклассами класса User.

person Sean    schedule 22.04.2012
comment
Большое спасибо Ноам, Ян и Шон, я исправил это с помощью ваших решений :) - person bdoughan; 22.04.2012

PS, не забудьте добавить тег Xml @XmlRootElement в класс Admin.

@XmlSeeAlso({Admin.class})

Это не может сработать, потому что JAXB должен создавать новые экземпляры (объекты) ваших классов при демаршалинге xml. И если тег из xml привязан к абстрактному классу, он просто не может создать экземпляр объекта из этого класса. Вам нужно либо сделать класс User неабстрактным, либо привязать тег xml к конкретному подклассу User.

person Noam    schedule 22.04.2012
comment
Исключение в потоке «main» javax.xml.bind.UnmarshalException: невозможно создать экземпляр platform.User - со связанным исключением: [java.lang.InstantiationException] в com.sun.xml.internal.bind.v2.runtime. unmarshaller.UnmarshallingContext.handleEvent (UnmarshallingContext.java:648) в com.sun.xml.internal.bind.v2.runtime.unmarshaller.Loader.reportError (Loader.java:236) в com.sun.xml.internal.bind. v2.runtime.unmarshaller.UnmarshallingContext.createInstance (UnmarshallingContext.java:615) в com.sun.xml.internal.bind.v2.runtime.unmarshaller.StructureLoader.startElement (StructureLoader.java:170.x) в com. internal.bind.v2.runtime.unmarshaller.UnmarshallingContext._startElement (UnmarshallingContext.java:487) в com.sun.xml.internal.bind.v2.runtime.unmarshaller.UnmarshallingContext.startEleing4. sun.xml.internal.bind.v2.runtime.unmarshaller.SAXConnector.startElement (SAXConnector.java:135) в com.sun. org.apache.xerces.internal.parsers.AbstractSAXParser.startElement (AbstractSAXParser.java:501) по адресу com.sun.org.apache.xerces.internal.impl.XMLNSDocumentScannerImpl.scanStartElement (XMLNSDocument.java:unner). или sun.org.apache.xerces.internal.impl.XMLNSDocumentScannerImpl.next (XMLNSDocumentScannerImpl.java:140) в com.sun.org.apache.xerces.internal.impl.XMLDocumentFragmentScannerImpl.scanDocument.javajannerfragment.scanDocument (XMLDocumentFragmentScannerImpl.scanDocument. sun.org.apache.xerces.internal.parsers.XML11Configuration.parse (XML11Configuration.java:808) на com.sun.org.apache.xerces.internal.parsers.XML11Configuration.parse (XML11Configuration.java:737) на com. sun.org.apache.xerces.internal.parsers.XMLParser .parse (XMLParser.java:119) на com.sun.org.apache.xerces.internal.parsers.AbstractSAXParser.parse (AbstractSAXParser.java:1205) на com.sun.org.apache.xerces.internal.jaxp.SAXParserImpl $ JAXPSAXParser.parse (SAXParserImpl.java:522) в com.sun.xml.internal.bind.v2.runtime.unmarshaller.UnmarshallerImpl.unmarshal0 (UnmarshallerImpl.java:200) в com.sun.xml.v2.bind. .runtime.unmarshaller.UnmarshallerImpl.unmarshal (UnmarshallerImpl.java:173) в javax.xml.bind.helpers. AbstractUnmarshallerImpl.unmarshal (AbstractUnmarshallerImpl.java:137) на javax.xml.bind.helpers.AbstractUnmarshallerImpl.unmarshal (AbstractUnmarshallerImpl.java:194) на платформе.UserTest.main (UserTestationException.java ::77 в sun.reflect.InstantiationExceptionConstructorAccessorImpl.newInstance (InstantiationExceptionConstructorAccessorImpl.java:30) в java.lang.reflect.Constructor.newInstance (Constructor.java:513) в com.sun.xml.internal.bindactory.v2.Class .java: 112) в com.sun.xml.internal.bind.v2.runtime.ClassBeanInfoImpl.createInstance (ClassBeanInfoImpl.java:231) в com.sun.xml.internal.bind.v2.runtime.unmarshaller.UnmarshallingContext.createInstance. (UnmarshallingContext.java:609) ... еще 20 - person wyche5000; 22.04.2012

Вы должны указать _1_ тип каждого элемента:

person Jan    schedule 22.04.2012

Добавление XmlTransient в суперкласс удалит отношения наследования, которые могут быть нежелательными, см. blog.bdoughan .com / search / label / XmlTransient. Заполненная статья может помочь: blog.bdoughan.com/ 2010/11 /

<user xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="Admin">
    ...
</user>
person Claus Radloff    schedule 06.02.2017