第一章補充
我看了自己寫第一章后,對于build.xml沒有顯示全面,現(xiàn)在補充如下:
<!-- Simple Ant build script to test an Ant installation -->
<project name="TestInstall" default="run" basedir=".">
<target name="init">
<available file="ASimpleHelloObject.java" property="ASimpleHelloObject"/>
</target>
<target name="ASimpleHelloObject" unless="ASimpleHelloObject" depends="init">
<echo file="ASimpleHelloObject.java">
public class ASimpleHelloObject
{
public static void main(String[] args)
{
System.out.println("ASimpleHelloObject.main was called");
}
}
</echo>
<echo message="Wrote ASimpleHelloObject.java" />
</target>
<target name="compile" depends="ASimpleHelloObject">
<javac destdir="." srcdir="." debug="on" classpath=".">
<include name="ASimpleHelloObject.java"/>
</javac>
</target>
<target name="run" depends="compile">
<java classname="ASimpleHelloObject" classpath="." />
<echo message="Ant appears to be successfully installed" />
</target>
</project>
1、下載例子源程序
所有例子的源代碼,都在文件documentation-example.zip(windows平臺) or
documentation-example.tar.gz (Unix/Lunix平臺)。你可以直接從
www.jboss.org進(jìn)行下載。下載完后放在一個目錄下。下載網(wǎng)址:http:
//www.jboss.org/docs/manual/files/documentation-example.zip
1.1 建立 BEAN
此節(jié)主要是建立一個簡單的EJB,可以查看代碼,這個“Interest”例子,是一個簡單無狀態(tài)的會話EJB。它的目的是根據(jù)說明的利息率,來對借的所有錢計算利息。實際上在整個包代碼中只有一行功能。
1.2 回顧EJBs
在我們查看代碼之前,我們先對EJB進(jìn)行復(fù)習(xí)一下。在EJB最小類型,也必須有三個類:remote interface, home interface和bean實現(xiàn)類。
remote interface是會把EJB中方法提供給外邊世界,讓外邊的代碼來進(jìn)行調(diào)用,在這個例子中類名稱是org.jboss.interest.Interrest。
home interface是管理remote interface類的類。包括建立、刪除等操作。在這個例子中類名稱是org.jboss.interest.InterrestHome。
bean實現(xiàn)類提供home interface和remote interface所有方法的實現(xiàn)。在這個例子中類名稱是org.jboss.interest.InterrestBean。
當(dāng)然一個Bean可能還包括其他類,甚至其他包。但是必須有此三個類,其他類是在此三個類之上建立的。所有類被打包進(jìn)一個JAR文件,此文件是用一個目錄
結(jié)構(gòu)來反映出包的層次關(guān)系。在此例子中所有類都打包在org.jboss.interest包中,所以他們需要在目錄
org/jboss/interest/下。
在包含所有類的jar文件建立之前,必須有一個META-INF目錄。此目錄存放了部署描述符(通常叫“ejb-jar.xml”),和可選的其他XML文件。這些文件告訴服務(wù)器關(guān)于應(yīng)用明確服務(wù)信息。對于JBoss 來講,文件名必須叫“jboss.xml”。
創(chuàng)建jar文件后部署到JBoss
Server上。在客戶端你需要一個jndi.properties文件,此文件告訴你的客戶端程序從哪里初始化查找JNDI
命名服務(wù)。從這個服務(wù),客戶端將查找到Interest bean,并且返回bean的home interface。home
interface用來得到bean的一個remote interface。它可以用遠(yuǎn)程接口來訪問bean提供的商業(yè)方法。
1.3 EJB類
我們需要三個類:remote interface, home interface 和bean實現(xiàn)類。remote interface遠(yuǎn)程接口類,文件名Interest.java。代碼如下:
package org.jboss.docs.interest;
import javax.ejb.EJBObject;
import java.rmi.RemoteException;
/**
This interface defines the `Remote' interface for the `Interest' EJB. Its
single method is the only method exposed to the outside world. The class
InterestBean implements the method.
*/
public interface Interest extends EJBObject
{
/** Calulates the compound interest on the sum `principle', with
interest rate per period `rate' over `periods' time periods. This
method also prints a message to standard output; this is picked up by
the EJB server and logged. In this way we can demonstrate that the
method is actually being executed on the server, rather than the
client. */
public double calculateCompoundInterest(double
principle, double rate, double periods) throws RemoteException;
}
遠(yuǎn)程接口只有一個商業(yè)方法calculateCompoundInterest。Home interface 文件名InterestHome.java。代碼如下:
package org.jboss.docs.interest;
import java.io.Serializable;
import java.rmi.RemoteException;
import javax.ejb.CreateException;
import javax.ejb.EJBHome;
/**
This interface defines the 'home' interface for the 'Interest' EJB.
*/
public interface InterestHome extends EJBHome
{
/** Creates an instance of the `InterestBean' class on the server, and
returns a remote reference to an Interest interface on the client. */
Interest create() throws RemoteException, CreateException;
}
最后我們給出bean實現(xiàn)類,文件名稱:InterestBean.java。代碼如下:
package org.jboss.docs.interest;
import java.rmi.RemoteException;
import javax.ejb.SessionBean;
import javax.ejb.SessionContext;
/**
This
class contains the implementation for the
'calculateCompoundInterest' method exposed by this Bean. It includes
empty method bodies for the methods prescribe by the SessionBean
interface; these don't need to do anything in this simple
example. */public class InterestBean implements SessionBean
{
/** Calulates the compound interest on the sum `principle', with
interest rate per period `rate' over `periods' time periods. This
method also prints a message to standard output; this is picked up by
the EJB server and logged. In this way we can demonstrate that the
method is actually being executed on the server, rather than the
client. */
public double calculateCompoundInterest(double principle, double rate, double periods)
{
System.out.println("Someone called `calculateCompoundInterest!'");
return principle * Math.pow(1+rate, periods) - principle;
}
/** Empty method body */
public void ejbCreate() {}
/** Every ejbCreate() method ALWAYS needs a corresponding
ejbPostCreate() method with exactly the same parameter types. */
public void ejbPostCreate() {}
/** Empty method body */
public void ejbRemove() {} /** Empty method body */
public void ejbActivate() {} /** Empty method body */
public void ejbPassivate() {} /** Empty method body */
public void setSessionContext(SessionContext sc) {}
}
注意大部分方法是空的。因為這些方法在SessionBean接口中說明,所以必須在InterestBean中存在,但在這個簡單例子中,不需要具體內(nèi)容。
2、部署描述符
當(dāng)你編輯完類文件后,我們來看部署描述符。此文件告訴EJB Server是哪個類應(yīng)該被喚醒bean、Home Interface 和remote Interface。如果一個包中有不止一個bean,它指明ejb同另外bean如何相和。在這
個簡單例子中只有一個ejb,所以我們不用關(guān)心這方面問題。
大部分商業(yè)EJB Server都提供圖形化部署工具,來構(gòu)造部署描述符,JBoss沒有XML編輯器,但是它能夠簡單手工構(gòu)造部署描述符。下面是Interest Bean部署描述符:
<?xml version="1.0" encoding="UTF-8"?>
<ejb-jar>
<description>JBoss Interest Sample Application</description>
<display-name>Interest EJB</display-name>
<enterprise-beans>
<session>
<ejb-name>Interest</ejb-name>
<home>org.jboss.docs.interest.InterestHome</home>
<remote>org.jboss.docs.interest.Interest</remote>
<ejb-class>org.jboss.docs.interest.InterestBean</ejb-class>
<session-type>Stateless</session-type>
<transaction-type>Bean</transaction-type>
</session>
</enterprise-beans>
</ejb-jar>
部署描述符文件必須命名為“ejb-jar.xml”,并且必須放在目錄META-INF下。一個大家容易犯的錯誤是目錄名字,有的寫成“META_INF”, “META~INF”或“meta-inf”所有這些都不能正常的工作。
我們在server上部署一個應(yīng)用程序,而不是一個ejb。在這個例子中我們的應(yīng)用程序只有一個ejb。在部署描述符中<ejb-name>
Interest</ejb-name>表明ejb的名稱。JBoss缺省是把bean的home
interface作為JNDI命名空間放在<ejb-name>中,除非你修改缺省值。實際上客戶端應(yīng)用程序不是必須使用這個名字的。開發(fā)
者不用為JNDI命名空間指明一個不同的名字而費心。然而一個由完整應(yīng)用程序的產(chǎn)品包含多個beans,通常用一個和開發(fā)者說明不同的名字。所以會發(fā)現(xiàn)
“[application name]/[bean name]”等,關(guān)于這部分我們以后介紹。
盡管部署描述符格式的文件ejb-jar.xml對所有ejb Server都是共有的。你可以從sun得到一個正確定義的DTD,
但是沒有為一個特殊EJB
Server指明每個必須要求。也沒有說明如何從ejb-name影射到部署的JNDI名字,例如“[application name]/[bean
name]”。通過JBoss提供一個缺省行為,來進(jìn)行工作。這些工作來自于ejb-jar.xml。在高級配置的案例中,你必須定義JBoss說明行
為, 此行為用jboss.xml部署描述符。在高級配置中我們會談到j(luò)boss.xml描述符的詳細(xì)信息的。
這里我們僅配置jboss.xml描述符的JNDI名字來用來訪問Interest EJB home
inrterface。通過缺省的JNDI名字用來訪問EJB home inrterface,
需要同ejb-jar.xml的ejb-name相同。對于這個簡單的例子,Interest
bean通過JNDI初始化上下文(作為“Interest”)來定位。 我們想用“interest/Interest”使home
interface
可以使用,所以我們必須用jboss.xml描述符來說明。在jboss.xml中,重新覆蓋缺省的JNDI命名。為了重新編寫缺省的行為,我們用ejb
-jar.xml中的ejb-name元素值作為bean home接口的JNDI命名,必須說明jndi命名,來寫jboss.xml描述符如下:
<?xml version="1.0" encoding="UTF-8"?>
<jboss>
<enterprise-beans>
<session>
<ejb-name>Interest</ejb-name>
<jndi-name>interest/Interest</jndi-name>
</session>
</enterprise-beans>
</jboss>
這個文件說明調(diào)用的Interest
BEAN被綁定在interest/Interest的JNDI名稱下。我們建立的標(biāo)準(zhǔn)ejb-jar.xml部署描述符同JBoss說明
jboss.xml共同設(shè)置Interest EJB
home接口的JNDI名稱為“interest/Interest”。我們現(xiàn)在有了EJB類,這些類是建立ejb 文件包必須的文件。
3、打包和部署B(yǎng)EAN
jar包是建立一個JAR文件,該文件包含EJB類文件和部署描述符。在下載的例子可能和你設(shè)置環(huán)
境不同,所以需要修改examplesuilduild.xml的內(nèi)容。關(guān)于各個ant方面的內(nèi)容,我會在Ant文檔中介紹給大家。關(guān)于如何修改,下面我一步步地來告訴大家。以下是windows平臺的。至于Unix/Lunix平臺基本相似,這里就不在敘述了。
1) 在修改之前,比首先設(shè)定JBOSS_DIST環(huán)境變量,指定到你安裝的Jboss的目錄。我安裝在C:jboss-3.0.6_tomcat-4.1.18, 所以 JBOSS_DIST設(shè)定為C:jboss-3.0.6_tomcat-4.1.18。
2)
你是否安裝了tomcat或jetty等web服務(wù)器。如果你沒有安裝,建議你安裝jboss-3.0.6_tomcat-4.1.18,
JBoss和tomcat直接結(jié)合在一起啦,不需要再單獨安裝,當(dāng)然也可以安裝jboss-jetty(
JBoss和jetty直接結(jié)合在一起)。取決你喜歡什么web服務(wù)器。
3) 完成上面兩步后,我們來討論修改examplesuilduild.xml文件。在原有的文件中的部分代碼:
<target name="validate-servlet">
<!-- Override with your web server servlet jar location.
The default assumes that JBOSS_DIST points to a
JBoss/Tomcat bundle distribution
-->
<available property="servlet.jar"
value="${env.JBOSS_DIST}/../tomcat/lib/servlet.jar"
file="${env.JBOSS_DIST}/../tomcat/lib/servlet.jar"/>
<available property="servlet.jar"
value="${env.JBOSS_DIST}/../jetty/lib/javax.servlet.jar"
file="${env.JBOSS_DIST}/../jetty/lib/javax.servlet.jar"/>
<available property="servlet.jar"
value="${env.JBOSS_DIST}/../catalina/common/lib/servlet.jar"
file="${env.JBOSS_DIST}/../catalina/common/lib/servlet.jar"/>
<available property="servlet.jar"
value="${env.JBOSS_DIST}/../catalina/common/lib/servlet.jar"
file="${env.TOMCAT_HOME}/lib/servlet.jar"/>
<property name="servlet.jar" value="COULD_NOT_FIND_SERVLET_JAR"/>
<path id="base.path_22">
<pathelement location="${jboss.dist}/client/ejb.jar"/>
<pathelement location="${jboss.dist}/client/jaas.jar"/>
<pathelement location="${jboss.dist}/client/jbosssx-client.jar"/>
<pathelement location="${jboss.dist}/client/jboss-client.jar"/>
<pathelement location="${jboss.dist}/client/jnp-client.jar"/>
<pathelement location="${servlet.jar}"/>
</path>
<path id="base.path_24">
<pathelement location="${jboss.dist}/client/jboss-j2ee.jar"/>
<pathelement location="${jboss.dist}/client/jaas.jar"/>
<pathelement location="${jboss.dist}/client/jbosssx-client.jar"/>
<pathelement location="${jboss.dist}/client/jboss-client.jar"/>
<pathelement location="${jboss.dist}/client/jnp-client.jar"/>
<pathelement location="${servlet.jar}"/>
</path>
</target>
<target name="validate-jboss" depends="validate-servlet">
<available property="classpath_id" value="base.path_22" file="${jboss.dist}/client/ejb.jar" />
<available property="classpath_id" value="base.path_24" file="${jboss.dist}/client/jboss-j2ee.jar" />
</target>
由于此代碼例子是jboss2.2中的所以對于我們不能適應(yīng),在此基礎(chǔ)上我們來進(jìn)行改動。在改動之前,來說明一下為什么改動,讓大家清楚改動原因,不是一團霧水。
<available property="servlet.jar"
value="${env.JBOSS_DIST}/../tomcat/lib/servlet.jar"
file="${env.JBOSS_DIST}/../tomcat/lib/servlet.jar"/>….
此部分說明你運用的web服務(wù)器是什么,根據(jù)你使用的web服務(wù)器來設(shè)置servlet.jar。
由于我使用的是jboss-tomcat.。
<path id="base.path_24"> … </path>說明客戶端要的CLASSPATH包。
修改以后的部分文件:
<target name="validate-servlet">
<!-- Override with your web server servlet jar location.
The default assumes that JBOSS_DIST points to a
JBoss/Tomcat bundle distribution
-->
<available property="servlet.jar"
value="${env.JBOSS_DIST}/tomcat-4.1.x/common/lib/servlet.jar"
file="${env.JBOSS_DIST}/tomcat-4.1.x/common/lib/servlet.jar"/>
<available property="servlet.jar"
value="${env.JBOSS_DIST}/../jetty/lib/javax.servlet.jar"
file="${env.JBOSS_DIST}/../jetty/lib/javax.servlet.jar"/>
<available property="servlet.jar"
value="${env.JBOSS_DIST}/../catalina/common/lib/servlet.jar"
file="${env.JBOSS_DIST}/../catalina/common/lib/servlet.jar"/>
<available property="servlet.jar"
value="${env.JBOSS_DIST}/../catalina/common/lib/servlet.jar"
file="${env.TOMCAT_HOME}/lib/servlet.jar"/>
<property name="servlet.jar" value="COULD_NOT_FIND_SERVLET_JAR"/>
<path id="base.path_22">
<pathelement location="${jboss.dist}/client/ejb.jar"/>
<pathelement location="${jboss.dist}/client/jaas.jar"/>
<pathelement location="${jboss.dist}/client/jbosssx-client.jar"/>
<pathelement location="${jboss.dist}/client/jboss-client.jar"/>
<pathelement location="${jboss.dist}/client/jnp-client.jar"/>
<pathelement location="${servlet.jar}"/>
</path>
<path id="base.path_306">
<pathelement location="${jboss.dist}/client/jboss-j2ee.jar"/>
<pathelement location="${jboss.dist}/client/jboss-jaas.jar"/>
<pathelement location="${jboss.dist}/client/jbosssx-client.jar"/>
<pathelement location="${jboss.dist}/client/jboss-client.jar"/>
<pathelement location="${jboss.dist}/client/jnp-client.jar"/>
<pathelement location="${jboss.dist}/client/jbossall-client.jar"/>
<pathelement location="${jboss.dist}/client/log4j.jar"/>
<pathelement location="${servlet.jar}"/>
</path>
</target>
<target name="validate-jboss" depends="validate-servlet">
<available property="classpath_id" value="base.path_22" file="${jboss.dist}/client/ejb.jar" />
<available property="classpath_id" value="base.path_306" file="${jboss.dist}/client/jboss-j2ee.jar" />
</target>
4)修改完后存盤。然后到exaplesuild目錄下運行以下命令:
ant intro-interest-jar
如果成功會提示 : BUILD SUCCESSFUL 。
在編譯EJB 類文件,建立ejb-jar文件時,如果有一個java.lang.NoClassDefFoundError異常
產(chǎn)生,你需要檢查你的classpath。修改<path id="base.path_306">中的內(nèi)容。
5)完成后你可以查看interest.jar文件中內(nèi)容:
jar ?tvf interest.jar為了把bean部署到server上,必須拷貝interest.jar文件到JBOSS_DIST/server/default/deploy目錄下。服務(wù)器自己會發(fā)現(xiàn)發(fā)生變化的文件,并重新部署它。
6)當(dāng)JBoss server運行時,通過在examlpesuild目錄下運行ant intro-interest-deploy來部署jar。在你部署期間,你會看到在server控制臺的一些信息。
7)如果你發(fā)現(xiàn)部署失敗的信息,通常是部署描述符ejb-jar.xml文件結(jié)構(gòu)壞了或丟失或在錯誤的目錄下。
8)如果在Server上部署完后,我們來編寫一個簡單客戶端來測試它是否正常工作。
4、 編碼和編譯客戶端
單
獨一個ejb是不能用,我們至少需要一個簡單客戶端來用它的服務(wù)。一個EJB可以被其他的EJB,普通的JavaBean,一個JSP
page,一個applet程序或是獨立的應(yīng)用程序調(diào)用。在這個例子中。我們編寫一個簡單的應(yīng)用程序。這個應(yīng)用程序?qū)nterest類的一個對
象。并且執(zhí)行它的方法。當(dāng)部署完bean后,server會綁定EJB home
接口到interest/Interest的JNDI名字下,并輸出此home 接口,能通過RMI被調(diào)用。客戶端代碼如下:
package org.jboss.docs.interest;
import javax.naming.InitialContext;
import javax.naming.Context;
import javax.rmi.PortableRemoteObject;
import org.jboss.docs.interest.Interest;
import org.jboss.docs.interest.InterestHome;
/** This simple application tests the 'Interest' Enterprise JavaBean which is
implemented in the package 'org.jboss.docs.interest'. For this to work, the
Bean must be deployed on an EJB server.
*/
public class InterestClient
{
/** This method does all the work. It creates an instance of the Interest EJB on
the EJB server, and calls its `calculateCompoundInterest()' method, then prints
the result of the calculation.
*/
public static void main(String[] args)
{
// Enclosing the whole process in a single `try' block is not an ideal way
// to do exception handling, but I don't want to clutter the program up
// with catch blocks
try
{
// Get a naming context
InitialContext jndiContext = new InitialContext();
System.out.println("Got context");
// Get a reference to the Interest Bean
Object ref = jndiContext.lookup("interest/Interest");
System.out.println("Got reference");
// Get a reference from this to the Bean's Home interface
InterestHome home = (InterestHome)
PortableRemoteObject.narrow(ref, InterestHome.class);
// Create an Interest object from the Home interface
Interest interest = home.create();
// call the calculateCompoundInterest() method to do the calculation
System.out.println("Interest on 1000 units, at 10% per period, compounded over 2 periods is:");
System.out.println(interest.calculateCompoundInterest(1000, 0.10, 2));
}
catch(Exception e)
{
System.out.println(e.toString());
}
}
}
客戶端在JNDI命名“interest/Interest”下查找InterestHome接口。如果你提供了一個正確的jboss.xml文件,
Bean home接口將會綁定到這個名字。否則,JNDI的名字是“Interest”。為了連接JBoss
JNDI,必須建立一個初始化上下文,該上下文是根據(jù)在客戶端的classpath路徑下jndi.properties來的。這個
jndi.property文件我們放在了examples
esourcesjndi.properties中。代碼如下:
java.naming.factory.initial=org.jnp.interfaces.NamingContextFactory
java.naming.provider.url=localhost:1099
java.naming.factory.url.pkgs=org.jboss.naming:org.jnp.interfaces
指定了InitialContextFactory(初始化上下文工廠),url,和工廠對象的包。在這里url是“localhost”,或運行
JBoss的主機名(IP),用的缺省端口是1099。為了不同RMI協(xié)議互相兼容,通過“narrow”方法來確保“ref”對象能轉(zhuǎn)換為
“InterestHome”對象。客戶端不需要必須和EJB類在同一個包中。所以它必須import ejb claess類。
為了能運行Ant命令。我修改了examplesorgjbossdocsinterestuild.xml文件,原文件的部分內(nèi)容:
<target name="deploy-ejb-jar" depends="ejb-jar">
<copy file="${build.interest.dir}/interest.jar" todir="${jboss.dist}/deploy" />
</target>
<target name="deploy-ear" depends="ear">
<copy file="${build.interest.dir}/interest.ear" todir="${jboss.dist}/deploy" />
</target>
修改后為:
<target name="deploy-ejb-jar" depends="ejb-jar">
<copy file="${build.interest.dir}/interest.jar" todir="${jboss.dist}/server/default/deploy" />
</target>
<target name="deploy-ear" depends="ear">
<copy file="${build.interest.dir}/interest.ear" todir="${jboss.dist}/server/default/deploy" />
</target>
修改完后保存。編譯和運行InterestClient,通過以下命令:
ant intro-interest-client會出現(xiàn)正確的結(jié)果。
5、補充說明
各個jar文件:
1) jboss-j2ee.jar :是標(biāo)準(zhǔn)的javax.ejb.* 包括EJB2.0接口和類。
2) jboss-jaas.jar :是java鑒定和授權(quán)服務(wù)安全類。
3) jbosssx-client.jar :是JbossSX安全擴展客戶端類。
4) jboss-client.jar :是Jboss EJB容器代理和stub客戶端類。
5) jnp-client.jar :是JBoss JNDI提供者客戶端類。
現(xiàn)在隱藏代碼,編譯和部署EJB, 運行簡單客戶端進(jìn)行測試。
6、 結(jié)論
最后來進(jìn)行一個總結(jié)。
Server:
為每個EJB(包括有、無狀態(tài)的Session,或?qū)嶓wBean),都需要三個文件: bean實現(xiàn)類,
remote接口類,home接口類。這些類和一個或多個部署描述符(ejb-jar.xml,
jboss.xml等),這些描述符必須在META-INF目錄下,一起打包進(jìn)jar文件。為部署這個jar文件,你只需要拷貝到
jbossserverdefaultdeploy下。
Client:
1)從JNDI服務(wù),查找bean的home 接口;
2)從home接口,建立一個新的bean或得到一個已經(jīng)存在bean;
3)用得到的remote接口來訪問服務(wù)器上bean實現(xiàn)類中商業(yè)方法。
在客戶端查找home 接口之前,客戶端必須知道如何從JNDI來定位。因此我們提供了一個jndi.properties文件。這個文件和一些jbossclientx下的jar,必須包含到CLASSPATH中,才能使客戶端運行。