package ycl.learn.effective.java;
public interface ServiceAPI {
/**
* translate context to appropriate language.
*
* @param content
* @return
*/
public String translate(String content);
}
package ycl.learn.effective.java;
public class ChineseServiceSPI implements ServiceAPI{
public String translate(String content) {
String trans = content.equals("Ch")?"Chinese":null;
return trans;
}
}
package ycl.learn.effective.java;
public class EnglishServiceSPI implements ServiceAPI{
public String translate(String content) {
String trans = content.equals("En")?"Englis":null;
return trans;
}
}
package ycl.learn.effective.java;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.ServiceLoader;
public class BaseServiceSPI{
private ServiceLoader<ServiceAPI> sl;
public BaseServiceSPI(ServiceLoader<ServiceAPI> sl){
this.sl = sl;
}
/**
* just find first one implement.
*
* @param clazz
* @return
* @throws Exception
*/
public static Object findServiceProvider(Class clazz) throws Exception
{
String factoryId =clazz.getName();
String serviceId = "META-INF/services/" + factoryId;
InputStream is = null;
// First try the Context ClassLoader
ClassLoader cl = Thread.currentThread().getContextClassLoader();
if (cl != null) {
is = cl.getResourceAsStream(serviceId);
} else {
// No Context ClassLoader, try the current ClassLoader
cl = BaseServiceSPI.class.getClassLoader();
is = cl.getResourceAsStream(serviceId);
}
if (is == null) {
// No provider found
return null;
}
BufferedReader rd;
try {
rd = new BufferedReader(new InputStreamReader(is, "UTF-8"));
}
catch (java.io.UnsupportedEncodingException e) {
rd = new BufferedReader(new InputStreamReader(is));
}
String factoryClassName = null;
try {
// XXX Does not handle all possible input as specified by the
// Jar Service Provider specification
factoryClassName = rd.readLine();
rd.close();
} catch (IOException x) {
// No provider found
return null;
}
if (factoryClassName != null && !"".equals(factoryClassName)) {
// Note: here we do not want to fall back to the current
// ClassLoader because we want to avoid the case where the
// resource file was found using one ClassLoader and the
// provider class was instantiated using a different one.
Object p = clazz.cast(Class.forName(factoryClassName, true, cl)
.newInstance());
return p;
}
// No provider found
return null;
}
/**
* auto match method
* 1. if that is multiple implements, and you just want to auto choose.
*
* rule
* *. sub class method is return null, if this params is not match of this process.
* *. sub class method will be return not null, if this params is match of this process.
*
* @param content
* @return
*/
public String translate(String content) {
for(ServiceAPI serviceAPI: sl){
String trans = serviceAPI.translate(content);
if(trans != null){
return trans;
}
}
return null;
}
}
package ycl.learn.effective.java;
import java.util.ServiceLoader;
public class TestServiceLoader {
public static void main(String args[]) throws Exception{
ServiceLoader<ServiceAPI> sl = ServiceLoader.load(ServiceAPI.class);
BaseServiceSPI bs = new BaseServiceSPI(sl);
//auto choose implements.
//in this situation , you can add more and more implements.
//as like add Languages.
//also the JDK components use Service Loader to implements.
//you can read javax.xml.parsers.FactoryFinder
// javax.xml.parsers.SAXParserFactory
// javax.xml.parsers.DocumentBuilderFactory
// 1. System property
// 2. jdk-dir/lib/jaxp.properties set default implements
//javax.xml.parsers.DocumentBuilderFactory javax.xml.parsers.SAXParserFactory
// 3. META-INF/services/javax.xml.parsers.DocumentBuilderFactory ==> org.apache.crimson.jaxp.DocumentBuilderFactoryImpl
// 4. if haven't find,user Crimson。ClassNotFound
// summary, so you know what's xml parser you shoud be to use.
// "javax.xml.parsers.SAXParserFactory", "com.sun.org.apache.xerces.internal.jaxp.SAXParserFactoryImpl"
// "javax.xml.parsers.DocumentBuilderFactory" "com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderFactoryImpl"
// you can have multiple implements at one time, but you just choose one implement is ok.
System.setProperty("javax.xml.parsers.SAXParserFactory", "org.apache.xerces.jaxp.SAXParserFactoryImpl");
System.out.println(bs.translate("Ch"));
System.out.println("===");
System.out.println(bs.translate("En"));
// that's right, you can just get one implements as below.
// you also can get all impements use rules to choose right one.
ServiceAPI serviceAPI = (ServiceAPI) BaseServiceSPI.findServiceProvider(ServiceAPI.class);
System.out.println(serviceAPI.translate("Ch"));
System.out.println("hahaah");
System.out.println(serviceAPI.translate("En"));
}
}
分享到:
相关推荐
在Java编程语言中,`java.util.ServiceLoader` 是一个实用工具类,用于加载符合特定接口或抽象类的服务提供者。这个工具在Java平台标准版(Java SE)中被广泛使用,尤其是在实现模块化和插件化系统时。下面将详细...
RocketMQ 奇技淫巧之 ServiceLoader 源码解读 抓下来打包成了HTML文件, 方便离线观看
ServiceLoader是Java中的一种服务加载机制,它位于`java.util.ServiceLoader`类中,主要用来发现并加载符合特定接口的实现类。这个机制是Java SPI(Service Provider Interface)的一部分,允许开发者在不修改代码的...
4. 使用 ServiceLoader:使用 ServiceLoader 加载和实例化提供者的实现类。 SPI 机制的破坏双亲委派: 1. SPI 机制可以破坏双亲委派,因为 ServiceLoader 可以加载第三方提供者的实现类。 2. 例如,在 JDBC 连接池...
ServiceLoader是Java中的一种服务加载机制,它允许我们根据接口动态发现并加载实现类,而无需显式地通过Class.forName或者反射等方式进行实例化。这个机制主要基于Java的SPI(Service Provider Interface)服务提供...
这个maven插件为Java 6中引入的ServiceLoader生成服务文件: : 用 例如: < groupId>eu.somatik.serviceloader-maven-plugin < artifactId>serviceloader-maven-plugin < version>1.3.1 < param>...
Java 的 SPI(服务提供者接口)和 java.util.ServiceLoader 使用的简单演示。 介绍 这个演示应用程序包括 6 个小 jar 项目。 云服务 演示伪“云服务”提供者。 为服务提供者定义spidemo.cloud.spi.Cloud接口。 ...
Spring Boot java.util.ServiceLoader 使用java.util.ServiceLoader示例Spring Boot应用程序在类路径上动态加载所有实现。 多项目Gradle构建 1个Spring Boot应用 1个SPI项目 2个SPI实施项目 有关java.util....
在Java编程语言中,`ServiceLoader`是一个非常重要的组件,它允许我们按照Java的SPI(Service Provider Interface)机制来加载和使用服务提供商。这个标题提到的"System源码java-serviceloader-migration"是一个项目...
**Java ServiceLoader详解** 在Java世界中,ServiceLoader是一个用于加载服务实现的工具类,它位于`java.util.ServiceLoader`包下。这个类是Java SPI(Service Provider Interface)机制的核心,允许开发者在运行时...
Java服务加载器(ServiceLoader)是Java平台标准版(Java SE)中的一种设计模式,它允许应用程序使用服务提供者接口(SPI,Service Provider Interface)来动态发现和加载服务实现。这个机制使得软件组件之间可以...
一个简单的演示项目,用于将ServiceLoader与本机图像一起使用。 用以下方法构建罐子: $ mvn package 使用以下命令在JVM上运行: $ java -jar target/ServiceLoaderTest-1.0-SNAPSHOT.jar 输出应为: services....
什么是 Java ServiceLoader? 如何拼写 SPI(服务提供者接口)? 用 Java 编程语言编写的服务可以由不同的服务提供者实现。 虽然服务通常是用接口或抽象类编写的,但服务提供者是实现这些接口服务的实现类。 java....
当我们调用 ServiceLoader.load(Driver.class) 时,JVM 会遍历 META-INF/services 目录下的所有文件,并加载其中的类。如果找到了与 Driver 接口对应的类,就会实例化该类并将其注册为驱动程序。 META-INF/services...
WMRouter是一款Android路由框架,基于组件化的...基于SPI (Service Provider Interfaces) 的设计思想,WMRouter提供了ServiceLoader模块,类似Java中的java.util.ServiceLoader,但功能更加完善。通过ServiceLoader可
ServiceLoader<MyService> loader = ServiceLoader.load(MyService.class); for (MyService service : loader) { service.doSomething(); } ``` 6. **优势**:SPI使得框架或库可以在不依赖具体实现的情况下...
ServiceLoader<MyService> serviceLoader = ServiceLoader.load(MyService.class); for (MyService myService : serviceLoader) { System.out.println("调用服务:" + myService.doSomething()); } } } ``` 在...
- (2) 使用`ServiceLoader.iterator()`或`ServiceLoader.stream()`遍历加载的服务提供者。 - (3) 对每个服务提供者进行实例化,然后调用其提供的方法。 3. 示例应用: - 在JDBC中,不同的数据库驱动如MySQL、...
这个机制在Java中主要由`java.util.ServiceLoader`类来实现。在本示例中,我们将探讨如何创建一个简单的SPI入门级Demo,具体是关于实现一个加法服务。 首先,理解SPI的基本流程: 1. **定义接口**:服务接口是所有...