Restlet实战(10)结合源代码分析Restlet-Spring配置文件
在Restlet实战(四)与Spring集成给出了Restlet如何与Spring进行集成。下面就结合Restlet的源码分析一下Spring配置文件。
在Spring的配置文件中,我们能看到这样的代码:
- <entry key="/customers">
- <bean class="org.restlet.ext.spring.SpringFinder">
- <lookup-method name="createResource" bean="customersResource" />
- </bean>
- </entry>
从配置上很容易看出SpringFinder有一个方法:createResource,作用就是实例化在Spring中定义的Resource。看源码中的crateResource方法:
- public Resource createResource() {
- Resource result = null;
- if (getTargetClass() != null) {
- try {
- // Invoke the default constructor
- result = (Resource) getTargetClass().newInstance();
- } catch (Exception e) {
- getLogger()
- .log(
- Level.WARNING,
- "Exception while instantiating the target resource.",
- e);
- }
- }
- return result;
- }
代码如此简单,就不用多做解释了。
接下来,我们看一下另外的一个配置项:
- <bean id="restRoute" class="org.restlet.ext.spring.SpringRouter">
- <property name="attachments">
- <map>
- <entry key="/customers">
- <bean class="org.restlet.ext.spring.SpringFinder">
- <lookup-method name="createResource" bean="customersResource" />
- </bean>
- </entry>
- <entry key="/customers/{customerId}">
- <bean class="org.restlet.ext.spring.SpringFinder">
- <lookup-method name="createResource" bean="customerResource" />
- </bean>
- </entry>
- </map>
- </property>
- </bean>
是的,接下去就要分析SpringRouter,从上面的配置能看到此类有一个attachments的属性,那么跟这个属性有关的方法就是setAttachments了
- public static void setAttachments(Router router, Map<String, Object> routes) {
- Object value;
- Class resourceClass;
- try {
- for (final String key : routes.keySet()) {
- value = routes.get(key);
- if (value instanceof Restlet) {
- router.attach(key, (Restlet) value);
- } else if (value instanceof Class) {
- router.attach(key, (Class<? extends Resource>) value);
- } else if (value instanceof String) {
- resourceClass = Engine.loadClass((String) value);
- if (Resource.class.isAssignableFrom(resourceClass)) {
- router.attach(key, resourceClass);
- } else {
- router .getLogger() .warning(
- "Unknown class found in the mappings. Only subclasses of org.restlet.resource.Resource are allowed.");
- }
- } else {
- router.getLogger().warning(
- "Unknown object found in the mappings. Only instances of Restlet and subclasses of org.restlet.resource.Resource are allowed.");
- }
- }
- } catch (ClassNotFoundException e) {
- router.getLogger().log(Level.WARNING,
- "Unable to set the router mappings", e);
- }
- }
上述代码首先从一个循环开始,从Map里面取出需要的元素。key就上上面配置中的URL了,例如:/customers,/customers/{customerId},而value就是我们配置的相应的bean,程序里首先会check得到的value是不是Restlet的实例,如果是,就把这个value attach到router上,如果不是,则检查这个value是一个class还是一个String类型的字符串。如果是class,则认为这个class必定是一个Resource,直接attach到router;如果是String类型的字符串,则首先会认为是一个指定Resource的字符串,实例化,并attach到router,否则记录警告的日志信息。
看到这里,不知各位是否有想法,当时我看的时候,对第二、三个判断倒是很清晰,无非是把Resource直接attach到Router上,而对于第一个判断,按照上述代码所写,只要是Restlet的实例貌似就可以,换句话说,是不是我定义一个Application也可以作为一个元素配置上去?
限于篇幅的长度以及可读性,下篇给出相关的测试。