今天看到spring mvc 使用Java Validation Api(JSR-303)进行校验,需要加载一个 其具体实现(比如Hibernate Validator), 本来没有什么问题,但是突然想到这其中到底是怎样一种加载过程呢,也就是说spring为什么能够找到Hibernate Validator来作为JSR-303的具体实现的呢?
1. java中的SPI机制
这篇文章对java的SPI机制讲的比较容易理解,就不多做记录. http://www.cnblogs.com/javaee6/p/3714719.html
下面的小例子,分别采用SPI机制和 反射类名的方式来获取服务(helloApi)
首先给出服务接口,将它导出为一个jar包, hello-api.jar
package helloApi; public interface Hello {
String say();
}
下面是它的一个具体实现:
package helloImpl; import helloApi.Hello; public class HelloImpl implements Hello {
@Override
public String say() {
return "hello Massclouds";
}
}
将这个实现导出为jar包,但是为了使用java 的SPI机制,我们需要在导出的jar包的META-INF中增加services文件夹,并在其中增加一个文件,以 所实现接口 helloApi.Hello 为名,以 具体实现类 helloImpl.HelloImpl为内容。
最后就是分别使用SPI和反射的方式来获得服务。
Hello hello = null;
//使用java的SPI机制加载Hello api的实现类
ServiceLoader<Hello> serviceLoader = ServiceLoader.load(Hello.class);
Iterator<Hello> iter = serviceLoader.iterator();
if(iter.hasNext()){
hello = iter.next();
} System.out.println(hello.say()); //通过反射类名的方式来获得Hello api的实现
hello = (Hello)Class.forName("helloImpl.HelloImpl").newInstance();
System.out.println(hello.say());
回过头来考虑jdbc的加载方式,跟我上面的例子是一样的。这两种方式肯定是SPI更好,采用类名反射的方式本质上还是硬编码嘛!
2. 如何加载Hibernate Validator来作为JSR-303的实现
我们打开Hibernate Validator的jar包来看看。
而这个文件的内容正是具体实现 org.hibernate.validator.HibernateValidator。至于具体Spring是怎么做的我就不详细研究了,哈哈!!!