2017-05-16 18 views
3

Java EEアプリケーションで、JavaのServiceLoaderを使用してサービスをロードしようとしています。私が読み込もうとしている.jarファイルは、テストJava SEアプリケーションで動作するので、正しいと思います。ただし、以下のコードはitr.nextで失敗:Glassfish4 Java EEアプリケーションでServiceLoaderを使用する場合ClassNotFoundException

ServiceLoader<WorkflowStepSPI> loader = ServiceLoader.load(WorkflowStepSPI.class, Thread.currentThread().getContextClassLoader()); 
     loader.reload(); 
     List<String> names = new LinkedList<>(); 
     Iterator<WorkflowStepSPI> itr = loader.iterator(); 
     while (itr.hasNext()) { 
      WorkflowStepSPI wss = itr.next(); 
      logger.log(Level.INFO, "Found WorkflowStepProvider: {0}", wss.getClass().getCanonicalName()); 
      names.add(wss.toString()); 
     } 

ClassNotFoundExceptionがスローされた例外で、(そうでない場合、コードがコンパイルされないでしょう)が明確である、WorkflowStepSPIを見つけることができない不満。別のテストでは、Class.forNameがこのクラスを返すことを検証しました。

私は自分のURLClassLoaderインスタンスをhereとして作成し、そのクラスをhereとして直接ロードしようとしましたが、役に立たなかったのです。

アイデア?

*更新*要求されたよう

、ここでのスタックトレースです:静的にコンパイルされたクラスは、実行時に利用できない場合(ClassNotFoundExceptionではなく)NoClassDefErrorが発生した一般

