反射最大的价值就是用来写框架,下面贴出自己的3篇代码,模拟实现SPING框架的bean工厂,IOC,AOP。当然这里重点是在利用反射实现功能,为了图方便,我用的是Properties文件,关于XML后面会有专门的博客来整理,到时候整合相关的解析XML的代码就可以了。

1,通过反射,读取配置文件来管理bean

package linkin;

import java.io.FileInputStream;
import java.util.HashMap;
import java.util.Map;
import java.util.Properties; //这里就是通过反射,来获得对象的工厂
public class Linkin
{
// 定义一个对象池,前面是对象名,后面是实际对象
private Map<String ,Object> objectPool = new HashMap<String ,Object>();
// 定义一个创建对象的方法,该方法只要传入一个字符串类名,程序可以根据该类名生成Java对象
private Object createObject(String clazzName) throws Exception
{
// 根据字符串来获取对应的Class对象
Class<?> clazz = Class.forName(clazzName);
// 使用clazz对应类的默认构造器创建实例
return clazz.newInstance();
}
// 该方法根据指定文件来初始化对象池,它会根据配置文件来创建对象
public void initPool(String fileName) throws Exception
{
FileInputStream fis = null;
try
{
fis = new FileInputStream(fileName);
Properties props = new Properties();
props.load(fis);
for (String name : props.stringPropertyNames())//返回此属性列表中的键集
{
// 每取出一对key-value对,就根据value创建一个对象,调用createObject()创建对象,并将对象添加到对象池中
objectPool.put(name ,createObject(props.getProperty(name)));
}
}
catch (Exception ex)
{
System.out.println("读取" + fileName + "异常");
}
finally
{
if(null != fis)
{
fis.close();
}
} }
public Object getObject(String name)
{
// 从objectPool中取出指定name对应的对象。
return objectPool.get(name);
} public static void main(String[] args)
throws Exception
{
Linkin linkin = new Linkin();
linkin.initPool("Linkin.txt");//linkin=linkin.Linkin
System.out.println(linkin.getObject("linkin"));
}
}

2,通过反射实现IOC注入功能

package linkin;

import java.io.FileInputStream;
import java.io.IOException;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;
import java.util.Properties; public class Linkin
{
private String name; public String getName()
{
return name;
}
public void setName(String name)
{
this.name = name;
}
// 定义一个对象池,前面是对象名,后面是实际对象
private Map<String ,Object> objectPool = new HashMap<String ,Object>();
private Properties config = new Properties();
// 从指定属性文件中初始化Properties对象。
public void init(String fileName) throws Exception
{
FileInputStream fis = null;
try
{
fis = new FileInputStream(fileName);
config.load(fis);
}
catch (IOException ex)
{
System.out.println("读取" + fileName + "异常");
}
finally
{
if(null != fis){
fis.close();
}
}
}
// 定义一个创建对象的方法,该方法只要传入一个字符串类名,程序可以根据该类名生成Java对象
private Object createObject(String clazzName) throws Exception
{
// 根据字符串来获取对应的Class对象
Class<?> clazz =Class.forName(clazzName);
// 使用clazz对应类的默认构造器创建实例,这一行道出了javabean的精髓,为什么javabean规范中要就有一个默认的无参的构造器
return clazz.newInstance();
}
// 该方法根据指定文件来初始化对象池,它会根据配置文件来创建对象
public void initPool() throws Exception
{
for (String name : config.stringPropertyNames())
{
// 每取出一对key-value对,如果key中不包含百分号(%)
// 这个就标明是根据value来创建一个对象
// 调用createObject创建对象,并将对象添加到对象池中
if (!name.contains("%"))
{
objectPool.put(name , createObject(config.getProperty(name)));
}
}
}
// 该方法根据指定文件来初始化对象池,它会根据配置文件来创建对象
public void initProperty()throws Exception
{
for (String name : config.stringPropertyNames())
{
// 每取出一对key-value对,如果key中包含百分号(%)
// 即可认为该key是用于为对象的Field设置值,
// %前半为对象名字,后半为Field名
// 程序将调用对应的setter方法来为对应Field设置值。
if (name.contains("%"))
{
// 将配置文件中key按%分割
String[] objAndProp = name.split("%");
// 取出需要设置Field值的目标对象
Object target = getObject(objAndProp[0]);
// 该Field对应的setter方法名:set + "属性的首字母大写" + 剩下部分
String mtdName = "set" + objAndProp[1].substring(0 , 1).toUpperCase() + objAndProp[1].substring(1);
// 通过target的getClass()获取它实现类所对应的Class对象
Class<?> targetClass = target.getClass();
// 获取该属性对应的setter方法,下面这一行道出了springIOC的精髓,为什么实现XML我们每次都要提供get和set方法,除了注解的哦
Method mtd = targetClass.getMethod(mtdName , String.class);
// 通过Method的invoke方法执行setter方法,将config.getProperty(name)的属性值作为调用setter的方法的实参
mtd.invoke(target , config.getProperty(name));
}
}
}
public Object getObject(String name)
{
// 从objectPool中取出指定name对应的对象。
return objectPool.get(name);
}
public static void main(String[] args)throws Exception
{
Linkin linkin = new Linkin();
linkin.init("Linkin.txt");//linkin=linkin.Linkin
//linkin%name=LinkinPark...
linkin.initPool();
linkin.initProperty();
linkin = (Linkin) linkin.getObject("linkin");
System.out.println(linkin.getName());
}
}

