Home接口的Weblogic實現類的stub類 ((Hello Bean))_HomeImpl_WLStub(部署的時候動態生成字節碼)
Home接口的Weblogic實現類的skeleton類 ((Hello Bean))_HomeImpl_WLSkeleton(部署的時候動態生成字節碼)
Remote接口:Hello (用戶編寫)
Remote接口的Weblogic實現類 ((Hello Bean))_EOImpl(EJBC生成)
Remote接口的Weblogic實現類的stub類 ((Hello Bean))_EOImpl_WLStub(部署的時候動態生成字節碼)
Remote接口的Weblogic實現類的skeleton類 ((Hello Bean))_EOImpl_WLSkeleton(部署的時候動態生成字節碼)
客戶端只需要Hello.class和HelloHome.class這兩個文件。
((Hello Home)) home = (Home)
((Portable Remote Object)).narrow(ctx.lookup("Hello"),
((Hello Home)).class);
這一行代碼是從JNDI獲得Home接口,但是請記住!接口是抽象的,那麼home這個對象到底是什麼類的對象實例呢?很簡單,用toString()輸出看一下就明白了,下面一行是輸出結果:
((Hello Bean))_HomeImpl_WLStub@18c458
這表明home這個通過從服務器的JNDI樹上查找獲得的對象實際上是HelloBean_HomeImpl_WLStub類的一個實例。
接下來客戶端代碼:
Hello h = home.create()
同樣Hello只是一個抽象的接口,那麼h對象是什麼東西呢?打印一下:
((Hello Bean))_EOImpl_WLStub@8fa0d1
原來是HelloBean_EOImpl_WLStub的一個對象實例。
用這個例子來簡述一遍EJB調用過程:
首先客戶端JNDI查詢,服務端JNDI樹上Hello這個名字實際上綁定的對象是HelloBean_HomeImpl_WLStub,所以服務端將創建HelloBean_HomeImpl_WLStub的一個對象實例,序列化返回給客戶端。
於是客戶端得到home對象,表面上是得到HelloHome接口的實例,實際上是進行了一次遠程調用得到了HelloBean_HomeImpl_WLStub類的對象實例,別忘記了HelloBean_HomeImpl_WLStub也實現了HelloHome接口。
然後home.create()實質上就是HelloBean_HomeImpl_WLStub.create(),該方法將發送信息給HelloBean_HomeImpl_WLSkeleton,而HelloBean_HomeImpl_WLSkeleton接受到信息後,再去調用HelloBean_HomeImpl的create方法,至此完成第1次完整的RMI循環。
注意在這次RMI循環過程中,遠程對象是HelloBean_HomeImpl,遠程對象的接口是HelloHome,對象的stub是HelloBean_HomeImpl_WLStub,對象的skeleton是HelloBean_HomeImpl_WLSkeleton。
然後HelloBean_HomeImpl再去調用HelloBean_Impl的ejbCreate方法,而HelloBean_Impl的ejbCreate方法將負責創建或者分配一個Bean實例,並且創建一個HelloBean_EOImpl_WLStub的對象實例。
這一步比較有趣的是,在前一步RMI循環中,遠程對象HelloBean_HomeImpl在客戶端有一個代理類HelloBean_HomeImpl_WLStub,但在這一步,HelloBean_HomeImpl自己卻充當了HelloBean_Impl的代理類,只不過HelloBean_HomeImpl不在客戶端,而是在服務端,因此不進行RMI。
然後HelloBean_EOImpl_WLStub的對象實例序列化返回給客戶端,這一步也很有趣,上次RMI過程,主角是HelloBean_HomeImpl和它的代理類HelloBean_HomeImpl_WLStub,但這這一次換成了HelloBean_EOImpl和它的代理類HelloBean_EOImpl_WLStub來玩了。
Hello h = home.create();
h.helloWorld();
假設Hello接口有一個helloWorld遠程方法,那麼表面上是在調用Hello接口的helloWorld方法,實際上是在調用HelloBean_EOImpl_WLStub的helloWorld方法。
然後HelloBean_EOImpl_WLStub的helloWorld方法將發送信息給服務器上的HelloBean_EOImpl_WLSkeleton,而HelloBean_EOImpl_WLSkeleton收到信息以後,再去調用HelloBean_EOImpl的helloWorld方法。至此,完成第2次完整的RMI循環過程。
在剛才HelloBean_EOImpl是作為遠程對象被調用的,它的代理類是HelloBean_EOImpl_WLStub,但現在HelloBean_EOImpl要作為HelloBean_Impl的代理類了。現在HelloBean_EOImpl去調用HelloBean_Impl的helloWorld方法。注意!HelloBean_Impl繼承了HelloBean,而HelloBean中的helloWorld方法是我們親自編寫的代碼,現在終於調用到了我們編寫的代碼了!
至此,一次EJB調用過程終於完成。在整個過程中,服務端主要要調用的類是HelloBean_Impl, Hello Bean?_HomeImpl,HelloBean_HomeImpl_WLSkeleton,HelloBean_EOImpl,HelloBean_EOImpl_WLSkeleton。
客戶端主要調用的類是HelloBean_HomeImpl_WLStub,HelloBean_EOImpl_WLStub,這兩個類在客戶端代碼中並不會直接出現,出現在代碼中的類是他們的接口HelloHome和Hello,因此客戶端需要這兩個接口文件,而Stub是服務器傳送給他們的。