草庐IT

具有多个类加载器的 Java ServiceLoader

coder 2023-05-15 原文

使用 ServiceLoader 的最佳实践是什么?在具有多个类加载器的环境中?该文档建议在初始化时创建并保存单个服务实例:

private static ServiceLoader<CodecSet> codecSetLoader = ServiceLoader.load(CodecSet.class);

这将使用当前上下文类加载器初始化 ServiceLoader。现在假设这个片段包含在一个使用 Web 容器中的共享类加载器加载的类中,并且多个 Web 应用程序想要定义自己的服务实现。这些不会在上面的代码中得到,甚至有可能使用第一个 webapps 上下文类加载器初始化加载器并向其他用户提供错误的实现。

总是创建一个新的 ServiceLoader 似乎很浪费性能,因为它每次都必须枚举和解析服务文件。 编辑:这甚至可能是一个很大的性能问题,如 this answer regarding java's XPath implementation 所示。 .

其他图书馆如何处理这个问题?他们是否缓存每个类加载器的实现,他们是否每次都重新解析他们的配置,或者他们只是忽略这个问题并且只对一个类加载器有效?

最佳答案

我个人不喜欢ServiceLoader在任何情况下。它很慢而且没有必要的浪费,而且你几乎无能为力来优化它。

我还发现它有点受限制——如果你想做的不仅仅是按类型搜索,你真的必须走自己的路。

