当前位置: 首页 > news >正文

使用注解属性绑定

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

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

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

1
<context:property-placeholder location= "classpath:component.properties"   ignore-unresolvable= "true" />

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

 
1
2
@Value ( "${open_office_install_home}" )
private  String openOfficeInstallHome;
 
当然属性如果是static的话是不能注入的。
 
其实这个自动注入的过程实现起来比较简单,我们下面通过一个例子来大致描述一下这个原理吧,这个例子是我写的,并不代表Spring的源码就是这么实现的。但是原理是一样的。
 
1.  我们先自定义一个注解:
1
2
3
4
5
6
7
8
9
10
11
12
import  java.lang.annotation.ElementType;
import  java.lang.annotation.Retention;
import  java.lang.annotation.RetentionPolicy;
import  java.lang.annotation.Target;
 
@Target (ElementType.METHOD)
@Retention (RetentionPolicy.RUNTIME)
public  @interface  Value {
 
     public  String value();
 
}

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
import  java.lang.reflect.InvocationHandler;
import  java.lang.reflect.Method;
import  java.util.Properties;
 
 
public  class  PropertyInvokationHandler  implements  InvocationHandler {
 
     private  Properties properties;
 
     public  PropertyInvokationHandler(Properties properties) {
         this .properties = properties;
     }
 
     @Override
     public  Object invoke(Object proxy, Method method, Object[] args)  throws  Throwable {
         Value annotation = method.getAnnotation(Value. class );
         if (annotation ==  null ){
             throw  new  RuntimeException(String.format( "Method:{} is not bound to a property." , method.getName()));
         }
         return  properties.getProperty(annotation.value());
     }
}

3.  创建一个公共方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
import  java.lang.reflect.Proxy;
import  java.util.Properties;
 
public  class  PropertyTool {
 
     private  PropertyTool() {
     }
 
     public  static  <T> T bindProperties(Class<T> clazz, Properties properties) {
         return  (T) Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(),
                 new  Class[]{clazz},
                 new  PropertyInvokationHandler(properties));
     }
}

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

 

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

我们创建一个接口:

1
2
3
4
5
6
7
8
9
public  interface  UserService {
 
     @Value ( "user.name" )
     public  String getUserName();
 
     @Value ( "user.password" )
     public  String getPassword();
 
}

然后编写测试类:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
import  java.io.FileInputStream;
import  java.io.IOException;
import  java.io.InputStream;
import  java.util.Properties;
 
public  class  UserServiceTester {
 
     public  static  void  main(String[] args) {
         Properties properties =  new  Properties();
 
         try  {
             String path = UserServiceTester. class .getResource( "/user.properties" ).getPath();
             InputStream in =  new  FileInputStream(path);
             properties.load(in);
             in.close();
         catch (IOException ex) {
             ex.printStackTrace();
         }
 
         UserService config = PropertyTool.bindProperties(UserService. class , properties);
         System.out.println( "User Name: "  + config.getUserName());
         System.out.println( "Password: "  + config.getPassword());
     }
}

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

1
2
user.name=rollenholt
user.password= 123

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

 

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

 

相关文章:

  • Wind7外接显示器选择拓展模式后,鼠标只能往右移动才能切换到外接显示器上,不能修改切换方向...
  • 策略模式和工厂模式的区别
  • Regsvr32.exe 用法
  • 【转】js 获取浏览器高度和宽度值(多浏览器)
  • XML注释导致VS2005崩溃
  • Bugtags,最适合移动应用的智能 Bug 管理系统
  • Java开发
  • Git技巧:右键菜单怎么去除?
  • RTMFP vs RTMP
  • Linux Distribution / ROM
  • android Git命令家底儿及Git数据通信原理详解
  • directsound抓取麦克风PCM数据封装类
  • spring 下载地址
  • ArrayList底层实现
  • shell脚本替换文件中字符
  • Computed property XXX was assigned to but it has no setter
  • echarts花样作死的坑
  • Java 内存分配及垃圾回收机制初探
  • LeetCode29.两数相除 JavaScript
  • Python学习之路16-使用API
  • Spark VS Hadoop:两大大数据分析系统深度解读
  • WinRAR存在严重的安全漏洞影响5亿用户
  • 记一次用 NodeJs 实现模拟登录的思路
  • 老板让我十分钟上手nx-admin
  • 微信如何实现自动跳转到用其他浏览器打开指定页面下载APP
  • 在electron中实现跨域请求,无需更改服务器端设置
  • 掌握面试——弹出框的实现(一道题中包含布局/js设计模式)
  • 【运维趟坑回忆录】vpc迁移 - 吃螃蟹之路
  • Java总结 - String - 这篇请使劲喷我
  • ​一、什么是射频识别?二、射频识别系统组成及工作原理三、射频识别系统分类四、RFID与物联网​
  • #100天计划# 2013年9月29日
  • $.proxy和$.extend
  • (¥1011)-(一千零一拾一元整)输出
  • (173)FPGA约束:单周期时序分析或默认时序分析
  • (备忘)Java Map 遍历
  • (论文阅读笔记)Network planning with deep reinforcement learning
  • (算法)求1到1亿间的质数或素数
  • (一)UDP基本编程步骤
  • (转)JAVA中的堆栈
  • (转)程序员技术练级攻略
  • (转)从零实现3D图像引擎:(8)参数化直线与3D平面函数库
  • .net core webapi Startup 注入ConfigurePrimaryHttpMessageHandler
  • .Net开发笔记(二十)创建一个需要授权的第三方组件
  • .net中应用SQL缓存(实例使用)
  • @property python知乎_Python3基础之:property
  • [ C++ ] STL_stack(栈)queue(队列)使用及其重要接口模拟实现
  • [ vulhub漏洞复现篇 ] GhostScript 沙箱绕过(任意命令执行)漏洞CVE-2019-6116
  • [ 云计算 | Azure 实践 ] 在 Azure 门户中创建 VM 虚拟机并进行验证
  • [100天算法】-实现 strStr()(day 52)
  • [Android]使用Retrofit进行网络请求
  • [Angular 基础] - 自定义指令,深入学习 directive
  • [BUG]Datax写入数据到psql报不能序列化特殊字符
  • [C#][DevPress]事件委托的使用
  • [C++ 从入门到精通] 12.重载运算符、赋值运算符重载、析构函数
  • [emuch.net]MatrixComputations(7-12)