天天看点

使用注解属性绑定

大家应该知道在spring中有一个注解@value,他可以帮助我们来讲spring加载的配置文件(*.perperties)文件中的信息自动的注入到我们的非静态属性中的。

一般情况下我们会这样使用:

1.  首先在spring的配置文件中加载属性文件:

1

<code>&lt;context:property-placeholder location=</code><code>"classpath:component.properties"</code>  <code>ignore-unresolvable=</code><code>"true"</code><code>/&gt;</code>

然后在java代码中使用@value注解就可以注入值了,比如:

2

<code>@value</code><code>(</code><code>"${open_office_install_home}"</code><code>)</code>

<code>private</code> <code>string openofficeinstallhome;</code>

当然属性如果是static的话是不能注入的。

其实这个自动注入的过程实现起来比较简单,我们下面通过一个例子来大致描述一下这个原理吧,这个例子是我写的,并不代表spring的源码就是这么实现的。但是原理是一样的。

1.  我们先自定义一个注解:

3

4

5

6

7

8

9

10

11

12

<code>import</code> <code>java.lang.annotation.elementtype;</code>

<code>import</code> <code>java.lang.annotation.retention;</code>

<code>import</code> <code>java.lang.annotation.retentionpolicy;</code>

<code>import</code> <code>java.lang.annotation.target;</code>

<code>@target</code><code>(elementtype.method)</code>

<code>@retention</code><code>(retentionpolicy.runtime)</code>

<code>public</code> <code>@interface</code> <code>value {</code>

<code>    </code><code>public</code> <code>string value();</code>

<code>}</code>

2. 然后新增一个处理类:

13

14

15

16

17

18

19

20

21

22

<code>import</code> <code>java.lang.reflect.invocationhandler;</code>

<code>import</code> <code>java.lang.reflect.method;</code>

<code>import</code> <code>java.util.properties;</code>

<code>public</code> <code>class</code> <code>propertyinvokationhandler </code><code>implements</code> <code>invocationhandler {</code>

<code>    </code><code>private</code> <code>properties properties;</code>

<code>    </code><code>public</code> <code>propertyinvokationhandler(properties properties) {</code>

<code>        </code><code>this</code><code>.properties = properties;</code>

<code>    </code><code>}</code>

<code>    </code><code>@override</code>

<code>    </code><code>public</code> <code>object invoke(object proxy, method method, object[] args) </code><code>throws</code> <code>throwable {</code>

<code>        </code><code>value annotation = method.getannotation(value.</code><code>class</code><code>);</code>

<code>        </code><code>if</code><code>(annotation == </code><code>null</code><code>){</code>

<code>            </code><code>throw</code> <code>new</code> <code>runtimeexception(string.format(</code><code>"method:{} is not bound to a property."</code><code>, method.getname()));</code>

<code>        </code><code>}</code>

<code>        </code><code>return</code> <code>properties.getproperty(annotation.value());</code>

3.  创建一个公共方法:

<code>import</code> <code>java.lang.reflect.proxy;</code>

<code>public</code> <code>class</code> <code>propertytool {</code>

<code>    </code><code>private</code> <code>propertytool() {</code>

<code>    </code><code>public</code> <code>static</code> <code>&lt;t&gt; t bindproperties(class&lt;t&gt; clazz, properties properties) {</code>

<code>        </code><code>return</code> <code>(t) proxy.newproxyinstance(thread.currentthread().getcontextclassloader(),</code>

<code>                </code><code>new</code> <code>class[]{clazz},</code>

<code>                </code><code>new</code> <code>propertyinvokationhandler(properties));</code>

这样我们就完成了这个功能了。

下面我们通过测试代码来验证一下我们的功能是否起作用:

我们创建一个接口:

<code>public</code> <code>interface</code> <code>userservice {</code>

<code>    </code><code>@value</code><code>(</code><code>"user.name"</code><code>)</code>

<code>    </code><code>public</code> <code>string getusername();</code>

<code>    </code><code>@value</code><code>(</code><code>"user.password"</code><code>)</code>

<code>    </code><code>public</code> <code>string getpassword();</code>

然后编写测试类:

23

24

<code>import</code> <code>java.io.fileinputstream;</code>

<code>import</code> <code>java.io.ioexception;</code>

<code>import</code> <code>java.io.inputstream;</code>

<code>public</code> <code>class</code> <code>userservicetester {</code>

<code>    </code><code>public</code> <code>static</code> <code>void</code> <code>main(string[] args) {</code>

<code>        </code><code>properties properties = </code><code>new</code> <code>properties();</code>

<code>        </code><code>try</code> <code>{</code>

<code>            </code><code>string path = userservicetester.</code><code>class</code><code>.getresource(</code><code>"/user.properties"</code><code>).getpath();</code>

<code>            </code><code>inputstream in = </code><code>new</code> <code>fileinputstream(path);</code>

<code>            </code><code>properties.load(in);</code>

<code>            </code><code>in.close();</code>

<code>        </code><code>} </code><code>catch</code><code>(ioexception ex) {</code>

<code>            </code><code>ex.printstacktrace();</code>

<code>        </code><code>userservice config = propertytool.bindproperties(userservice.</code><code>class</code><code>, properties);</code>

<code>        </code><code>system.out.println(</code><code>"user name: "</code> <code>+ config.getusername());</code>

<code>        </code><code>system.out.println(</code><code>"password: "</code> <code>+ config.getpassword());</code>

而我们的user.properties属性文件中的内容为:

<code>user.name=rollenholt</code>

<code>user.password=</code><code>123</code>

运行上面的main方法,就会输出属性文件中的内容了。

不知道大家有没有注意到,我们在测试代码中使用的userservice是一个接口,我们并没有创建他的实现类,但是我们在main函数中依旧可以钓鱼他的方法。那是因为在运行时自动生成了一个实现。是不是觉的这个功能可以用在很多的地方呀。