3,通过反射实现AOP框架,并且跑了代理,(人为可以控制跑不跑代理)。

package linkin;

import java.io.FileInputStream;
import java.io.IOException;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.HashMap;
import java.util.Map;
import java.util.Properties; public class Linkin implements Person
{
private String name; public String getName()
{
return name;
}
public void setName(String name)
{
this.name = name;
}
//定义这个方法来作为测试
public void test()
{
System.out.println("这里是原始的方法");
}
// 定义一个对象池,前面是对象名,后面是实际对象
private Map<String ,Object> objectPool = new HashMap<String ,Object>();
private Properties config = new Properties();
// 从指定属性文件中初始化Properties对象。
public void init(String fileName) throws Exception
{
FileInputStream fis = null;
try
{
fis = new FileInputStream(fileName);
config.load(fis);
}
catch (IOException ex)
{
System.out.println("读取" + fileName + "异常");
}
finally
{
if(null != fis){
fis.close();
}
}
}
// 定义一个创建对象的方法,该方法只要传入一个字符串类名,程序可以根据该类名生成Java对象
private Object createObject(String clazzName) throws Exception
{
// 根据字符串来获取对应的Class对象
Class<?> clazz = Class.forName(clazzName);
// 使用clazz对应类的默认构造器创建实例,这一行道出了javabean的精髓,为什么javabean规范中要就有一个默认的无参的构造器
return clazz.newInstance();
}
// 该方法根据指定文件来初始化对象池,它会根据配置文件来创建对象
public void initPool() throws Exception
{
for (String name : config.stringPropertyNames())
{
// 每取出一对key-value对,如果key中不包含百分号(%),这个就标明是根据value来创建一个对象,调用createObject创建对象,并将对象添加到对象池中
// 如果他是我们自己定义的LinkinProxy,就说明这里是跑代理的,不然就是不跑代理的
String className = config.getProperty(name);
if (!name.contains("%") && !name.contains("."))
{
Object bean = createObject(className);
if(bean instanceof LinkinProxy)
{
LinkinProxy linkinProxyBean = (LinkinProxy) bean;
LinkinAdvice linkinAdvice = (LinkinAdvice) createObject(config.getProperty(name + ".advice"));
Object target = createObject(config.getProperty(name + ".target"));
initProperty(target);
linkinProxyBean.setLinkinAdvice(linkinAdvice);
linkinProxyBean.setTarget(target);
bean = linkinProxyBean.getProxyBean();
}
else
{
initProperty(bean);
}
objectPool.put(name , bean);
}
}
}
// 该方法根据指定文件来初始化对象池,它会根据配置文件来创建对象
public void initProperty(Object target)throws Exception
{
for (String name : config.stringPropertyNames())
{
// 每取出一对key-value对,如果key中包含百分号(%)
// 即可认为该key是用于为对象的Field设置值,
// %前半为对象名字,后半为Field名
// 程序将调用对应的setter方法来为对应Field设置值。
if (name.contains("%"))
{
// 将配置文件中key按%分割
String[] objAndProp = name.split("%");
// 取出需要设置Field值的目标对象,这里加一个控制,如果他是代理的话,就不能从池里面去取le
//Object target = getObject(objAndProp[0]);
// 该Field对应的setter方法名:set + "属性的首字母大写" + 剩下部分
String mtdName = "set" + objAndProp[1].substring(0 , 1).toUpperCase() + objAndProp[1].substring(1);
// 通过target的getClass()获取它实现类所对应的Class对象。
Class<?> targetClass = target.getClass();
// 获取该属性对应的setter方法,下面这一行道出了springIOC的精髓,为什么实现XML我们每次都要提供get和set方法,除了注解的哦
Method mtd = targetClass.getMethod(mtdName , String.class);
// 通过Method的invoke方法执行setter方法,将config.getProperty(name)的属性值作为调用setter的方法的实参
mtd.invoke(target , config.getProperty(name));
}
}
}
public Object getObject(String name)
{
// 从objectPool中取出指定name对应的对象。
return objectPool.get(name);
}
public static void main(String[] args)throws Exception
{
Linkin linkin = new Linkin();
linkin.init("Linkin.txt");
linkin.initPool();
Person huhu = (Person) linkin.getObject("XXX");
System.out.println(huhu.getClass());
huhu.test();
}
} class LinkinProxy
{
private Object target;//原始对象
private LinkinAdvice linkinAdvice;//服务 public LinkinAdvice getLinkinAdvice()
{
return linkinAdvice;
} public void setLinkinAdvice(LinkinAdvice linkinAdvice)
{
this.linkinAdvice = linkinAdvice;
} public Object getTarget()
{
return target;
} public void setTarget(Object target)
{
this.target = target;
} public Object getProxyBean()
{
LinkinInvocationHandler linkinInvocationHandler = new LinkinInvocationHandler();
linkinInvocationHandler.setTarget(target);
linkinInvocationHandler.setLinkinAdvice(linkinAdvice);
Object proxyBean = Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(),
linkinInvocationHandler);
return proxyBean;
}
} //定义接口
interface Person
{
void test();
}
interface LinkinAdvice
{
public void test();
public void test1();
}
//定义服务1
class LinkinAdvice1 implements LinkinAdvice
{
public void test()
{
System.out.println("这里是第1个通用的方法。。。");
}
public void test1()
{
System.out.println("这里是第2个通用的方法");
}
}
//定义服务2
class LinkinAdvice2 implements LinkinAdvice
{
public void test()
{
System.out.println("这里是第3个通用的方法。。。");
}
public void test1()
{
System.out.println("这里是第4个通用的方法");
}
} class LinkinInvocationHandler implements InvocationHandler
{
private Object target; private LinkinAdvice linkinAdvice;//服务 public Object getTarget()
{
return target;
} public void setTarget(Object target)
{
this.target = target;
} public LinkinAdvice getLinkinAdvice()
{
return linkinAdvice;
} public void setLinkinAdvice(LinkinAdvice linkinAdvice)
{
this.linkinAdvice = linkinAdvice;
} @Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable
{
linkinAdvice.test();
Object result = method.invoke(target, args);
linkinAdvice.test1();
return result;
} }
#一下配置文件的说明:要是直接生成原始对象就直接指定全限定类名就好了,要是想跑代理就指定下面那个名字,然后设置服务和原始对象#
#下面的属性注入模拟了IOC功能,不管是代理对象还是原始对象,都可以把这个属性值注入,原始对象就是注入原始对象,代理对象就是注入到被代理的那个原始对象中#
#XXX=linkin.Linkin
##以这个作为控制,要是得到的类放完反射属于这个类,那么就认为是要代理,不然就是不代理#
XXX=linkin.LinkinProxy
#模拟了属性注入,XML是通过自己的标签,这里我自己使用“%”模拟了下#
XXX%name=LinkinPark...
#这里是代理对象的服务,也就是AOP中插入的通用服务代码,可以随便切换,自己定义个服务,这里配上就可以用了#
XXX.advice=linkin.LinkinAdvice2
#这里就是上面那个代理类的原始对象#
XXX.target=linkin.Linkin



