代码之家  ›  专栏  ›  技术社区  ›  Niklas P

无法使EclipseLink使用方法而不是字段

  •  1
  • Niklas P  · 技术社区  · 9 年前

    我对JPA相当陌生,我正在尝试与 EclipseLink 2.6.1-RC1 sqlite 3.8.10.1 .

    我的实际挑战是让 持久性提供者 使用get-/set方法,而不是直接访问字段。

    我在使用EclipseLink时遇到了很多问题,目前我正在将我的简单测试项目与Hibernate进行比较,我认为Hibernate没有问题。

    我是以错误的方式使用JPA还是我的错误是EclipseLink错误?

    下面是我的例子:

    persistence.xml :

    <persistence-unit name="inmemorydb" transaction-type="RESOURCE_LOCAL">
        <provider>org.eclipse.persistence.jpa.PersistenceProvider</provider>
        <exclude-unlisted-classes>false</exclude-unlisted-classes>
        <properties>
            <property name="eclipselink.target-database" value="Database" />
            <property name="javax.persistence.jdbc.url" value="jdbc:sqlite::memory:"/>
            <property name="javax.persistence.jdbc.driver" value="org.sqlite.JDBC"/>
        </properties>
    </persistence-unit>
    

    SimpleEntity.java :

    @Entity
    public class SimpleEntity implements Serializable {
    private static final long serialVersionUID = 1L;
    
    @Id
    private long id;
    
    @Transient
    private String otherName;
    
    public long getId() {
        return id;
    }
    
    public void setId(long id) {
        this.id = id;
    }
    
    @Basic
    @Column(name="name")
    public String getOtherName() {
        System.out.println("getOtherName: " + otherName);
        return otherName;
    }
    
    public void setOtherName(String otherName) {
        System.out.println("setOtherName: " + otherName);
        this.otherName = otherName;
    }
    }
    

    JpaTest.java :

    public class JpaTest {
    
    public static final String PERSISTENCE_UNIT = "inmemorydb";
    
    public static final String SQL_CREATE_TABLE = "CREATE TABLE simpleentity (id INTEGER PRIMARY KEY, name TEXT);";
    
    private static EntityManagerFactory dbConnectionPool;
    
    @BeforeClass
    public static void initialiseDatabase() {
        System.out.println("initialiseDatabase");
        dbConnectionPool = Persistence.createEntityManagerFactory(PERSISTENCE_UNIT);
        EntityManager dbConnection = dbConnectionPool.createEntityManager();
        dbConnection.getTransaction().begin();
    
        Query createTableQuery = dbConnection.createNativeQuery(SQL_CREATE_TABLE);
        createTableQuery.executeUpdate();
    
        dbConnection.getTransaction().commit();
        dbConnection.close();
    }
    
    @Test
    public void createTable_InsertEntity() {
        System.out.println("createTable_InsertEntity");
        EntityManager dbConnection = dbConnectionPool.createEntityManager();
        dbConnection.getTransaction().begin();
    
        SimpleEntity testEntity = new SimpleEntity();
        testEntity.setId(1);
        testEntity.setOtherName("my name");
        dbConnection.persist(testEntity);
    
        dbConnection.getTransaction().commit();
        dbConnection.close();
    }
    
    @Test
    public void createTable_InsertEntity_ReadEntity() {
        System.out.println("createTable_InsertEntity_ReadEntity");
        EntityManager dbConnection = dbConnectionPool.createEntityManager();
        dbConnection.getTransaction().begin();
    
        SimpleEntity testEntity = new SimpleEntity();
        testEntity.setId(2);
        testEntity.setOtherName("my other name");
        dbConnection.persist(testEntity);
    
        dbConnection.getTransaction().commit();
        dbConnection.getTransaction().begin();
    
        TypedQuery<SimpleEntity> findEntityQuery = dbConnection.createQuery("SELECT e FROM SimpleEntity e WHERE e.id = 2", SimpleEntity.class);
        SimpleEntity foundEntity = findEntityQuery.getSingleResult();
        dbConnection.close();
    
        assertEquals(2, foundEntity.getId());
        assertEquals("my other name", foundEntity.getOtherName());
    }
    }
    

    正如你所想,我会看到 Eclipse链接 呼叫我的 getOtherName() setOtherName(String) (我不确定)。但简化的结果是:

    initialiseDatabase
    [EL Info]: server: 2015-09-16 01:07:07.877--Detected server platform: org.eclipse.persistence.platform.server.NoServerPlatform.
    [EL Info]: server: 2015-09-16 01:07:07.969--ServerSession(1845066581)--Detected server platform: org.eclipse.persistence.platform.server.NoServerPlatform.
     [EL Info]: server: 2015-09-16 01:07:08.184--Detected server platform: org.eclipse.persistence.platform.server.NoServerPlatform.
    [EL Info]: server: 2015-09-16 01:07:08.184--ServerSession(1845066581)--Detected server platform: org.eclipse.persistence.platform.server.NoServerPlatform.
    [EL Info]: 2015-09-16 01:07:08.187--ServerSession(1845066581)--EclipseLink, version: Eclipse Persistence Services - 2.6.1.v20150605-31e8258
    [EL Info]: connection: 2015-09-16 01:07:09.017--ServerSession(1845066581)--/file:/C:/eclipse_ee/workspaces/jpaproperty/target/classes/_inmemorydb login successful
    createTable_InsertEntity_ReadEntity
    setOtherName: my other name
    getOtherName: my other name
    createTable_InsertEntity
    setOtherName: my name
    

    set和get方法的打印调用是我在测试类中的调用,因此 Eclipse链接 ? 我真的尝试了很多不同的设置 @Transient , @Basic , @Column 但没有成功。

    当我尝试使用所谓的标准 @Access(AccessType.PROPERTY) 我甚至犯了一个非常奇怪的错误:

    Exception [EclipseLink-30005] (Eclipse Persistence Services - 2.6.1.v20150605-31e8258): org.eclipse.persistence.exceptions.PersistenceUnitLoadingException
    Exception Description: An exception was thrown while searching for persistence archives with ClassLoader: sun.misc.Launcher$AppClassLoader@29453f44
    Internal Exception: javax.persistence.PersistenceException: Exception [EclipseLink-28018] (Eclipse Persistence Services - 2.6.1.v20150605-31e8258): org.eclipse.persistence.exceptions.EntityManagerSetupException
    Exception Description: Predeployment of PersistenceUnit [inmemorydb] failed.
    Internal Exception: java.lang.ClassCastException: org.eclipse.persistence.internal.jpa.metadata.accessors.objects.MetadataField    cannot be cast to org.eclipse.persistence.internal.jpa.metadata.accessors.objects.MetadataMethod
    

    这不是JPA标准吗?EclipseLink不符合这一标准吗?

    顺便说一下,当我使用Hibernate时,输出如下:

    createTable_InsertEntity
    setOtherName: my name
    getOtherName: my name
    getOtherName: my name
    getOtherName: my name
    createTable_InsertEntity_ReadEntity
    setOtherName: my other name
    getOtherName: my other name
    getOtherName: my other name
    getOtherName: my other name
    getOtherName: my other name
    getOtherName: my other name
    getOtherName: my other name
    

    你能帮我吗?请尽量不要说服我 冬眠 -我的第一个目标是了解什么是JPA标准,什么不是!

    2 回复  |  直到 9 年前
        1
  •  1
  •   Brian Vosburgh    9 年前

    你的问题不清楚你是如何使用 @Access 注释;但是您需要在两个地方指定:一个是在类上指定实体属性的默认值,另一个是不使用默认访问权限的属性。因此:您的示例应该如下所示:

    @Entity
    @Access(AccessType.FIELD)
    public class SimpleEntity implements Serializable {
    private static final long serialVersionUID = 1L;
    
    @Id
    private long id;
    
    @Transient
    private String otherName;
    
    public long getId() {
        return id;
    }
    
    public void setId(long id) {
        this.id = id;
    }
    
    @Basic
    @Column(name="name")
    @Access(AccessType.PROPERTY)
    public String getOtherName() {
        System.out.println("getOtherName: " + otherName);
        return otherName;
    }
    
    public void setOtherName(String otherName) {
        System.out.println("setOtherName: " + otherName);
        this.otherName = otherName;
    }
    }
    
        2
  •  0
  •   Jacek Cz    9 年前

    不要使用@Transient over otherName .

    在方法(getter)上设置JPA注释与在字段上设置注释具有相同的含义,这是唯一不同的“方言”。 Eclipselink(在我看来)需要以一种方式实现,Hibernate更宽容(可以混合字段/方法注释)

    在我的个人工作中,永远不要对成员使用@Access(AccessType.PROPERTY)-如果满足上述规则,自动配置是可以的。