Java SE還可以。
Java EE是邪惡的。
這就是我一直想的。 好吧,現在不再了。 讓我分享我的經驗。
幾周前,我開始考慮將舊版spring + hibernate + tomcat應用程序移植到新平臺上:
SAP NetWeaver云 。 我知道您在極客那里的想法:這篇文章越來越糟。 它從Java EE開始,不完全是一個怪胎,而現在進入SAP ,不完全是一個怪胎公司……請再給我十分鐘!
我的遺留應用程序的spring層的配置是基于xml的(它是在注釋進入游戲之前編寫的)。 我對再次陷入我自己的xml恐怖的前景感到恐懼。
然后出現此推文:
歡迎@ sap參加#JavaEE6聚會,NetWeaver Cloud現在已通過Java EE 6 Web Profile認證: bit.ly/Wf8VNS
— GlassFish(@glassfish) 2012年11月16日
幾天后,還有本文檔 。 我嘗試了。 而且有效。 我改變了對Java EE的想法。 “水管工比爾”(Bill the Plumber)有一篇博客文章, 準確地描述了我在經歷之后的想法 。
bla bla bla太多了。 讓我們開始編碼! 如果您趕時間,請從https://github.com/cthiebaud/adventscloud克隆完整的應用程序
在瘋狂剪切和粘貼之前,讓我們簡要描述一下下面的代碼。 我們將免費構建一個小型Web應用程序并將其部署到云中:
1.登錄用戶(對不起,您需要一個SAP Community Network帳戶,不用擔心它是免費的),
2.登錄后,代表用戶向世界其他地方問好,
3.連續登錄后,不必一遍又一遍地說“你好”,而只是在數據庫中存儲說了多少個“ hellos”,并且
4.就是這樣。
為此,我們需要一個Java接口,三個Java類,一個Java服務器頁面以及persistence.xml(用于數據庫配置)和web.xml(用于安全性約束向導)的最終版本。
為了簡潔起見,以下代碼中省略了包,導入,getter和setter。 但是,正如剛剛所說的, 完整的源代碼可以在github上找到。
編寫一個Hello.java POJO( 此處是完整的類):
public class Hello {private Long id;private String username;private Integer counter;private Timestamp when;// ... getters and setters ...
}
顯而易見:對于每個username
,此POJO都會在counter
存儲用戶點擊我們應用程序的index.jsp的次數,以及上次是when
。
使用JPA批注對此Hello.java POJO批注( 在此處完成類):
@Entity
@Table(name="T_HELLO")
@NamedQueries( {@NamedQuery(name = "allHellos", query = "select h from Hello h"),@NamedQuery(name = "helloFromUsername", query = "select h from Hello h where h.username = :username")
})
public class Hello {@Id@GeneratedValueprivate Long id;@Column(name="A_USER", unique=true, nullable=false)private String username;@Column(name="A_COUNTER", nullable=false)private Integer counter;@Version@Column(name="A_TIMESTAMP", nullable=false)private Timestamp when;public Hello() {this.counter = 1;}// ... getters and setters ...
}
編寫一個訪問POJO的HelloDao.java接口( 此處為完整接口)
@Local
public interface HelloDao {List<hello> getAll();Hello fromUsername(String username);Hello save(Hello hello);
}
編寫一個用EJB注釋注釋的HelloBean.java,它實現了HelloDao接口( 此處是完整的類):
@Stateless
public class HelloBean implements HelloDao {@PersistenceContextprivate EntityManager em;@Overridepublic List<hello> getAll() {@SuppressWarnings("unchecked")List<hello> hellos = (List<hello>)em.createNamedQuery("allHellos").getResultList();Collections.sort(hellos, new Comparator<hello>() {@Overridepublic int compare(Hello o1, Hello o2) {return o2.getWhen().compareTo(o1.getWhen()); // latest first}});return hellos;}@Overridepublic Hello fromUsername(String username) {Query query = em.createNamedQuery("helloFromUsername");query.setParameter("username", username);Hello hello = null;try {hello = (Hello)query.getSingleResult();} catch (NoResultException ignored) {}return hello;}@TransactionAttribute@Overridepublic Hello save(Hello hello) {hello = em.merge(hello);return hello;}
}
編寫一個HelloFilter.java Java Servlet 3過濾器,該過濾器是1.登錄時更改計數器,并且2.在即將推出的Java Server頁面( 此處是完整類)中公開HelloBean實例:
@WebFilter("/index.jsp")
public final class HelloFilter implements Filter {@EJBHelloDao helloDao;@Overridepublic void init(FilterConfig fConfig) throws ServletException {}@Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {try {request.setAttribute("helloDao", helloDao);String username = ((HttpServletRequest)request).getRemoteUser();Hello hello = helloDao.fromUsername(username);if (hello == null) {hello = new Hello();hello.setUsername(username);} else {hello.setCounter(hello.getCounter()+1);}hello = helloDao.save(hello);chain.doFilter(request, response);} finally {request.removeAttribute("helloDao");}}@Overridepublic void destroy() {}
}
注意 上面的粗體字是我們的Java EE 6 Web Profile容器在所有這些類之間完成的神奇工作:
@PersistenceContext EntityManager em;
@EJB HelloDao helloDao;
@WebFilter('/index.jsp')
編寫一個persistence.xml JPA配置( 在此處完成xml)
<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.0" xmlns="http://java.sun.com/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"><persistence-unit name="adventscloud-persist" transaction-type="JTA"><provider>org.eclipse.persistence.jpa.PersistenceProvider</provider><jta-data-source>jdbc/DefaultDB</jta-data-source><class>net.aequologica.adventscloud.Hello</class><properties><property name="eclipselink.ddl-generation" value="create-or-extend-tables" /></properties></persistence-unit>
</persistence>
編寫一個web.xml,以在用戶訪問index.jsp時觸發登錄,并通知Web應用程序容器管理數據庫的存在( 在此處完整xml):
<?xml version="1.0" encoding="UTF-8"?>
<web-appxmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns="http://java.sun.com/xml/ns/javaee"xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"version="3.0"><login-config><auth-method>FORM</auth-method></login-config><security-constraint><web-resource-collection><web-resource-name>Protected Area</web-resource-name><url-pattern>/index.jsp</url-pattern></web-resource-collection><auth-constraint><role-name>Everyone</role-name></auth-constraint></security-constraint><security-role><description>All SAP NetWeaver Cloud users</description><role-name>Everyone</role-name></security-role><resource-ref><res-ref-name>jdbc/DefaultDB</res-ref-name><res-type>javax.sql.DataSource</res-type></resource-ref></web-app>
再次,NB。 以上以粗體顯示的進一步魔術效果:
<jta-data-source>jdbc/DefaultDB</jta-data-source>
<class>net.aequologica.adventscloud.Hello</class>
<res-ref-name>jdbc/DefaultDB</res-ref-name>
最后,編寫一個index.jsp Java服務器頁面,其中顯示所有“ hellos”( 此處完整頁面):
<%@ taglib prefix="c" uri="http://java.sun.com/jstl/core_rt" %>
<%@ taglib prefix="fmt" uri="http://java.sun.com/jstl/fmt_rt" %>
<!DOCTYPE html>
<html><head><title>adventscloud</title></head><body><table><tbody><c:forEach var="hello" items="${requestScope.helloDao.all}" varStatus="status"><tr><td><fmt:formatDate type="both" value="${hello.when}" /></td><td>${hello.counter}</td><td>hello<c:if test = "${hello.counter > 1}">(s)</c:if> from</td><td>${hello.username}</td></tr></c:forEach></tbody></table> </body></html>
我們幾乎完成了……最后兩件事:1.類路徑地獄,以及2.使用javac -processor生成JPA 2.0元模型。
1.類路徑地獄。
為了編譯所有這些東西,您需要某種方式在類路徑上具有以下jar:
group | artifact | version
javax.persistence : persistence-api : >= 1.0
javax.ejb : ejb-api : >= 3.0
javax.servlet : javax.servlet-api : >= 3.0
javax.servlet : jstl : >= 1.2
當然,最簡單的方法是在像我的maven項目中聲明這些依賴關系,但是如果您是厭惡Maven的人,我會花一些時間超鏈接到上面的jars,以便在maven Central中節省一些時間來追逐jars。
2.使用javac -processor生成JPA 2.0元模型。
最后,該構建必須能夠生成JPA 2.0元模型類 。 在這里,我選擇eclipselink生成器,因為eclipselink最終是SAP NetWeaver Cloud使用的JPA實現。 我相信,任何符合JPA 2.0的生成器也應完成此工作。 同樣,maven在pom.xml的<build> <plugins>部分中提供了以下xml片段,可以為您提供幫助:
<plugin><groupId>org.bsc.maven</groupId><artifactId>maven-processor-plugin</artifactId><version>2.1.0</version><executions><execution><id>process</id><goals><goal>process</goal></goals><phase>generate-sources</phase><configuration><processors><processor>org.eclipse.persistence.internal.jpa.modelgen.CanonicalModelProcessor</processor></processors></configuration></execution></executions>
</plugin>
Maven-averse可以參考有關JPA 2.0 matamodel生成的eclipselink文檔,以獲取生成JPA 2.0元模型類的替代方法。
至此,我們有了一個adventscloud.war文件,該文件應在所有符合Java EE 6 Web Profile的容器上逐字運行。
其中包括SAP NetWeaver Cloud。 您可以在我的SAP NetWeaver Cloud終生免費試用版實例上運行該應用程序。 它比本博客文章中顯示的代碼更豐富,并帶有Twitter引導程序的響聲。 如果您對火花感興趣,請遵循應用程序中的github功能區。
如果你想獲得的SAP NetWeaver云的免費的終身試用實例,請描述的初始步驟在這里 。
參考: Java EE 6 Web配置文件。 在云上。 簡單。 在Java Advent Calendar博客上可以從我們的JCG合作伙伴 Christophe Thiebaud獲得。
翻譯自: https://www.javacodegeeks.com/2012/12/java-ee-6-web-profile-on-the-cloud-easy.html