反射应用--IOC和AOP的更多相关文章

  1. Spring的IOC和AOP之深剖

    今天,既然讲到了Spring 的IOC和AOP,我们就必须要知道 Spring主要是两件事: 1.开发Bean:2.配置Bean.对于Spring框架来说,它要做的,就是根据配置文件来创建bean实例 ...

  2. spring - ioc和aop

    1.程序中为什么会用到spring的ioc和aop 2.什么是IOC,AOP,以及使用它们的好处,即详细回答了第一个问题 3.原理 关于1: a:我们平常使用对象的时候,一般都是直接使用关键字类new ...

  3. 【转】spring - ioc和aop

    [转]spring - ioc和aop 1.程序中为什么会用到spring的ioc和aop 2.什么是IOC,AOP,以及使用它们的好处,即详细回答了第一个问题 3.原理 关于1: a:我们平常使用对 ...

  4. Spring 的IOC和AOP总结

    Spring 的IOC和AOP IOC 1.IOC 许多应用都是通过彼此间的相互合作来实现业务逻辑的,如类A要调用类B的方法,以前我们都是在类A中,通过自身new一个类B,然后在调用类B的方法,现在我 ...

  5. Spring IOC、AOP、Transaction、MVC小结

    1.IOC.AOP:把对象交给Spring进行管理,通过面向切面编程来实现一些“模板式”的操作,使得程序员解放出来,可以更多的关注业务实现.                             - ...

  6. 【Java】模拟Sping,实现其IOC和AOP核心(二)

    接着上一篇,在上一篇完成了有关IOC的注解实现,这一篇用XML的方式实现IOC,并且完成AOP. 简易的IOC框图 注解的方式实现了左边的分支,那么就剩下右边的XML分支: XmlContext:这个 ...

  7. 【Java】模拟Sping,实现其IOC和AOP核心(一)

    在这里我要实现的是Spring的IOC和AOP的核心,而且有关IOC的实现,注解+XML能混合使用! 参考资料: IOC:控制反转(Inversion of Control,缩写为IoC),是面向对象 ...

  8. Spring IOC AOP的原理 如果让你自己设计IOC,AOP如何处理(百度)

    百度的面试官问,如果让你自己设计一个IOC,和AOP,如何设计, 我把IOC的过程答出来了,但是明显不对, (1) IOC 利用了反射,自己有个id,classtype,hashmap,所有的功能都在 ...

  9. 详谈 Spring 中的 IOC 和 AOP

    这篇文章主要讲 Spring 中的几个点,Spring 中的 IOC,AOP,下一篇说说 Spring 中的事务操作,注解和 XML 配置. Spring 简介 Spring 是一个开源的轻量级的企业 ...