Warning: Class not found: edu/harvard/iq/dataverse/workflow/WorkflowStepSPI 
java.lang.NoClassDefFoundError: edu/harvard/iq/dataverse/workflow/WorkflowStepSPI 
at java.lang.ClassLoader.defineClass1(Native Method) 
at java.lang.ClassLoader.defineClass(ClassLoader.java:763) 
at java.security.SecureClassLoader.defineClass(SecureClassLoader.java:142) 
at java.net.URLClassLoader.defineClass(URLClassLoader.java:467) 
at java.net.URLClassLoader.access$100(URLClassLoader.java:73) 
at java.net.URLClassLoader$1.run(URLClassLoader.java:368) 
at java.net.URLClassLoader$1.run(URLClassLoader.java:362) 
at java.security.AccessController.doPrivileged(Native Method) 
at java.net.URLClassLoader.findClass(URLClassLoader.java:361) 
at java.lang.ClassLoader.loadClass(ClassLoader.java:424) 
at java.lang.ClassLoader.loadClass(ClassLoader.java:411) 
at java.lang.ClassLoader.loadClass(ClassLoader.java:357) 
at org.glassfish.web.loader.WebappClassLoader.loadClass(WebappClassLoader.java:1730) 
at org.glassfish.web.loader.WebappClassLoader.loadClass(WebappClassLoader.java:1633) 
at java.lang.Class.forName0(Native Method) 
at java.lang.Class.forName(Class.java:348) 
at java.util.ServiceLoader$LazyIterator.nextService(ServiceLoader.java:370) 
at java.util.ServiceLoader$LazyIterator.next(ServiceLoader.java:404) 
at java.util.ServiceLoader$1.next(ServiceLoader.java:480) 
at edu.harvard.iq.dataverse.api.WorkflowsAdmin.testSpi(WorkflowsAdmin.java:215) 
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) 
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) 
at java.lang.reflect.Method.invoke(Method.java:498) 
at org.glassfish.jersey.server.model.internal.ResourceMethodInvocationHandlerFactory$1.invoke(ResourceMethodInvocationHandlerFactory.java:81) 
at org.glassfish.jersey.server.model.internal.AbstractJavaResourceMethodDispatcher$1.run(AbstractJavaResourceMethodDispatcher.java:151) 
at org.glassfish.jersey.server.model.internal.AbstractJavaResourceMethodDispatcher.invoke(AbstractJavaResourceMethodDispatcher.java:171) 
at org.glassfish.jersey.server.model.internal.JavaResourceMethodDispatcherProvider$ResponseOutInvoker.doDispatch(JavaResourceMethodDispatcherProvider.java:152) 
at org.glassfish.jersey.server.model.internal.AbstractJavaResourceMethodDispatcher.dispatch(AbstractJavaResourceMethodDispatcher.java:104) 
at org.glassfish.jersey.server.model.ResourceMethodInvoker.invoke(ResourceMethodInvoker.java:387) 
at org.glassfish.jersey.server.model.ResourceMethodInvoker.apply(ResourceMethodInvoker.java:331) 
at org.glassfish.jersey.server.model.ResourceMethodInvoker.apply(ResourceMethodInvoker.java:103) 
at org.glassfish.jersey.server.ServerRuntime$1.run(ServerRuntime.java:271) 
at org.glassfish.jersey.internal.Errors$1.call(Errors.java:271) 
at org.glassfish.jersey.internal.Errors$1.call(Errors.java:267) 
at org.glassfish.jersey.internal.Errors.process(Errors.java:315) 
at org.glassfish.jersey.internal.Errors.process(Errors.java:297) 
at org.glassfish.jersey.internal.Errors.process(Errors.java:267) 
at org.glassfish.jersey.process.internal.RequestScope.runInScope(RequestScope.java:297) 
at org.glassfish.jersey.server.ServerRuntime.process(ServerRuntime.java:254) 
at org.glassfish.jersey.server.ApplicationHandler.handle(ApplicationHandler.java:1028) 
at org.glassfish.jersey.servlet.WebComponent.service(WebComponent.java:372) 
at org.glassfish.jersey.servlet.ServletContainer.service(ServletContainer.java:381) 
at org.glassfish.jersey.servlet.ServletContainer.service(ServletContainer.java:344) 
at org.glassfish.jersey.servlet.ServletContainer.service(ServletContainer.java:221) 
at org.apache.catalina.core.StandardWrapper.service(StandardWrapper.java:1682) 
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:344) 
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:214) 
at org.ocpsoft.rewrite.servlet.RewriteFilter.doFilter(RewriteFilter.java:205) 
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:256) 
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:214) 
at edu.harvard.iq.dataverse.api.ApiBlockingFilter$3.doBlock(ApiBlockingFilter.java:65) 
at edu.harvard.iq.dataverse.api.ApiBlockingFilter.doFilter(ApiBlockingFilter.java:157) 
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:256) 
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:214) 
at edu.harvard.iq.dataverse.api.ApiRouter.doFilter(ApiRouter.java:30) 
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:256) 
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:214) 
at org.apache.catalina.core.ApplicationDispatcher.doInvoke(ApplicationDispatcher.java:873) 
at org.apache.catalina.core.ApplicationDispatcher.invoke(ApplicationDispatcher.java:739) 
at org.apache.catalina.core.ApplicationDispatcher.processRequest(ApplicationDispatcher.java:575) 
at org.apache.catalina.core.ApplicationDispatcher.doDispatch(ApplicationDispatcher.java:546) 
at org.apache.catalina.core.ApplicationDispatcher.dispatch(ApplicationDispatcher.java:428) 
at org.apache.catalina.core.ApplicationDispatcher.forward(ApplicationDispatcher.java:378) 
at edu.harvard.iq.dataverse.api.ApiRouter.doFilter(ApiRouter.java:34) 
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:256) 
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:214) 
at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:316) 
at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:160) 
at org.apache.catalina.core.StandardPipeline.doInvoke(StandardPipeline.java:734) 
at org.apache.catalina.core.StandardPipeline.invoke(StandardPipeline.java:673) 
at com.sun.enterprise.web.WebPipeline.invoke(WebPipeline.java:99) 
at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:174) 
at org.apache.catalina.connector.CoyoteAdapter.doService(CoyoteAdapter.java:415) 
at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:282) 
at com.sun.enterprise.v3.services.impl.ContainerMapper$HttpHandlerCallable.call(ContainerMapper.java:459) 
at com.sun.enterprise.v3.services.impl.ContainerMapper.service(ContainerMapper.java:167) 
at org.glassfish.grizzly.http.server.HttpHandler.runService(HttpHandler.java:201) 
at org.glassfish.grizzly.http.server.HttpHandler.doHandle(HttpHandler.java:175) 
at org.glassfish.grizzly.http.server.HttpServerFilter.handleRead(HttpServerFilter.java:235) 
at org.glassfish.grizzly.filterchain.ExecutorResolver$9.execute(ExecutorResolver.java:119) 
at org.glassfish.grizzly.filterchain.DefaultFilterChain.executeFilter(DefaultFilterChain.java:284) 
at org.glassfish.grizzly.filterchain.DefaultFilterChain.executeChainPart(DefaultFilterChain.java:201) 
at org.glassfish.grizzly.filterchain.DefaultFilterChain.execute(DefaultFilterChain.java:133) 
at org.glassfish.grizzly.filterchain.DefaultFilterChain.process(DefaultFilterChain.java:112) 
at org.glassfish.grizzly.ProcessorExecutor.execute(ProcessorExecutor.java:77) 
at org.glassfish.grizzly.nio.transport.TCPNIOTransport.fireIOEvent(TCPNIOTransport.java:561) 
at org.glassfish.grizzly.strategies.AbstractIOStrategy.fireIOEvent(AbstractIOStrategy.java:112) 
at org.glassfish.grizzly.strategies.WorkerThreadIOStrategy.run0(WorkerThreadIOStrategy.java:117) 
at org.glassfish.grizzly.strategies.WorkerThreadIOStrategy.access$100(WorkerThreadIOStrategy.java:56) 
at org.glassfish.grizzly.strategies.WorkerThreadIOStrategy$WorkerThreadRunnable.run(WorkerThreadIOStrategy.java:137) 
at org.glassfish.grizzly.threadpool.AbstractThreadPool$Worker.doWork(AbstractThreadPool.java:565) 
at org.glassfish.grizzly.threadpool.AbstractThreadPool$Worker.run(AbstractThreadPool.java:545) 
at java.lang.Thread.run(Thread.java:745) 
Caused by: java.lang.ClassNotFoundException: edu.harvard.iq.dataverse.workflow.WorkflowStepSPI 
at java.net.URLClassLoader.findClass(URLClassLoader.java:381) 
at java.lang.ClassLoader.loadClass(ClassLoader.java:424) 
at java.lang.ClassLoader.loadClass(ClassLoader.java:357) 
... 94 more 
+0

