JavaSE实现IoC
作者:Grey
原文地址:
Java SE 提供了三种方式,可以实现IoC,分别为:
- Java Beans
- Java ServiceLoader SPI
- JNDI(Java Naming and Directory Interface)
Java Beans 方式
java.beans包下的 Introspector 类提供了一个 getBeanInfo的方法,可以获取一个类的信息
BeanInfo bi=Introspector.getBeanInfo(User.class,Object.class);
如上,则可以获取User类对象的BeanInfo, 然后我们通过BeanInfo中的 getPropertyDescriptors 方法,可以获取到User对象中的所有属性和方法,
注意:java beans中,对于set(xxx)方法,统一叫:writeMethod(), 对于get() 方法,统一叫:readMethod()
Stream.of(bi.getPropertyDescriptors()).forEach(pd->{
Class<?> propertyType=pd.getPropertyType();
Method writeMethod=pd.getWriteMethod();
});
获取到方法和属性名称后,通过反射即可把对应的值设置到对应的属性中
writeMethod.invoke(name,value);
由于我们注入属性值的时候,我们注入的东西永远是一个字符串类型,如果需要注入的属性是其他类型(非字符串), 比如User类中,有一个属性是address,这个address是一个对象类型,
我们应该如何定义一个转换器,将字符串类型的值转换为我们需要的对象类型呢?
我们需要通过设置一个AddressEditor来实现这个转换,这个AddressEditor有两种实现方式, 一是实现PropertyEditor接口,另外一种方式是继承PropertyEditorSupport类,
由于我们只需要实现一些简单的转换,PropertyEditorSupport提供了更为便利的实现方式,所以我们采用继承PropertyEditorSupport类的方法,来实现类型的转换,
Address类的设计是:
public class Address {
private String name;
private Integer num;
// 省略 get / set / toString
}
我们的定义的规则如下,
输入的字符串用|来分割 name 和 num属性
例如: “贝克街|221” 这个字符串 会将“贝克街”赋给name,221赋给num
public class AddressEditor extends PropertyEditorSupport {
@Override
public void setAsText(String text) throws IllegalArgumentException {
String[] tokens = text.split("\\|");
Address address = new Address();
address.setName(tokens[0]);
address.setNum(Integer.valueOf(tokens[1]));
setValue(address);
}
}
但是我们需要重写setAsText方法,即:将字符串类型按照我们定义的规则转换成对应需要的类型即可,同理,我们可以实现一个DateEditor,让“yyyy-MM-dd”这样类型的字符串转换成日期格式。
public class DateEditor extends PropertyEditorSupport {
static final DateTimeFormatter dtf = DateTimeFormatter.ofPattern("yyyy-MM-dd");
@Override
public void setAsText(String text) throws IllegalArgumentException {
LocalDate localDate = LocalDate.parse(text, dtf);
ZoneId zone = ZoneId.systemDefault();
Instant instant = localDate.atStartOfDay().atZone(zone).toInstant();
setValue(Date.from(instant));
}
}
然后,我们需要使用java beans中的PropertyEditorManager类的registerEditor方法把这两个Editor注册进来
registerEditor(Address.class,AddressEditor.class);
registerEditor(Date.class,DateEditor.class);
最后,PropertyEditorManager的findEditor方法就可以根据我们前面得到的属性类型,找到对应的Editor来对值进行转换,转换成我们需要的属性类型的值
PropertyEditor editor = findEditor(propertyType);
if (editor != null) {
editor.setAsText(parameters.get(pd.getName()));
try {
writeMethod.invoke(user, editor.getValue());
} catch (IllegalAccessException | InvocationTargetException e) {
e.printStackTrace();
}
} else {
System.out.println("no editor for:" + pd.getName());
}
主函数调用示例
public static void main(String[] args) throws Exception {
Map<String, String> parameters = new HashMap<String, String>() {
{
//这里的key要和Node里面的属性名一致
put("name", "福尔摩斯");
put("address", "贝克街|221");
put("birthday", "1854-01-06");
}
};
User convert = PropertyEditorSample.convert(parameters);
System.out.println(convert);
}
运行结果
User{name='福尔摩斯', birthday=Thu Jan 05 23:54:17 CST 1854, address=Address{name='贝克街, 221 号}}
SPI方式
定义支付接口PayService
public interface PayService {
void pay();
}
定义多个实现:
public class WeixinpayService implements PayService{
@Override
public void pay() {
System.out.println("微信支付");
}
}
public class AlipayService implements PayService{
@Override
public void pay() {
System.out.println("支付宝支付");
}
}
在resources目录下建立META-INF文件夹,在META-INF文件夹下建立services目录,同时建立一个文件,名称为接口的全路径名,以这个项目为例, PayService的全路径名称为:
org.snippets.ioc.java.spi.PayService
在这个文件内,把实现类的全路径名写进去:
org.snippets.ioc.java.spi.AlipayService
org.snippets.ioc.java.spi.WeixinpayService
客户端调用:
ServiceLoader<PayService> serviceLoader = ServiceLoader.load(PayService.class);
for (PayService ele : serviceLoader) {
ele.pay();
}
其中ServiceLoader.load方法可以把所有配置的PayService实现得到
执行结果:
支付宝支付
微信支付
JNDI方式
定义一个Person类
public class Person implements Remote, Serializable {
private static final long serialVersionUID = 1L;
private String name;
private String password;
// 省略set / get方法
}
实现JNDI的客户端,实现初始化Person和查找Person两个功能
public static void initPerson() throws Exception {
//配置JNDI工厂和JNDI的url和端口。如果没有配置这些信息,将会出现NoInitialContextException异常
LocateRegistry.createRegistry(3000);
System.setProperty(Context.INITIAL_CONTEXT_FACTORY, "com.sun.jndi.rmi.registry.RegistryContextFactory");
System.setProperty(Context.PROVIDER_URL, "rmi://localhost:3000");
InitialContext ctx = new InitialContext();
//实例化person对象
Person p = new Person();
p.setName("zc");
p.setPassword("123");
//将person对象绑定到JNDI服务中,JNDI的名字叫做:person。
ctx.bind("person", p);
ctx.close();
}
public static void findPerson() throws Exception {
//因为前面已经将JNDI工厂和JNDI的url和端口已经添加到System对象中,这里就不用在绑定了
InitialContext ctx = new InitialContext();
//通过lookup查找person对象
Person person = (Person) ctx.lookup("person");
//打印出这个对象
System.out.println(person.toString());
ctx.close();
}
完整代码
参考资料
JavaSE实现IoC的更多相关文章
- Spring之IoC总结帖
Spring是一个开源框架,Spring是于2003 年兴起的一个轻量级的Java 开发框架,由Rod Johnson 在其著作Expert One-On-One J2EE Development a ...
- Spring 实践 -IoC
Spring 实践 标签: Java与设计模式 Spring简介 Spring是分层的JavaSE/EE Full-Stack轻量级开源框架.以IoC(Inverse of Control 控制反转) ...
- Spring框架之IOC(控制反转)
[TOC] 第一章Spring框架简介 IOC(控制反转)和AOP(面向方面编程)作为Spring框架的两个核心,很好地实现了解耦合.所以,简单来说,Spring是一个轻量级的控制反转(IoC)和面向 ...
- ioc(Inversion of Control)控制反转和DI
ioc意味着将你设计好的交给容器控制,而不是传统在你的对象中直接控制 谁控制了谁:传统的javaSE程序设计,我们直接在对象内部通过new进行创建对象,是程序主动去创建依赖对象:而ioc是有专门一个容 ...
- 框架学习之Spring(一IOC)----HelloWrod
一.概述 Spring是一个开源框架,它的核心是控制反转(IOC)和面向切面(AOP).简单来说,Spring是一个分层的JavaSE/EEfull-stack(一站式)轻量级开源框架. EE 开发分 ...
- Spring第一天——入门与IOC
大致内容 spring基本概念 IOC入门 [17.6.9更新],如何学习spring? 掌握用法 深入理解 不断实践 反复总结 再次深入理解与实践 一.Spring相关概念 1.概述: Sprin ...
- Spring框架-IOC和AOP简单总结
参考博客: https://blog.csdn.net/qq_22583741/article/details/79589910 1.Spring框架是什么,为什么,怎么用 1.1 Spring框架是 ...
- Spring系列之IOC的原理及手动实现
目录 Spring系列之IOC的原理及手动实现 Spring系列之DI的原理及手动实现 导语 Spring是一个分层的JavaSE/EE full-stack(一站式) 轻量级开源框架.也是几乎所有J ...
- Spring 框架学习—控制反转(IOC)
Spring是一个开源框架,Spring是于2003 年兴起的一个轻量级的Java 开发框架,由Rod Johnson创建. 简单来说,Spring是一个分层的JavaSE/EEfull-st ...
随机推荐
- C/C++函数与变量前面的标识符的作用
作者:良知犹存 转载授权以及围观->欢迎添加Wx:Allen-Iverson-me-LYN 缅怀逝者,向英雄致敬.愿山河无恙,国泰民安. 在用C/C++写代码的时候我们经常会使用一些标识符 ...
- HDU 6762 Mow (2020 Multi-University Training Contest 1 1012) 半平面交
Mow 题目链接 分析 将多边形的边向内部缩 r 个单位长度,然后这些边所围成的内部区域,就是圆心的合法范围,该范围也是一个多边形,假设面积是\(a\),周长是\(b\),那么可以知道圆可以覆盖的面积 ...
- 配置VS2013 + opencv 2.4.10
其实我内心是极力反对装这么老的版本的,但是要交课堂作业~~好无奈 [注] : 如果按照本文配置不成功,可以试一下其他博客里面的配置(多试一试总能成功的) https://jingyan.baidu.c ...
- 2019牛客暑期多校训练营(第二场)D Kth Minimum Clique(第k团)
题意:给你n个点 求第k小的团 思路:暴力bfs+bitset压位 #include <bits/stdc++.h> using namespace std; const int N = ...
- Codeforces Round #672 (Div. 2) A. Cubes Sorting (思维)
题意:有一长度为\(n\)的一组数,每次可以交换两个数的位置,问能否在\(\frac{n*(n-1)}{2}-1\)次操作内使得数组非递减. 题解:不难发现,只有当整个数组严格递减的时候,操作次数是\ ...
- Codeforces Round #670 (Div. 2) A. Subset Mex (贪心)
题意:给你一长度为\(n\)的序列,将其分为两个集合,求两个集合中未出现的最小元素的最大值, 题解:用桶存一下每个元素的个数,两次枚举\([1,100]\),找出两个最小值即可. 代码: int t; ...
- Dubbo从入门到实践
1 Dubbo出现的背景 随着互联网的发展,网站应用的规模不断扩大,常规的垂直应用架构已无法应对,分布式服务架构以及流动计算架构势在必行,亟需一个治理系统确保架构有条不紊的演进. 我们传统的网站结构为 ...
- mybatis(十一)mybatis常见问题
用注解还是用 xml 配置? 常用注解:@Insert.@Select.@Update.@Delete.@Param.@Results. @Result 在 MyBatis 的工程中,我们有两种配置 ...
- OAuth2授权流程
- Windows font-size: 10px; bug
Windows font-size: 10px; bug Windows 最小只能渲染 font-size: 12px; ???屏幕分辨率 macOS 正常渲染 10px PC 最小只能渲染 font ...