随机推荐

  1. Java学习笔记8(面向对象一:概念、private)

    面向过程的思想:遇到问题,想,我该怎么解决这个问题?然后一步一步解决 面向对象的思想:遇到一件事的时候,思考,我该让谁来做,至于他怎样去做,不是我需要考虑的事情,只要最后做好就行 实际举例:我们要组装 ...

  2. 表迁移工具的选型-xtrabackup的使用

    1.1. 场景 有的时候test人员可能需要在测试库上比较新的数据,这时候只能是从生产库上面去那了.如果是小表还好实用mysqldump/mysqlpump就可以轻松的解决.但是,如果遇到了大表这将是 ...

  3. CubeMX使用及感受

    简介 CubeMX这几年刚流行起来,是一个STM32代码的初始化配置工具,里面封装了硬件层.中间层,以及示例代码. cube使用 该软件的安装需要较高版本jdk支持,固件库安装时需要注意和主程序的版本 ...

  4. POI读取excel工具类(xls,xlsx通用)

    package com.boot.utils; import java.io.File; import java.io.FileInputStream; import java.io.FileNotF ...

  5. LeetCode第[16]题(Java):3Sum Closest 标签:Array

    题目难度:Medium 题目: Given an array S of n integers, find three integers in S such that the sum is closes ...

  6. Java关于BufferedWriter.newline()换行的注意事项

    最近项目中需要导出文件,其实导出文件是一个挺简单的事情.但是却遇到了很奇怪的问题. 首先导出到文件需要用到 BufferedWriter.而换行则是通过 bw.newline() 方法,问题将出在 n ...

  7. Java--高效的定时任务设计

    相信你在日常的开发中肯定遇到过这种问题: 需要对实体类的状态信息进行管理,比如一定时间后修改它为XXX状态. 举个例子: 订单服务,当用户提交了订单后,如果在30分钟内没有支付,自动取消订单,这就是一 ...

  8. JDK源码 - ArrayList

    /** * ArrayList源码分析 * @author liyong * */ public class Util { @SuppressWarnings("unchecked" ...

  9. LVS集群之NAT模式实例(3)

    LVS集群NAT模式实例 1. 实验拓扑图 DS 必须有两块网卡,需要在上面做NAT. 2. 实验环境 3台CentOS6.4 64bit的服务器. 类型 IP DR eth0:10.20.73.20 ...

  10. BZOJ:4873: [Shoi2017]寿司餐厅

    4873: [Shoi2017]寿司餐厅 首先很开心在膜你赛的时候做了出来. 看到数据范围,看到不能dp,看到贡献去重后计算,咦,流? 那就容易了,转最大权闭合子图,每个区间建一个点,取了就一定要取他 ...