Java反射-简单应用
为了程序更好的维护和扩展,在面向对象思维的世界里,首先是面向接口编程,然后我们应该把做什么和怎么做进行分离。
以下我将用一个开晚会的样例来演示一下,终于达到的效果是:工厂+反射+配置文件实现程序的灵活应用。会详细说明一下这个过程是怎么来的,明确了这个,就会对反射和配置文件的结合更加深刻一些。
想要实现的功能是:晚会有一个唱歌、舞蹈、小品的节目单,详细各个节目的表演者仅仅须要一个就能够,每个表演接口都有两个实现类(表演者)。通过client调用不同的实现类来实现不同的节目单。表演者就是“做什么”,那么“怎么做”就是节目单了。首先来看看类图结构:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvd2xjY29tZW9u/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">
以下放代码,首先是接口,即各类表演:
<span style="font-size:14px;">/**跳舞接口*/
</span><pre name="code" class="java">public interface Dancer {
public void dance();} /**小品接口*/public interface Performer { public void performance();}/**歌唱接口*/ public interface Singer { public void sing();}
然后是各个实现类。即表演者:
Dancer接口实现类:
public class YangLiPing implements Dancer { @Override
public void dance() {
System.out.println("杨丽萍跳舞:孔雀舞");
} } public class XiaoHuDui implements Dancer { @Override
public void dance() {
System.out.println("小虎队跳舞:霹雳舞");
} }
Performer接口实现类:
public class GongHanLin implements Performer { @Override
public void performance() {
System.out.println("巩汉林表演:功夫令");
} }
public class ZhaoBenShan implements Performer { @Override
public void performance() {
System.out.println("赵本山表演:卖拐");
} }
Singer接口实现类:
public class ZhouHuaJian implements Singer { @Override
public void sing() {
System.out.println("周华健演唱:刀剑如梦");
} }
public class WangFei implements Singer { @Override
public void sing() {
System.out.println("王菲演唱:我愿意");
} }
client调用方式:
<span style="font-size:14px;"> public static void main(String[] args) {
//定义晚会流程
//演出:歌曲、舞蹈、表演
//第1种方法:单纯使用多态
System.out.println("晚会開始=======>>");
Singer singer = new ZhouHuaJian();
singer.sing();
Performer performer = new ZhaoBenShan();
performer.performance();
Dancer dancer = new YangLiPing();
dancer.dance();
System.out.println("<<========晚会结束");
}</span>
代码挺简单,只是发现一个问题。作为节目组织者,在现实中基本上是不应该直接与表演者打交道的,而是与其所在的公司或者代理人进行交流,如果如今全部的表演者都同属于一个娱乐公司,那么我仅仅须要从这个娱乐公司里获得我想要的那些表演者就能够了,那么UML图将会变为例如以下结构(Factory就代表这个娱乐公司):
Facotry代码:
public classFactory {
//提供准备歌手方法
public static Singer getSinger(){
return new ZhouHuaJian();
}
//提供准备舞蹈方法
public static Dancer getDancer(){
return new YangLiPing();
}
//提供准备表演方法
public static PerformergetPerformer(){
return new GongHanLin();
}
}
那么client代码就变成了这样:
public staticvoidmain(String[] args) {
//定义晚会流程
//演出:歌曲、小品、舞蹈
//第1种方法:使用工厂获取各个表演种类
System.out.println("晚会開始=======>>");
Singersinger = Factory.getSinger();
singer.sing();
Performerperformer = Factory.getPerformer();
performer.performance();
Dancerdancer = Factory.getDancer();
dancer.dance();
System.out.println("<<========晚会结束");
}
大家能够看到,在Factory代码中我们发现还是将各个实现类写死了,也就是说娱乐公司强制某些表演者必须參加,但实际上非常多时候有些表演者可能没法參加,而另外一些闲着的表演者则能够參加。那么我是不是应该在须要表演者的时候动态获取有空參加更好一些呢?这个时候就须要用上配置文件+反射了。
Factory代码更改例如以下:
<span style="font-size:14px;">public class Factory {
/**提供准备歌手方法*/
public static Singer getSinger(){
//读取配置文件及获取key相应的value
StringclassName = ResourceBundle.getBundle("party").getString("Singer");
Singersinger = null;
try {
//将配置文件里读取的完整类名通过反射获取该类的实例
singer= (Singer)Class.forName(className).newInstance();
}catch(InstantiationException | IllegalAccessException
|ClassNotFoundException e) {
e.printStackTrace();
}
return singer;
}
/**提供准备舞蹈方法*/
public static Dancer getDancer(){
StringclassName = ResourceBundle.getBundle("party").getString("Dancer");
Dancerdancer = null;
try {
dancer= (Dancer)Class.forName(className).newInstance();
}catch(InstantiationException | IllegalAccessException
|ClassNotFoundException e) {
e.printStackTrace();
}
return dancer;
}
/**提供准备表演方法*/
public static PerformergetPerformer(){
StringclassName = ResourceBundle.getBundle("party").getString("Performer");
Performerperformer = null;
try {
performer= (Performer)Class.forName(className).newInstance();
}catch(InstantiationException | IllegalAccessException
|ClassNotFoundException e) {
e.printStackTrace();
}
return performer;
}
}</span>
配置文件party.properties中的代码:
Singer = com.lc.reflect.demo3.person.WangFei
Performer = com.lc.reflect.demo3.person.ZhaoBenShan
Dancer =com.lc.reflect.demo3.person.XiaoHuDui
这样就能把代码解耦了。
细致看的话,在Factory中的代码3个方法基本上没啥不同,除了变量和对象不同。发现了代码的坏味道。那就重构吧:
public classFactory1 {
/**提供准备歌手方法*/
public static Singer getSinger(){
Singersinger = null;
return (Singer)getObject("Singer",singer);
}
/**提供准备舞蹈方法*/
public static Dancer getDancer(){
Dancerdancer = null;
return (Dancer)getObject("Dancer",dancer);
}
/**提供准备表演方法*/
public static PerformergetPerformer(){
Performerperformer = null;
return (Performer)getObject("Performer",performer);
}
/**
* 获取目标对象的方法
* @param objName 对象名称
* @param obj 对象类型
* @return对象
*/
public static Object getObject(StringobjName,Object obj){
//获取配置文件里的目标对象路径
StringclassName = ResourceBundle.getBundle("party").getString(objName);
try {
//获取目标对象实例
obj= Class.forName(className).newInstance();
}catch(InstantiationException | IllegalAccessException
|ClassNotFoundException e) {
e.printStackTrace();
}
return obj;
}
}
这样就简洁多了。
学习一块知识不只要学习API是怎么用的,还要与生活相结合,做出一个个简单有效的Demo,同一时候在做Demo的过程中,假设发现了编写的代码有值得重构的地方,一定不要停下思考的脚步,Just do it!尽管对反射的详细实现原理和过程还不太清楚,但首先会用才干有继续研究的动力。继续努力中~~
Java反射-简单应用的更多相关文章
- Java反射+简单工厂模式总结
除了 new 之外的创建对象的方法 通过 new 创建对象,会使得程序面向实现编程,先举个例子,某个果园里现在有两种水果,一种是苹果,一种是香蕉,有客户想采摘园子里的水果,要求用get()方法表示即可 ...
- Java反射 - 简单的给Bean赋值和取值
由于项目的实际需要,所以利用java反射原理写了一个简单给bean赋值和取值通用的类,在此记录下方便自己日后用到,也为需要的兄弟提供个参考例子. 工具类BeanRefUtil: package c ...
- Java反射简单使用--第一次细致阅读底层代码
1:所写的东西都经过验证,保证正确,环境jdk8,eclipse2:在例子中,尽量以生产环境中实际代码为例,那种固定值什么的没什么意义 问题: 1:想获取调用方法所需要的参数 2:参数是以json形式 ...
- Java 反射(简单捋一下)
有Student类,Person类,还有一个叫Class的类,这是反射的源头. 正常方式:通过完整的类名 > 通过new实例化 > 取得实例化对象 反射方式:实例化对象 > getC ...
- java反射简单实例
这篇博友的总结的反射知识点是比较全面的 http://www.cnblogs.com/rollenholt/archive/2011/09/02/2163758.html 下面介绍我用反射做的两个功能 ...
- java反射机制简单实例
目录 Java反射 简单实例 @(目录) Java反射 Java语言允许通过程序化的方式间接对Class进行操作.Class文件由类装载器装载后,在JVM中将形成一份描述Class结构的元信息对象,通 ...
- java反射机制的简单介绍
参考博客: https://blog.csdn.net/mlc1218559742/article/details/52754310 先给出反射机制中常用的几个方法: Class.forName (& ...
- java反射并不是什么高深技术,面向对象语言都有这个功能,而且功能也很简单,就是利用jvm动态加载时生成的class对象
java反射并不是什么高深技术,面向对象语言都有这个功能. 面向对象语言都有这个功能,而且功能也很简单,就是利用jvm动态加载时生成的class对象,去获取类相关的信息 2.利用java反射可以调用类 ...
- Java反射(六)纯面向接口编程的简单框架实践
我们知道在使用MyBatis开发时,只需要添加DAO接口和对应的映射XML文件,不需要写DAO的实现类,其实底层是通过动态代理实现. 本文将使用前几篇文章的知识点实现一个纯面向接口编程的简单框架,与M ...
随机推荐
- arrowTip 提示
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- python sys.exit()函数说明
sys.exit()函数是通过抛出异常的方式来终止进程的,也就是说如果它抛出来的异常被捕捉到了的话程序就不会退出了. #!/usr/bin/python #!coding:utf-8 import s ...
- jquery validation plugin 使用
<!DOCTYPE html> <!-- To change this license header, choose License Headers in Project Prope ...
- 【Leetcode】Set Matrix Zeroes
给定一个m x n的矩阵,如果某个元素为0,则把该元素所在行和列全部置0. Given a m x n matrix, if an element is 0, set its entire row a ...
- Android设计中的.9.png
在Android的设计过程中,为了适配不同的手机分辨率,图片大多需要拉伸或者压缩,这样就出现了可以任意调整大小的一种图片格式“.9.png”.这种图片是用于Android开发的一种特殊的图片格式,它的 ...
- CodeFirst EF中导航属性的个人理解
>导航属性: 01.个人理解就是Ef中的属性在实体数据表中不存在(先这么认为); 02.就是除了根据表中列映射出的属性 之外根据表与表之间的关系的关联属性.方便操作与之关联的表; 例如: 有 表 ...
- C# DataTable的用法
建表DataTable dtP = new DataTable("Man");加列DataColumn dcP;dcP = new DataColumn("Name&qu ...
- 2014.12.01 B/S 使用VS建立Web网站
要求:从hr数据库info表读取数据,在Web网站中显示为如图: 用DW绘制一个表格,然后将代码拷贝到新建的网站主页代码中 <div> <table bgcolor=" w ...
- iOS技术
iOS技术 OC:分类(好处,和延展的区别) 分类: 在不修改原有的类的基础上增加新的方法 一个庞大的类可以分模块开发 一个庞大的类可以由多个人来编写,更有利于团队合作 分类是对原有类的一种扩展,在 ...
- 浅谈C中的指针和数组(三)
上一个博客我们得到了一个结论: 指针和数组根本就是两个完全不一样的东西.只是它们都可以“以指针形式”或“以下标形式”进行访问.一个是完全的匿名访问,一个是典型的具名+匿名访问.一定要注意的是这个“以X ...