作者: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();
}

完整代码

Github

参考资料

小马哥讲Spring核心编程思想

PropertyEditorSupport的使用

Java SPI机制 - ServiceLoader

02_Java通信_JNDI_demo1

JavaSE实现IoC的更多相关文章

  1. Spring之IoC总结帖

    Spring是一个开源框架,Spring是于2003 年兴起的一个轻量级的Java 开发框架,由Rod Johnson 在其著作Expert One-On-One J2EE Development a ...

  2. Spring 实践 -IoC

    Spring 实践 标签: Java与设计模式 Spring简介 Spring是分层的JavaSE/EE Full-Stack轻量级开源框架.以IoC(Inverse of Control 控制反转) ...

  3. Spring框架之IOC(控制反转)

    [TOC] 第一章Spring框架简介 IOC(控制反转)和AOP(面向方面编程)作为Spring框架的两个核心,很好地实现了解耦合.所以,简单来说,Spring是一个轻量级的控制反转(IoC)和面向 ...

  4. ioc(Inversion of Control)控制反转和DI

    ioc意味着将你设计好的交给容器控制,而不是传统在你的对象中直接控制 谁控制了谁:传统的javaSE程序设计,我们直接在对象内部通过new进行创建对象,是程序主动去创建依赖对象:而ioc是有专门一个容 ...

  5. 框架学习之Spring(一IOC)----HelloWrod

    一.概述 Spring是一个开源框架,它的核心是控制反转(IOC)和面向切面(AOP).简单来说,Spring是一个分层的JavaSE/EEfull-stack(一站式)轻量级开源框架. EE 开发分 ...

  6. Spring第一天——入门与IOC

    大致内容 spring基本概念 IOC入门 [17.6.9更新],如何学习spring? 掌握用法 深入理解 不断实践 反复总结 再次深入理解与实践 一.Spring相关概念  1.概述: Sprin ...

  7. Spring框架-IOC和AOP简单总结

    参考博客: https://blog.csdn.net/qq_22583741/article/details/79589910 1.Spring框架是什么,为什么,怎么用 1.1 Spring框架是 ...

  8. Spring系列之IOC的原理及手动实现

    目录 Spring系列之IOC的原理及手动实现 Spring系列之DI的原理及手动实现 导语 Spring是一个分层的JavaSE/EE full-stack(一站式) 轻量级开源框架.也是几乎所有J ...

  9. Spring 框架学习—控制反转(IOC)

        Spring是一个开源框架,Spring是于2003 年兴起的一个轻量级的Java 开发框架,由Rod Johnson创建. 简单来说,Spring是一个分层的JavaSE/EEfull-st ...

随机推荐

  1. 校招有感:计算机专业毕业生如何找工作(Java方向)

    这两年毕业生的就业形势,大家都看在眼里.由于我平时本职会到校园去做校招,兼职会做培训,所以也接触到了不少计算机应届毕业生. 我看到的计算机专业毕业生,各种情况的都有.虽然其中有学校的因素,但学校的因素 ...

  2. Codeforces Round #585 (Div. 2) E. Marbles(状压dp)

    题意:给你一个长度为n的序列 问你需要多少次两两交换 可以让相同的数字在一个区间段 思路:我们可以预处理一个数组cnt[i][j]表示把i放到j前面需要交换多少次 然后二进制枚举后 每次选择一个为1的 ...

  3. 3. Linear Regression with Multiple Variables

    前面还有一章主要讲解,基本的Linear Algebra线性代数的知识,都比较简单,这里就直接跳过了. Speaker: Andrew Ng 1. Multiple featues 训练集的特征变成了 ...

  4. codeforces628D. Magic Numbers (数位dp)

    Consider the decimal presentation of an integer. Let's call a number d-magic if digit d appears in d ...

  5. 洛谷 P1429 平面最近点对(加强版) (分治模板题)

    题意:有\(n\)个点对,找到它们之间的最短距离. 题解:我们先对所有点对以\(x\)的大小进行排序,然后分治,每次左右二等分递归下去,当\(l+1=r\)的时候,我们计算一下距离直接返回给上一层,若 ...

  6. kubernetes实战-配置中心(二)交付apollo配置中心到k8s

    apollo官网:官方地址 apollo架构图: apollo需要使用数据库,这里使用mysql,注意版本需要在5.6以上: 本次环境mysql部署在10.4.7.11上,使用mariadb:10.1 ...

  7. 字节笔试题 leetcode 69. x 的平方根

    更多精彩文章请关注公众号:TanLiuYi00 题目 解题思路 题目要求非负整数 x 的平方根,相当于求函数 y = √x 中 y 的值. 函数 y = √x  图像如下: 从上图中,可以看出函数是单 ...

  8. 1076D Edge Deletion 【最短路】

    题目:戳这里 题意:求出1到所有点的最短路径后,把边减到小于等于k条,问保留哪些边可以使仍存在的最短路径最多. 解题思路:这题就是考求最短路的原理.比如dijkstra,用优先队列优化后存在队列中的前 ...

  9. 如何实现一个简易版的 Spring - 如何实现 @Component 注解

    前言 前面两篇文章(如何实现一个简易版的 Spring - 如何实现 Setter 注入.如何实现一个简易版的 Spring - 如何实现 Constructor 注入)介绍的都是基于 XML 配置文 ...

  10. LWIP再探----内存池管理

    这这里是接上一篇内存池管理部分的,这里如果读者一打开memp.c的话会感觉特别那一理解原作者在干嘛,但是看懂了就明白原作者是怎么巧妙的使用了宏.废话不多说先说了下我分析是一下宏的条件是 前提条件MEM ...