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 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 choosing between multiple
service providers (based on the functionality they expose through the service),
and handling the possibility that no service providers are located.
An application obtains a service loader for a given service by invoking
one of the static 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 application module does not
contain the service, then its module declaration must have a requires
directive that specifies the module which exports the service. It is strongly
recommended that the application module does not require modules which
contain providers of the service.
A service loader can be used to locate and instantiate providers of the
service by means of the iterator
method. ServiceLoader
also defines the stream
method to obtain a stream of providers
that can be inspected and filtered without instantiating them.
As an example, suppose the service is com.example.CodecFactory
, an
interface that defines methods for producing encoders and decoders:
package com.example;
public interface CodecFactory {
Encoder getEncoder(String encodingName);
Decoder getDecoder(String encodingName);
}
The following code obtains a service loader for the CodecFactory
service, then uses its iterator (created automatically by the enhanced-for
loop) to yield instances of the service providers that are located:
ServiceLoader<CodecFactory> 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;
}
If this code resides in a module, then in order to refer to the
com.example.CodecFactory
interface, the module declaration would
require the module which exports the interface. The module declaration would
also specify use of com.example.CodecFactory
:
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 CodecFactory
that is capable of producing a "PNG" encoder may be annotated
with @PNG
. The following code uses service loader's stream
method to yield instances of Provider<CodecFactory>
in contrast to
how the iterator yields instances of CodecFactory
:
ServiceLoader<CodecFactory> loader = ServiceLoader.load(CodecFactory.class);
Set<CodecFactory> pngFactories = loader
.stream() // Note a below
.filter(p -> p.type().isAnnotationPresent(PNG.class)) // Note b
.map(Provider::get) // Note c
.collect(Collectors.toSet());
Provider<CodecFactory>
objects p.type()
yields a Class<CodecFactory>
get()
yields an instance of CodecFactory
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:
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.
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 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.
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 Provider
objects in
the service loader's stream, without knowledge of the service providers'
locations.
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:
In this case, the service provider itself need not be assignable to the service's interface or class.
In this case, the service provider must be assignable to the service's interface or class
A service provider that is deployed as an 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 directive:
provides com.example.CodecFactory with com.example.impl.StandardCodecs,
com.example.impl.ExtendedCodecsFactory;
where
com.example.CodecFactory
is the two-method service from
earlier. com.example.impl.StandardCodecs
is a public class that implements
CodecFactory
and has a public no-args constructor. com.example.impl.ExtendedCodecsFactory
is a public class that
does not implement CodecFactory, but it declares a public static no-args
method named "provider" with a return type of CodecFactory
. A service loader will instantiate StandardCodecs
via its
constructor, and will instantiate ExtendedCodecsFactory
by invoking
its 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.
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
com.example.impl.StandardCodecs
is packaged in a JAR file for the
class path. The JAR file will contain a provider-configuration file named:
META-INF/services/com.example.CodecFactory
that contains the line:
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 '#'
(U+0023
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.
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 iterator
method returns an 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 stream method returns a 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 reload
method.
When using the service loader's iterator
, the hasNext
and next
methods will
fail with ServiceConfigurationError
if an error occurs locating,
loading or instantiating a service provider. When processing the service
loader's stream then ServiceConfigurationError
may be thrown by any
method that causes a service provider to be located or loaded.
When loading or instantiating a service provider in a module, ServiceConfigurationError
can be thrown for the following reasons:
provider
". null
or throwing an exception. When reading a provider-configuration file, or loading or instantiating
a provider class named in a provider-configuration file, then ServiceConfigurationError
can be thrown for the following reasons:
IOException
occurs while reading the
provider-configuration file; Service loaders always execute in the security context of the caller of the iterator or stream methods and may also be restricted by the security context of the caller that created the service loader. Trusted system code should typically invoke the methods in this class, and the methods of the iterators which they return, from within a privileged security context.
Instances of this class are not safe for use by multiple concurrent threads.
Unless otherwise specified, passing a null
argument to any
method in this class will cause a NullPointerException
to be thrown.
Modifier and Type | Class and Description |
---|---|
private class | ServiceLoader.
Implements lazy service provider lookup of service providers that are provided by modules in a module layer (or parent layers) |
private class | ServiceLoader.
Implements lazy service provider lookup where the service providers are configured via service configuration files. |
private class | ServiceLoader.
Implements lazy service provider lookup of service providers that are provided by modules defined to a class loader or to modules in layers with a module defined to the class loader. |
public static interface | |
private static class | ServiceLoader.
A Provider implementation that supports invoking, with reduced permissions, the static factory to obtain the provider or the provider's no-arg constructor. |
private class |
Modifier and Type | Field and Description |
---|---|
private final AccessControlContext | acc
References Deprecated
AccessControlContext is deprecated or references (maybe indirectly) at least one deprecated element.
|
private final List | |
private static final JavaLangAccess | |
private final ModuleLayer | |
private boolean | |
private final List | |
private final ClassLoader | |
private Iterator | |
private Iterator | |
private int | |
private final Class | |
private final String |
Access | Constructor and Description |
---|---|
private | ServiceLoader(Class<?> caller, ModuleLayer layer, Class<S> svc)
Initializes a new instance of this class for locating service providers in a module layer. |
private | ServiceLoader(Class<?> caller, Class<S> svc, ClassLoader cl)
Initializes a new instance of this class for locating service providers via a class loader. |
private | ServiceLoader(Module callerModule, Class<S> svc, ClassLoader cl)
Initializes a new instance of this class for locating service providers via a class loader. |
Modifier and Type | Method and Description |
---|---|
private static void | checkCaller(Class<?> caller, Class<?> svc)
Checks that the given service type is accessible to types in the given module, and check that the module declares that it uses the service type. |
private static void | |
private static void | |
private static void | |
public Optional | Returns: The first service provider or emptyOptional if no
service providers are locatedLoad the first available service provider of this loader's service. |
private Method | |
private Constructor | |
private boolean | |
public Iterator | Returns: An iterator that lazily loads providers for this loader's serviceImplements java. Returns an iterator to lazily load and instantiate the available providers of this loader's service. |
pack-priv static < the class of the service type S> ServiceLoader | Returns: A new service loaderThe interface or abstract class representing the service service, ClassLoader The class loader to be used to load provider-configuration files
and provider classes, or loader, Module null if the system class
loader (or, failing that, the bootstrap class loader) is to be
usedThe caller's module for which a new service loader is created callerModule)Creates a new service loader for the given service type, class loader, and caller. |
public static < the class of the service type S> ServiceLoader | Returns: A new service loaderThe interface or abstract class representing the service service, ClassLoader The class loader to be used to load provider-configuration files
and provider classes, or loader)null if the system class
loader (or, failing that, the bootstrap class loader) is to be
usedCreates a new service loader for the given service. |
public static < the class of the service type S> ServiceLoader | Returns: A new service loaderThe interface or abstract class representing the service service)Creates a new service loader for the given service type, using the current thread's context class loader. |
public static < the class of the service type S> ServiceLoader | Returns: A new service loaderThe module layer layer, Class<S> The interface or abstract class representing the service service)Creates a new service loader for the given service type to load service providers from modules in the given module layer and its ancestors. |
public static < the class of the service type S> ServiceLoader | Returns: A new service loaderThe interface or abstract class representing the service service)Creates a new service loader for the given service type, using the platform class loader. |
private ServiceLoader. | |
private Iterator | |
public void | |
public Stream | Returns: A stream that lazily loads providers for this loader's serviceReturns a stream to lazily load available providers of this loader's service. |
public String | Returns: A descriptive stringOverrides java. Returns a string describing this service. |