diff --git a/jdk/src/java.base/share/classes/java/lang/module/package-info.java b/jdk/src/java.base/share/classes/java/lang/module/package-info.java index 902019b8ce4..74021cd0bef 100644 --- a/jdk/src/java.base/share/classes/java/lang/module/package-info.java +++ b/jdk/src/java.base/share/classes/java/lang/module/package-info.java @@ -159,21 +159,28 @@ * *

Observable modules

* - *

The set of observable modules at both compile-time and run-time is determined - * by searching an abstract module path. The module path is typically composed - * of search paths that are searched in order:

+ *

The set of observable modules at both compile-time and run-time is + * determined by searching several different paths, and also by searching + * the compiled modules built in to the environment. The search order is as + * follows:

* - * + *
    + *
  1. At compile time only, the compilation module path. This path + * contains module definitions in source form.

  2. + * + *
  3. The upgrade module path. This path contains compiled definitions of + * modules that will be observed in preference to the compiled definitions of + * any upgradeable modules that are present in (3) and (4). See the Java + * SE Platform for the designation of which standard modules are upgradeable. + *

  4. + * + *
  5. The system modules, which are the compiled definitions built in to + * the environment.

  6. + * + *
  7. The application module path. This path contains compiled definitions + * of library and application modules.

  8. + * + *
* *

'requires' directives with 'static' modifier