xbean-finder 的 ResourceFinder

  • ResourceFinder是一个独立的 java 文件,能够替代 ServiceLoader 的使用。复制/粘贴重用没问题。它是一个 Java 文件,获得了 ASL 2.0 许可,可从 Apache 获得。

  • 在我们的注意力变得太短之前,这里是它如何取代 ServiceLoader
    ResourceFinder finder = new ResourceFinder("META-INF/services/");
    List<Class<? extends Plugin>> impls = finder.findAllImplementations(Plugin.class);
    

    这将找到所有 META-INF/services/org.acme.Plugin类路径中的实现。

    请注意,它实际上并未实例化所有实例。选择你想要的,你就是一个 newInstance()调用远离实例。

    为什么这很好?
  • 打电话有多难newInstance()有适当的异常处理?不难。
  • 拥有仅实例化您想要的实例的自由是很好的。
  • 现在你可以支持构造函数参数了!

  • 缩小搜索范围

    如果您只想检查特定的 URL,您可以轻松完成:
    URL url = new File("some.jar").toURI().toURL();
    ResourceFinder finder = new ResourceFinder("META-INF/services/", url);
    

    在这里,将只搜索“some.jar”这个 ResourceFinder 实例的任何用法。

    还有一个名为 UrlSet 的便利类这可以使从类路径中选择 URL 变得非常容易。
    ClassLoader webAppClassLoader = Thread.currentThread().getContextClassLoader(); 
    UrlSet urlSet = new UrlSet(webAppClassLoader);
    urlSet = urlSet.exclude(webAppClassLoader.getParent());
    urlSet = urlSet.matching(".*acme-.*.jar");
    
    List<URL> urls = urlSet.getUrls();
    

    替代“服务”风格

    假设您想申请 ServiceLoader类型概念来重新设计 URL 处理和查找/加载 java.net.URLStreamHandler对于特定的协议(protocol)。

    以下是在类路径中布局服务的方法:
  • META-INF/java.net.URLStreamHandler/foo
  • META-INF/java.net.URLStreamHandler/bar
  • META-INF/java.net.URLStreamHandler/baz

  • 哪里foo是一个纯文本文件,其中包含与以前一样的服务实现的名称。现在说有人创建了一个 foo://...网址。我们可以通过以下方式快速找到实现:
    ResourceFinder finder = new ResourceFinder("META-INF/");
    Map<String, Class<? extends URLStreamHandler>> handlers = finder.mapAllImplementations(URLStreamHandler.class);
    Class<? extends URLStreamHandler> fooHandler = handlers.get("foo");
    

    替代“服务”样式 2

    假设您想在您的服务文件中放入一些配置信息,因此它包含的不仅仅是一个类名。这是将服务解析为属性文件的替代样式。按照惯例,一个键是类名,其他键是可注入(inject)的属性。

    所以这里red是一个属性文件
  • META-INF/org.acme.Plugin/red
  • META-INF/org.acme.Plugin/blue
  • META-INF/org.acme.Plugin/green

  • 您可以像以前一样查找内容。
    ResourceFinder finder = new ResourceFinder("META-INF/");
    
    Map<String,Properties> plugins = finder.mapAllProperties(Plugin.class.getName());
    Properties redDefinition = plugins.get("red");
    

    以下是如何将这些属性与 xbean-reflect 一起使用,另一个可以为您提供无框架 IoC 的小库。你只需给它类名和一些名称值对,它就会构造和注入(inject)。
    ObjectRecipe recipe = new ObjectRecipe(redDefinition.remove("className").toString());
    recipe.setAllProperties(redDefinition);
    
    Plugin red = (Plugin) recipe.create();
    red.start();
    

    以下是长格式的“拼写”方式:
    ObjectRecipe recipe = new ObjectRecipe("com.example.plugins.RedPlugin");
    recipe.setProperty("myDateField","2011-08-29");
    recipe.setProperty("myIntField","100");
    recipe.setProperty("myBooleanField","true");
    recipe.setProperty("myUrlField","http://www.stackoverflow.com");
    Plugin red = (Plugin) recipe.create();
    red.start();
    
    xbean-reflect library 是内置 JavaBeans API 之外的一步,但在不需要您一直使用完整的 IoC 框架(如 Guice 或 Spring)的情况下要好一些。它支持工厂方法和构造函数参数以及 setter /字段注入(inject)。

    为什么ServiceLoader 如此有限?

    JVM 中弃用的代码会损坏 Java 语言本身。许多东西在被添加到 JVM 之前被修剪到骨骼,因为你不能在之后修剪它们。 ServiceLoader就是一个典型的例子。 API 是有限的,OpenJDK 实现大约有 500 行,包括 javadoc。

    那里没有什么花哨的东西,更换它很容易。如果它对您不起作用,请不要使用它。

    类路径范围

    撇开 API 不谈,在纯粹的实践中,缩小搜索 URL 的范围是解决这个问题的真正方法。应用服务器本身有很多 URL,不包括应用程序中的 jar。例如,OSX 上的 Tomcat 7 仅在 StandardClassLoader 中就有大约 40 个 URL(这是所有 webapp 类加载器的父类)。

    您的应用服务器越大,即使是简单的搜索也需要更长的时间。

    如果您打算搜索多个条目,则缓存无济于事。同样,它会增加一些不好的泄漏。可能是一个真正的双输局面。

    将 URL 缩小到您真正关心的 5 或 12 个,您可以执行各种服务加载而不会注意到命中。

    关于具有多个类加载器的 Java ServiceLoader,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7039467/

    有关具有多个类加载器的 Java ServiceLoader的更多相关文章

    1. ruby-on-rails - Rails 3 中的多个路由文件 - 2

      Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题

    2. ruby - 具有身份验证的私有(private) Ruby Gem 服务器 - 2

      我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..

    3. ruby-on-rails - 在 Ruby 中循环遍历多个数组 - 2

      我有多个ActiveRecord子类Item的实例数组,我需要根据最早的事件循环打印。在这种情况下,我需要打印付款和维护日期,如下所示:ItemAmaintenancerequiredin5daysItemBpaymentrequiredin6daysItemApaymentrequiredin7daysItemBmaintenancerequiredin8days我目前有两个查询,用于查找maintenance和payment项目(非排他性查询),并输出如下内容:paymentrequiredin...maintenancerequiredin...有什么方法可以改善上述(丑陋的)代

    4. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

      我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

    5. ruby - 如何在续集中重新加载表模式? - 2

      鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

    6. ruby - 多个属性的 update_column 方法 - 2

      我有一个具有一些属性的模型:attr1、attr2和attr3。我需要在不执行回调和验证的情况下更新此属性。我找到了update_column方法,但我想同时更新三个属性。我需要这样的东西:update_columns({attr1:val1,attr2:val2,attr3:val3})代替update_column(attr1,val1)update_column(attr2,val2)update_column(attr3,val3) 最佳答案 您可以使用update_columns(attr1:val1,attr2:val2

    7. ruby-on-rails - 在 ruby​​ .gemspec 文件中,如何指定依赖项的多个版本? - 2

      我正在尝试修改当前依赖于定义为activeresource的gem:s.add_dependency"activeresource","~>3.0"为了让gem与Rails4一起工作,我需要扩展依赖关系以与activeresource的版本3或4一起工作。我不想简单地添加以下内容,因为它可能会在以后引起问题:s.add_dependency"activeresource",">=3.0"有没有办法指定可接受版本的列表?~>3.0还是~>4.0? 最佳答案 根据thedocumentation,如果你想要3到4之间的所有版本,你可以这

    8. ruby - RuntimeError(自动加载常量 Apps 多线程时检测到循环依赖 - 2

      我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("

    9. ruby-on-rails - 使用 config.threadsafe 时从 lib/加载模块/类的正确方法是什么!选项? - 2

      我一直致力于让我们的Rails2.3.8应用程序在JRuby下正确运行。一切正常,直到我启用config.threadsafe!以实现JRuby提供的并发性。这导致lib/中的模块和类不再自动加载。使用config.threadsafe!启用:$rubyscript/runner-eproduction'pSim::Sim200Provisioner'/Users/amchale/.rvm/gems/jruby-1.5.1@web-services/gems/activesupport-2.3.8/lib/active_support/dependencies.rb:105:in`co

    10. ruby - 使用多个数组创建计数 - 2

      我正在尝试按0-9和a-z的顺序创建数字和字母列表。我有一组值value_array=['0','1','2','3','4','5','6','7','8','9','a','b','光盘','e','f','g','h','i','j','k','l','m','n','o','p','q','r','s','','u','v','w','x','y','z']和一个组合列表的数组,按顺序,这些数字可以产生x个字符,比方说三个list_array=[]和一个当前字母和数字组合的数组(在将它插入列表数组之前我会把它变成一个字符串,]current_combo['0','0','0']

    随机推荐