可扩展性是许多体系结构的重要特征。 它衡量是否容易(或困难) 它是在不影响现有核心系统功能的情况下添加或更改功能。
让我们举一个简单的例子。 假设您的公司拥有一个核心产品来跟踪体育俱乐部中的所有用户。 在您的产品体系结构中,您有一个由JPA POJO表示的域模型。 域模型包含许多POJO,当然包括用户POJO。
package com.alex.staveley.persistence
/**
* User entity. Represents Users in the Sports Club.
* * Note: The SQL to generate a table for this in MySQL is:
*
* CREATE TABLE USER (ID INT NOT NULL auto_increment, NAME varchar(255) NOT NULL,
* PRIMARY KEY (ID)) ENGINE=InnoDB;
*/
@Entity
public class User {
/* Surrogate Key - automatically generated by DB. */
@GeneratedValue(strategy=GenerationType.IDENTITY)
@Id
private int id;
private String name;
public int getId() {
return id;
}
public void setName(String name) {
this.name=name;
}
public String getName() {
return name;
}
}
现在,一些客户喜欢您的产品,但是他们需要在购买之前进行一些定制。 例如, 一位客户希望将属性出生地添加到用户,并希望此属性持续存在。 当然,此属性的逻辑位置是在用户POJO中,但是没有其他客户想要此属性。 所以你会怎么做?
您是否仅为此客户创建了特定的User类,然后才为他们交换了该类? 怎么了 当您更改产品用户类别时呢? 如果另一个客户想要另一个定制怎么办? 还是改变主意? 您是否感觉事情会变得混乱?
幸运的是,JPA的一种实现:Eclipselink在这里提供了帮助。 2.3版本 (自2011年6月开始提供,最新版本是2011年12月9日最近发布的2.3.2维护)包括一些非常好的功能,可以解决这种情况。 让我们详细说明。 通过将@VirtualAccessmethods Eclipselink注释简单地添加到POJO,我们向Eclipselink发出信号,表明POJO可能具有一些额外的( 也称为虚拟 )属性。 您不必在代码中指定任何这些额外的属性, 否则它们将不是非常虚拟的 ! 您只需要指定一个通用的getter和setter来满足他们的获取和设置。 您还必须在某个地方将它们存储在内存中,就像一个很好的旧哈希图-当然这应该是暂时的,因为我们不持久保存哈希图本身。 注意:它们不必存储在HashMap中,这只是一个流行的选择!
让我们看一下我们现在可以扩展的用户。
@Entity
@VirtualAccessMethods
public class User {
/* Surrogate Key - automatically generated by DB. */
@GeneratedValue(strategy=GenerationType.IDENTITY)
@Id
private int id;
private String name;
@Transient
private Map<String, Object> extensions = new HashMap();
public int getId() {
return id;
}
public void setName(String name) {
this.name=name;
}
public String getName() {
return name;
}
public <t> T get(String name) {
return (T) extensions.get(name);
}
public Object set(String name, Object value) {
return extensions.put(name, value);
}
}
那是吗? 好吧,还有更多的魔术。 您必须告诉eclipselink您的其他属性。 更具体地说:它们的名称和数据类型是什么。 您可以通过更新eclipselink-orm.xml来做到这一点,该文件位于persistent.xml所在的同一META-INF文件夹中。
<?xml version="1.0" encoding="UTF-8"?>
<entity-mappings xmlns="http://www.eclipse.org/eclipselink/xsds/persistence/orm"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.eclipse.org/eclipselink/xsds/persistence/orm
http://www.eclipse.org/eclipselink/xsds/eclipselink_orm_2_1.xsd"
version="2.1">
<entity class="com.alex.staveley.persistence.User">
<attributes>
<basic name="thebirthplace" attribute-type="String" access="VIRTUAL">
<column name="birthplace"/>
<access-methods get-method="get" set-method="set"/>
</basic>
</attributes>
</entity>
</entity-mappings>
现在,此配置简单说明,用户实体具有一个附加属性,在Java中,该属性是“ thebirthplace”,并且是虚拟的。 这意味着它不是在POJO中明确定义的,但是如果我们要调试东西,我们会在内存中看到该属性的名称为“出生地”。
此配置还指出该属性的相应数据库列是出生地。 eclipselink可以使用通用的get / set方法来获取和设置此方法。
你想测试吗?
将列添加到数据库表中。 在MySql中,这将是:
alter table用户添加列的出生地varchar(64)
然后运行以下简单测试:
@Test
public void testCreateUser() {
User user = new User();
user.setName("User1Name");
user.set("thebirthplace", "donabate");
entitymanager.getTransaction().begin();
entitymanager.persist(user);
entitymanager.getTransaction().commit();
entitymanager.close();
}
因此,现在,我们可以在产品代码中添加一个可扩展的用户POJO。 每个客户都可以根据需要将自己的属性添加到用户。 当然,每个客户都与 仅通过确保每个客户的扩展都驻留在特定的eclipslink-orm.xml中,即可非常轻松地实现所有其他客户。 请记住,您可以根据需要随意命名这些文件,如果不使用默认名称,则只需更新persistence.xml文件以声明正在使用的名称
因为我们确保只有一个 )。 但是,当必须为特定客户添加特定属性时,我们无需触摸用户POJO代码。 我们简单地对XML进行更改,而不必重新编译核心产品中的任何内容。 当然,在任何时候,只要查看适当的eclipselink-orm.file,就可以轻松了解针对任何客户的定制。
是的 快乐扩展!
参考文献:
- 在都柏林的技术博客上,从我们的JCG合作伙伴 Alex Staveley 扩展您的JPA POJO
- http://wiki.eclipse.org/EclipseLink/UserGuide/JPA/Advanced_JPA_Development/Extensible_Entities
- http://www.eclipse.org/eclipselink/
相关文章 :
- Spring Data JPA的持久层
- 具有GlassFish和一致性的高性能JPA –第1部分
- 避免延迟的JPA集合
- JBoss 4.2.x Spring 3 JPA Hibernate教程
- Java Persistence API:快速入门
翻译自: https://www.javacodegeeks.com/2012/01/extending-your-jpa-pojos.html