ここで取得しているスタックトレースはどこですか?スタックトレースにはどのクラスがありませんか?展開構造はどのように見えますか? jarファイルをwarファイルにデプロイしていますか、それともjarファイルですか? githubにあなたのアプリの簡略版を投稿できますか? ServiceLoaderはEEコンテナでうまく動作するため、問題はデプロイメントのより根本的な問題が原因であると見なします。 – yntelectual

+0

プロジェクトは、mavenに基づいた標準の.warファイルです。これはローカルのGlassfish 4.1、Mac OSを搭載したノートパソコンで発生します。 SPI実装の.jarはdomain/libフォルダにあります。 –

答えて

0

、 (ClassNotFoundExceptionは、リフレクションによってクラスを取得しようとすると発生します(例えば、Class.forName()を使用している場合など)。

あなたのコードをコンパイルすると、そのクラスはクラスパスに存在していました。ただし、さまざまな理由(構成を見ていないと判断することは不可能です)では、クラスはサーバー上で使用できません。可能な理由は次のとおりです。

  • これは、プロジェクトをクロスコンパイルしたプロジェクト(.classファイルを生成したもの)であり、他のプロジェクトはWARには含まれていませんその内容はWEB-INF/classesにコピーされなかった)
  • それはいくつかの理由のためWARと一緒に出荷されていなかったライブラリに含まれていた別の出力ディレクトリ
  • あなたのWARが不正であると(クラスファイルを適切な構造を含んでいないはずWEB-INF/classesに入る、図書館はWEB-INF/libに入る)
  • あなたは一般的に

を使用すると、コードのコンパイルとは異なるJavaの分布にサーバーを実行している(これは、クラスのパッケージには、ここで明確でない場合に、ちょうど完全性のためにそれを指摘したかった)されていますサーバーと一緒にバンドルされるアプリケーション固有のWAR(一部のサーバー固有のフォルダー)を使用するのは悪い習慣です。さらに、アプリケーションサーバー上のいくつかのクラスローダー分離の問題のため、微妙なエラーが発生します。代わりに、あなたのWARには、必要なjarがパッケージされている必要があります。

0

@ piotr-wilkinの練習への優れた答えをここに示します。

時間を節約するためにポイント

を開始(皮肉!)私は、SPI実装プロジェクトの依存関係として、Webアプリケーションの.warファイルを使用していました。これにより、SPIの実装でメインプロジェクトのクラスを使用することができました。しかし、明らかに、Glassfishのクラスローダーは混乱していました(Java SEのクラスローダーはこれでうまくいきました)。

  1. 修正は、私はSPIに実装する必要があるインターフェイスを持つ新しい.jarファイルを作成しました。このjarファイルは、WebアプリケーションとSPI実装プロジェクトの両方に依存して追加されました。

  2. 通常通りSPIの.jarをパックしました(META-INF/services/.....)。
  3. SPI実装の.jarとインターフェイスの.jarの両方をドメインのlibフォルダに追加しました。
  4. リスタートGlassfishの

今、私は私があまりにも程度満足していないインタフェースの2つのコピーを持っていることを実現します。しかし、それは単なるインタフェースであり、静的なフィールドには状態を含んでいないので、それは審美的な問題であると思います。

関連する問題