* diff --git a/jdk/src/java.base/share/classes/java/util/ServiceLoader.java b/jdk/src/java.base/share/classes/java/util/ServiceLoader.java index 58961a0661d..72081980298 100644 --- a/jdk/src/java.base/share/classes/java/util/ServiceLoader.java +++ b/jdk/src/java.base/share/classes/java/util/ServiceLoader.java @@ -57,215 +57,290 @@ import jdk.internal.reflect.Reflection; /** - * A simple service-provider loading facility. + * A facility to load implementations of a service. * - *

A service is a well-known set of interfaces and (usually - * abstract) classes. A service provider is a specific implementation - * of a service. The classes in a provider typically implement the interfaces - * and subclass the classes defined in the service itself. - * Providers may be developed and deployed as modules and made available using - * the application module path. Providers may alternatively be packaged as JAR - * files and made available by adding them to the application class path. The - * advantage of developing a provider as a module is that the provider can be - * fully encapsulated to hide all details of its implementation. + *

A service is a well-known interface or class for which zero, one, + * or many service providers exist. A service provider (or just + * provider) is a class that implements or subclasses the well-known + * interface or class. A {@code ServiceLoader} is an object that locates and + * loads service providers deployed in the run time environment at a time of an + * application's choosing. Application code refers only to the service, not to + * service providers, and is assumed to be capable of differentiating between + * multiple service providers as well as handling the possibility that no service + * providers are located. * - *

For the purpose of loading, a service is represented by a single type, - * that is, a single interface or abstract class. (A concrete class can be - * used, but this is not recommended.) A provider of a given service contains - * one or more concrete classes that extend this service type with data - * and code specific to the provider. The provider class is typically - * not the entire provider itself but rather a proxy which contains enough - * information to decide whether the provider is able to satisfy a particular - * request together with code that can create the actual provider on demand. - * The details of provider classes tend to be highly service-specific; no - * single class or interface could possibly unify them, so no such type is - * defined here. + *

Obtaining a service loader

* - *

A service loader is created by invoking one of the static {@code load} - * methods that {@code ServiceLoader} defines. The resulting service loader - * can be used to locate and instantiate service provider implementations by - * means of its {@link #iterator() iterator} ({@code ServiceLoader} implements - * {@code Iterable}) or by consuming elements from its {@link #stream() stream}. + *

An application obtains a service loader for a given service by invoking + * one of the static {@code load} methods of ServiceLoader. If the application + * is a module, then its module declaration must have a uses directive + * that specifies the service; this helps to locate providers and ensure they + * will execute reliably. In addition, if the service is not in the application + * module, then the module declaration must have a requires directive + * that specifies the module which exports the service. + * + *

A service loader can be used to locate and instantiate providers of the + * service by means of the {@link #iterator() iterator} method. {@code ServiceLoader} + * also defines the {@link #stream() stream} method to obtain a stream of providers + * that can be inspected and filtered without instantiating them. + * + *

As an example, suppose the service is {@code com.example.CodecFactory}, an + * interface that defines methods for producing encoders and decoders: * - *

As an example, suppose the service type is {@code com.example.CodecSet} - * and it defines two abstract methods to obtain encoders and decoders: *

{@code
  *     package com.example;
- *     public interface CodecSet {
+ *     public interface CodecFactory {
  *         Encoder getEncoder(String encodingName);
  *         Decoder getDecoder(String encodingName);
  *     }
  * }
- * With this example, the following uses the service loader's iterator to find - * a provider that supports a specific encoding: + * + *

The following code obtains a service loader for the {@code CodecFactory} + * service, then uses its iterator (created automatically by the enhanced-for + * loop) to yield instances of the service providers that are located: + * *

{@code
- *     public Encoder getEncoder(String encodingName) {
- *         ServiceLoader loader = ServiceLoader.load(CodeSet.class);
- *         for (CodecSet cs : loader) {
- *             Encoder encoder = cs.getEncoder(encodingName);
- *             if (encoder != null)
- *                 return encoder;
+ *     ServiceLoader loader = ServiceLoader.load(CodecFactory.class);
+ *     for (CodecFactory factory : loader) {
+ *         Encoder enc = factory.getEncoder("PNG");
+ *         if (enc != null)
+ *             ... use enc to encode a PNG file
+ *             break;
  *         }
- *         return null;
- *    }
  * }
* - *

Selecting a provider or filtering providers will usually involve invoking - * a provider method. In the {@code CodeSet} example, the {@code getEncoder} - * method is used to select the implementation. Where selection or filtering based - * on the provider class is needed then it can be done when consuming the elements - * of the service loader's stream. As an example, the following collects the - * {@code CodeSet} implementations that have a specific annotation: + *

If this code resides in a module, then in order to refer to the + * {@code com.example.CodecFactory} interface, the module declaration would + * require the module which exports the interface. The module declaration would + * also specify use of {@code com.example.CodecFactory}: *

{@code
- *     Set providers = ServiceLoader.load(CodecSet.class)
- *            .stream()
- *            .filter(p -> p.type().isAnnotationPresent(Managed.class))
- *            .map(Provider::get)
+ *     requires com.example.codec.core;
+ *     uses com.example.CodecFactory;
+ * }
+ * + *

Sometimes an application may wish to inspect a service provider before + * instantiating it, in order to determine if an instance of that service + * provider would be useful. For example, a service provider for {@code + * CodecFactory} that is capable of producing a "PNG" encoder may be annotated + * with {@code @PNG}. The following code uses service loader's {@code stream} + * method to yield instances of {@code Provider} in contrast to + * how the iterator yields instances of {@code CodecFactory}: + *

{@code
+ *     ServiceLoader loader = ServiceLoader.load(CodecFactory.class);
+ *     Set pngFactories = loader
+ *            .stream()                                              // Note a below
+ *            .filter(p -> p.type().isAnnotationPresent(PNG.class))  // Note b
+ *            .map(Provider::get)                                    // Note c
  *            .collect(Collectors.toSet());
  * }
+ *
    + *
  1. A stream of {@code Provider} objects
  2. + *
  3. {@code p.type()} yields a {@code Class}
  4. + *
  5. {@code get()} yields an instance of {@code CodecFactory}
  6. + *
* - *

Providers are located and instantiated lazily, that is, on demand. A - * service loader maintains a cache of the providers that have been loaded so - * far. Each invocation of the {@code iterator} method returns an iterator that - * first yields all of the elements cached from previous iteration, in + *

Designing services

+ * + *

A service is a single type, usually an interface or abstract class. A + * concrete class can be used, but this is not recommended. The type may have + * any accessibility. The methods of a service are highly domain-specific, so + * this API specification cannot give concrete advice about their form or + * function. However, there are two general guidelines: + *

    + *
  1. A service should declare as many methods as needed to allow service + * providers to communicate their domain-specific properties and other + * quality-of-implementation factors. An application which obtains a service + * loader for the service may then invoke these methods on each instance of + * a service provider, in order to choose the best provider for the + * application.

  2. + *
  3. A service should express whether its service providers are intended + * to be direct implementations of the service or to be an indirection + * mechanism such as a "proxy" or a "factory". Service providers tend to be + * indirection mechanisms when domain-specific objects are relatively + * expensive to instantiate; in this case, the service should be designed + * so that service providers are abstractions which create the "real" + * implementation on demand. For example, the {@code CodecFactory} service + * expresses through its name that its service providers are factories + * for codecs, rather than codecs themselves, because it may be expensive + * or complicated to produce certain codecs.

  4. + *
+ * + *

Developing service providers

+ * + *

A service provider is a single type, usually a concrete class. An + * interface or abstract class is permitted because it may declare a static + * provider method, discussed later. The type must be public and must not be + * an inner class. + * + *

A service provider and its supporting code may be developed in a module, + * which is then deployed on the application module path or in a modular + * image. Alternatively, a service provider and its supporting code may be + * packaged as a JAR file and deployed on the application class path. The + * advantage of developing a service provider in a module is that the provider + * can be fully encapsulated to hide all details of its implementation. + * + *

An application that obtains a service loader for a given service is + * indifferent to whether providers of the service are deployed in modules or + * packaged as JAR files. The application instantiates service providers via + * the service loader's iterator, or via {@link Provider Provider} objects in + * the service loader's stream, without knowledge of the service providers' + * locations. + * + *

Deploying service providers as modules

+ * + *

A service provider that is developed in a module must be specified in a + * provides directive in the module declaration. The provides directive + * specifies both the service and the service provider; this helps to locate the + * provider when another module, with a uses directive for the service, + * obtains a service loader for the service. It is strongly recommended that the + * module does not export the package containing the service provider. There is + * no support for a module specifying, in a provides directive, a service + * provider in another module. + + *

A service provider that is developed in a module has no control over when + * it is instantiated, since that occurs at the behest of the application, but it + * does have control over how it is instantiated: + * + *

+ * + *

A service provider that is deployed as an + * {@linkplain java.lang.module.ModuleDescriptor#isAutomatic automatic module} on + * the application module path must have a provider constructor. There is no + * support for a provider method in this case. + * + *

As an example, suppose a module specifies the following directives: + *

{@code
+ *     provides com.example.CodecFactory with com.example.impl.StandardCodecs;
+ *     provides com.example.CodecFactory with com.example.impl.ExtendedCodecsFactory;
+ * }
+ * + *

where + * + *

+ * + *

A service loader will instantiate {@code StandardCodecs} via its + * constructor, and will instantiate {@code ExtendedCodecsFactory} by invoking + * its {@code provider} method. The requirement that the provider constructor or + * provider method is public helps to document the intent that the class (that is, + * the service provider) will be instantiated by an entity (that is, a service + * loader) which is outside the class's package. + * + *

Deploying service providers on the class path

+ * + * A service provider that is packaged as a JAR file for the class path is + * identified by placing a provider-configuration file in the resource + * directory {@code META-INF/services}. The name of the provider-configuration + * file is the fully qualified binary name of the service. The provider-configuration + * file contains a list of fully qualified binary names of service providers, one + * per line. + * + *

For example, suppose the service provider + * {@code com.example.impl.StandardCodecs} is packaged in a JAR file for the + * class path. The JAR file will contain a provider-configuration file named: + * + *

{@code + * META-INF/services/com.example.CodecFactory + * }
+ * + * that contains the line: + * + *
{@code + * com.example.impl.StandardCodecs # Standard codecs + * }
+ * + *

The provider-configuration file must be encoded in UTF-8. + * Space and tab characters surrounding each service provider's name, as well as + * blank lines, are ignored. The comment character is {@code '#'} + * ({@code '\u0023'} NUMBER SIGN); + * on each line all characters following the first comment character are ignored. + * If a service provider class name is listed more than once in a + * provider-configuration file then the duplicate is ignored. If a service + * provider class is named in more than one configuration file then the duplicate + * is ignored. + * + *

A service provider that is mentioned in a provider-configuration file may + * be located in the same JAR file as the provider-configuration file or in a + * different JAR file. The service provider must be visible from the class loader + * that is initially queried to locate the provider-configuration file; this is + * not necessarily the class loader which ultimately locates the + * provider-configuration file. + * + *

Timing of provider discovery

+ * + *

Service providers are loaded and instantiated lazily, that is, on demand. + * A service loader maintains a cache of the providers that have been loaded so + * far. Each invocation of the {@code iterator} method returns an {@code Iterator} + * that first yields all of the elements cached from previous iteration, in * instantiation order, and then lazily locates and instantiates any remaining - * providers, adding each one to the cache in turn. Similarly, each invocation - * of the {@code stream} method returns a stream that first processes all + * providers, adding each one to the cache in turn. Similarly, each invocation + * of the stream method returns a {@code Stream} that first processes all * providers loaded by previous stream operations, in load order, and then lazily * locates any remaining providers. Caches are cleared via the {@link #reload * reload} method. * - *

Deploying provider classes in modules

+ *

Errors

* - *

A provider deployed as an explicit module must have an appropriate - * provides clause in its module descriptor to declare that the module - * provides an implementation of the service. - * - *

A provider deployed as an explicit module is instantiated by a - * provider factory or directly via the provider's constructor. In the - * module declaration then the class name specified in the provides clause - * is a provider factory if it is public and explicitly declares a public static - * no-args method named "{@code provider}". The return type of the method must be - * assignable to the service type. If the class is not a provider factory - * then it is public with a public zero-argument constructor. The requirement - * that the provider factory or provider class be public helps to document the - * intent that the provider will be instantiated by the service-provider loading - * facility. - * - *

Providers deployed as {@link - * java.lang.module.ModuleDescriptor#isAutomatic automatic-modules} on the - * module path must have a public zero-argument constructor. If the provider - * also declares a public static method named "{@code provider}" then it is - * ignored. - * - *

As an example, suppose a module declares the following: - * - *

{@code
- *     provides com.example.CodecSet with com.example.impl.StandardCodecs;
- *     provides com.example.CodecSet with com.example.impl.ExtendedCodecsFactory;
- * }
- * - * where - * - * - *

For this example then {@code StandardCodecs}'s no-arg constructor will - * be used to instantiate {@code StandardCodecs}. {@code ExtendedCodecsFactory} - * will be treated as a provider factory and {@code - * ExtendedCodecsFactory.provider()} will be invoked to obtain the provider. - * - *

Deploying provider classes on the class path

- * - *

A service provider that is packaged as a JAR file for - * the class path is identified by placing a provider-configuration file - * in the resource directory {@code META-INF/services}. The file's name is - * the fully-qualified binary name - * of the service's type. The file contains a list of fully-qualified binary - * names of concrete provider classes, one per line. Space and tab characters - * surrounding each name, as well as blank lines, are ignored. The comment - * character is {@code '#'} ('\u0023', - * NUMBER SIGN); on - * each line all characters following the first comment character are ignored. - * The file must be encoded in UTF-8. - * If a particular concrete provider class is named in more than one - * configuration file, or is named in the same configuration file more than - * once, then the duplicates are ignored. The configuration file naming a - * particular provider need not be in the same JAR file or other distribution - * unit as the provider itself. The provider must be visible from the same - * class loader that was initially queried to locate the configuration file; - * note that this is not necessarily the class loader from which the file was - * actually located. - * - *

For the example, then suppose {@code com.example.impl.StandardCodecs} is - * packaged in a JAR file for the class path then the JAR file will contain a - * file named: - *

{@code - * META-INF/services/com.example.CodecSet - * }
- * that contains the line: - *
{@code - * com.example.impl.StandardCodecs # Standard codecs - * }
- * - *

Using ServiceLoader from code in modules

- * - *

An application or library using this loading facility and developed - * and deployed as an explicit module must have an appropriate uses - * clause in its module descriptor to declare that the module uses - * implementations of the service. Combined with the requirement is that a - * provider deployed as an explicit module must have an appropriate - * provides clause allows consumers of a service to be linked - * to modules containing providers of the service. - * - *

For the example, if code in a module uses a service loader to load - * implementations of {@code com.example.CodecSet} then its module will declare - * the usage with:

{@code    uses com.example.CodecSet; }
- * - *

Errors

- * - *

When using the service loader's {@code iterator} then its {@link + *

When using the service loader's {@code iterator}, the {@link * Iterator#hasNext() hasNext} and {@link Iterator#next() next} methods will - * fail with {@link ServiceConfigurationError} if an error occurs locating or - * instantiating a provider. When processing the service loader's stream then - * {@code ServiceConfigurationError} is thrown by whatever method causes a - * provider class to be loaded. + * fail with {@link ServiceConfigurationError} if an error occurs locating, + * loading or instantiating a service provider. When processing the service + * loader's stream then {@code ServiceConfigurationError} may be thrown by any + * method that causes a service provider to be located or loaded. * - *

When loading or instantiating a provider class in a named module then - * {@code ServiceConfigurationError} can be thrown for the following reasons:

+ *

When loading or instantiating a service provider in a module, {@code + * ServiceConfigurationError} can be thrown for the following reasons: * *

* - *

When reading a provider-configuration file, or loading or instantiating a - * provider class named in a provider-configuration file, then {@code + *

When reading a provider-configuration file, or loading or instantiating + * a provider class named in a provider-configuration file, then {@code * ServiceConfigurationError} can be thrown for the following reasons: * *

* @@ -1232,42 +1308,26 @@ public final class ServiceLoader } /** - * Lazily load and instantiate the available providers of this loader's - * service. - * - *

The iterator returned by this method first yields all of the - * elements of the provider cache, in the order that they were loaded. - * It then lazily loads and instantiates any remaining providers, - * adding each one to the cache in turn. + * Returns an iterator to lazily load and instantiate the available + * providers of this loader's service. * *

To achieve laziness the actual work of locating and instantiating - * providers must be done by the iterator itself. Its {@link - * java.util.Iterator#hasNext hasNext} and {@link java.util.Iterator#next - * next} methods can therefore throw a {@link ServiceConfigurationError} - * if a provider class cannot be loaded, doesn't have an appropriate static - * factory method or constructor, can't be assigned to the service type or - * if any other kind of exception or error is thrown as the next provider - * is located and instantiated. To write robust code it is only necessary - * to catch {@link ServiceConfigurationError} when using a service iterator. - * - *

If such an error is thrown then subsequent invocations of the + * providers is done by the iterator itself. Its {@link Iterator#hasNext + * hasNext} and {@link Iterator#next next} methods can therefore throw a + * {@link ServiceConfigurationError} for any of the reasons specified in + * the Errors section above. To write robust code it + * is only necessary to catch {@code ServiceConfigurationError} when using + * the iterator. If an error is thrown then subsequent invocations of the * iterator will make a best effort to locate and instantiate the next * available provider, but in general such recovery cannot be guaranteed. * - *

Design Note - * Throwing an error in these cases may seem extreme. The rationale for - * this behavior is that a malformed provider-configuration file, like a - * malformed class file, indicates a serious problem with the way the Java - * virtual machine is configured or is being used. As such it is - * preferable to throw an error rather than try to recover or, even worse, - * fail silently.
- * - *

If this loader's provider caches are cleared by invoking the {@link - * #reload() reload} method then existing iterators for this service - * loader should be discarded. - * The {@link java.util.Iterator#hasNext() hasNext} and {@link - * java.util.Iterator#next() next} methods of the iterator throw {@link + *

Caching: The iterator returned by this method first yields all of + * the elements of the provider cache, in the order that they were loaded. + * It then lazily loads and instantiates any remaining service providers, + * adding each one to the cache in turn. If this loader's provider caches are + * cleared by invoking the {@link #reload() reload} method then existing + * iterators for this service loader should be discarded. + * The {@code hasNext} and {@code next} methods of the iterator throw {@link * java.util.ConcurrentModificationException ConcurrentModificationException} * if used after the provider cache has been cleared. * @@ -1275,6 +1335,12 @@ public final class ServiceLoader * Invoking its {@link java.util.Iterator#remove() remove} method will * cause an {@link UnsupportedOperationException} to be thrown. * + * @apiNote Throwing an error in these cases may seem extreme. The rationale + * for this behavior is that a malformed provider-configuration file, like a + * malformed class file, indicates a serious problem with the way the Java + * virtual machine is configured or is being used. As such it is preferable + * to throw an error rather than try to recover or, even worse, fail silently. + * * @return An iterator that lazily loads providers for this loader's * service * @@ -1331,35 +1397,36 @@ public final class ServiceLoader } /** - * Returns a stream that lazily loads the available providers of this - * loader's service. The stream elements are of type {@link Provider - * Provider}, the {@code Provider}'s {@link Provider#get() get} method - * must be invoked to get or instantiate the provider. + * Returns a stream to lazily load available providers of this loader's + * service. The stream elements are of type {@link Provider Provider}, the + * {@code Provider}'s {@link Provider#get() get} method must be invoked to + * get or instantiate the provider. * - *

When processing the stream then providers that were previously + *

To achieve laziness the actual work of locating providers is done + * when processing the stream. If a service provider cannot be loaded for any + * of the the reasons specified in the Errors section + * above then {@link ServiceConfigurationError} is thrown by whatever method + * caused the service provider to be loaded.

+ * + *

Caching: When processing the stream then providers that were previously * loaded by stream operations are processed first, in load order. It then - * lazily loads any remaining providers. If a provider class cannot be - * loaded, can't be assigned to the service type, or some other error is - * thrown when locating the provider then it is wrapped with a {@code - * ServiceConfigurationError} and thrown by whatever method caused the - * provider to be loaded.

+ * lazily loads any remaining service providers. If this loader's provider + * caches are cleared by invoking the {@link #reload() reload} method then + * existing streams for this service loader should be discarded. The returned + * stream's source {@link Spliterator spliterator} is fail-fast and + * will throw {@link ConcurrentModificationException} if the provider cache + * has been cleared.

* - *

If this loader's provider caches are cleared by invoking the {@link - * #reload() reload} method then existing streams for this service loader - * should be discarded. The returned stream's source {@code Spliterator} is - * fail-fast and will throw {@link ConcurrentModificationException} - * if the provider cache has been cleared.

- * - *

The following examples demonstrate usage. The first example - * creates a stream of providers, the second example is the same except - * that it sorts the providers by provider class name (and so locate all - * providers). + *

The following examples demonstrate usage. The first example creates + * a stream of {@code CodecFactory} objects, the second example is the same + * except that it sorts the providers by provider class name (and so locate + * all providers). *

{@code
-     *    Stream providers = ServiceLoader.load(CodecSet.class)
+     *    Stream providers = ServiceLoader.load(CodecFactory.class)
      *            .stream()
      *            .map(Provider::get);
      *
-     *    Stream providers = ServiceLoader.load(CodecSet.class)
+     *    Stream providers = ServiceLoader.load(CodecFactory.class)
      *            .stream()
      *            .sorted(Comparator.comparing(p -> p.type().getName()))
      *            .map(Provider::get);
@@ -1463,68 +1530,67 @@ public final class ServiceLoader
     }
 
     /**
-     * Creates a new service loader for the given service type and class
-     * loader. The service loader locates service providers in both named and
-     * unnamed modules:
+     * Creates a new service loader for the given service. The service loader
+     * uses the given class loader as the starting point to locate service
+     * providers for the service. The service loader's {@link #iterator()
+     * iterator} and {@link #stream() stream} locate providers in both named
+     * and unnamed modules, as follows:
      *
      * 
    - *
  • Service providers are located in named modules defined to the - * class loader, or any class loader that is reachable via parent - * delegation.

    + *
  • Step 1: Locate providers in named modules.

    * - *

    Additionally, and with the exception of the bootstrap and {@linkplain - * ClassLoader#getPlatformClassLoader() platform} class loaders, if the - * class loader, or any class loader reachable via parent delegation, - * defines modules in a module layer then the providers in the module layer - * are located. For example, suppose there is a module layer where each - * module is defined to its own class loader (see {@link - * ModuleLayer#defineModulesWithManyLoaders defineModulesWithManyLoaders}). - * If this {@code ServiceLoader.load} method is invoked to locate providers - * using any of the class loaders created for this layer then it will locate - * all of the providers in that layer, irrespective of their defining class - * loader.

  • + *

    Service providers are located in all named modules of the class + * loader or to any class loader reachable via parent delegation.

    * - *
  • A provider is an unnamed modules is located if its class - * name is listed in a service configuration file located by the the class - * loader's {@link ClassLoader#getResources(String) getResources} method. - * The provider class must be visible to the class loader. If a provider - * class is in a named module is listed then it is ignored (this is to - * avoid duplicates that would otherwise arise when a module has both a - * provides clause and a service configuration file in {@code - * META-INF/services} that lists the same provider).

  • - *
+ *

In addition, if the class loader is not the bootstrap or {@linkplain + * ClassLoader#getPlatformClassLoader() platform class loader}, then service + * providers may be located in the named modules of other class loaders. + * Specifically, if the class loader, or any class loader reachable via + * parent delegation, has a module in a {@linkplain ModuleLayer module + * layer}, then service providers in all modules in the module layer are + * located.

* - *

The ordering that the service loader's iterator and stream locate - * providers and yield elements is as follows: + *

For example, suppose there is a module layer where each module is + * in its own class loader (see {@link ModuleLayer#defineModulesWithManyLoaders + * defineModulesWithManyLoaders}). If this {@code ServiceLoader.load} method + * is invoked to locate providers using any of the class loaders created for + * the module layer, then it will locate all of the providers in the module + * layer, irrespective of their defining class loader.

* - *
    - *
  • Providers in named modules are located before service - * providers in unnamed modules.

  • + *

    Ordering: The service loader will first locate any service providers + * in modules defined to the class loader, then its parent class loader, + * its parent parent, and so on to the bootstrap class loader. If a class + * loader has modules in a module layer then all providers in that module + * layer are located (irrespective of their class loader) before the + * providers in the parent class loader are located. The ordering of + * modules in same class loader, or the ordering of modules in a module + * layer, is not defined.

    * - *
  • When locating providers in named modules then the service - * loader will first locate any service providers in modules defined to - * the class loader, then its parent class loader, its parent parent, - * and so on to the bootstrap class loader. If a class loader or any - * class loader in the parent delegation chain, defines modules in a - * module layer then all providers in that layer are located - * (irrespective of their class loader) before providers in the parent - * class loader are located. The ordering of modules defined to the - * same class loader, or the ordering of modules in a layer, is not - * defined.

  • + *

    If a module declares more than one provider then the providers + * are located in the order that its module descriptor {@linkplain + * java.lang.module.ModuleDescriptor.Provides#providers() lists the + * providers}. Providers added dynamically by instrumentation agents (see + * {@link java.lang.instrument.Instrumentation#redefineModule redefineModule}) + * are always located after providers declared by the module.

    * - *
  • If a named module declares more than one provider then the - * providers are located in the order that its module descriptor - * {@linkplain java.lang.module.ModuleDescriptor.Provides#providers() - * lists the providers}. Providers added dynamically by instrumentation - * agents (see {@link java.lang.instrument.Instrumentation#redefineModule - * redefineModule}) are always located after providers declared by the - * module.

  • + *
  • Step 2: Locate providers in unnamed modules.

    + * + *

    Service providers in unnamed modules are located if their class names + * are listed in provider-configuration files located by the class loader's + * {@link ClassLoader#getResources(String) getResources} method.

    + * + *

    The ordering is based on the order that the class loader's {@code + * getResources} method finds the service configuration files and within + * that, the order that the class names are listed in the file.

    + * + *

    In a provider-configuration file, any mention of a service provider + * that is deployed in a named module is ignored. This is to avoid + * duplicates that would otherwise arise when a named module has both a + * provides directive and a provider-configuration file that mention + * the same service provider.

    + * + *

    The provider class must be visible to the class loader.

  • * - *
  • When locating providers in unnamed modules then the - * ordering is based on the order that the class loader's {@link - * ClassLoader#getResources(String) getResources} method finds the - * service configuration files and within that, the order that the class - * names are listed in the file.

  • *
* * @apiNote If the class path of the class loader includes remote network @@ -1657,9 +1723,8 @@ public final class ServiceLoader /** * Creates a new service loader for the given service type to load service * providers from modules in the given module layer and its ancestors. It - * does not locate providers in unnamed modules. - * - *

The ordering that the service loader's iterator and stream locate + * does not locate providers in unnamed modules. The ordering that the service + * loader's {@link #iterator() iterator} and {@link #stream() stream} locate * providers and yield elements is as follows: * *

    @@ -1671,8 +1736,8 @@ public final class ServiceLoader * loader to locate providers with L3 as the context will locate providers * in the following order: L3, L1, L0, L2.

    * - *
  • If a named module declares more than one provider then the - * providers are located in the order that its module descriptor + *

  • If a module declares more than one provider then the providers + * are located in the order that its module descriptor * {@linkplain java.lang.module.ModuleDescriptor.Provides#providers() * lists the providers}. Providers added dynamically by instrumentation * agents are always located after providers declared by the module.

  • @@ -1708,26 +1773,25 @@ public final class ServiceLoader } /** - * Load the first available provider of this loader's service. This + * Load the first available service provider of this loader's service. This * convenience method is equivalent to invoking the {@link #iterator() * iterator()} method and obtaining the first element. It therefore * returns the first element from the provider cache if possible, it * otherwise attempts to load and instantiate the first provider. * - *

    The following example loads the first available provider. If there - * are no providers deployed then it uses a default implementation. + *

    The following example loads the first available service provider. If + * no service providers are located then it uses a default implementation. *

    {@code
    -     *    CodecSet provider =
    -     *        ServiceLoader.load(CodecSet.class).findFirst().orElse(DEFAULT_CODECSET);
    +     *    CodecFactory factory = ServiceLoader.load(CodecFactory.class)
    +     *                                        .findFirst()
    +     *                                        .orElse(DEFAULT_CODECSET_FACTORY);
          * }
    - * @return The first provider or empty {@code Optional} if no providers - * are located + * @return The first service provider or empty {@code Optional} if no + * service providers are located * * @throws ServiceConfigurationError - * If a provider class cannot be loaded, doesn't have the - * appropriate static factory method or constructor, can't be - * assigned to the service type, or if any other kind of exception - * or error is thrown when locating or instantiating the provider. + * If a provider class cannot be loaded for any of the reasons + * specified in the Errors section above. * * @since 9 * @spec JPMS @@ -1747,11 +1811,11 @@ public final class ServiceLoader * *

    After invoking this method, subsequent invocations of the {@link * #iterator() iterator} or {@link #stream() stream} methods will lazily - * look up providers (and instantiate in the case of {@code iterator}) - * from scratch, just as is done by a newly-created loader. + * locate providers (and instantiate in the case of {@code iterator}) + * from scratch, just as is done by a newly-created service loader. * - *

    This method is intended for use in situations in which new providers - * can be installed into a running Java virtual machine. + *

    This method is intended for use in situations in which new service + * providers can be installed into a running Java virtual machine. */ public void reload() { lookupIterator1 = null;