有段时间没来写博客了,心里一直念叨空了来,今天有时间来记录一篇。前段时间领导提出优化部分系统模块,根据业务要求系统中有很多产品,产品下面有N个指标,一个指标就对应一个方法,所以系统代码中就是这样一个情况:一个产品下面会写很多调用的方法,这些方法其他产品也可以调用的,抽象出来的。然后我就想到了Java反射,通过反射机制java执行方法,通过数据库配置达到灵活调用,不管以后增加产品还是增删指标方法,都可以不用修改Java代码直接数据库配置就行了,话不多说,上干货。


ClassData.java  用于配置
 public class ClassData {

     //java类名称
private String packages; //方法名
private String className;
//方法需要的参数类型名
private String parameter1;
//方法需要的参数类型名
private String parameter2;
//这个方法得出的值用什么名字接收 比如 User类的name属性 这个值就是name
private String names; public String getNames() {
return names;
} public void setNames(String names) {
this.names = names;
} public String getPackages() {
return packages;
} public void setPackages(String packages) {
this.packages = packages;
} public String getClassName() {
return className;
} public void setClassName(String className) {
this.className = className;
} public String getParameter1() {
return parameter1;
} public void setParameter1(String parameter1) {
this.parameter1 = parameter1;
} public String getParameter2() {
return parameter2;
} public void setParameter2(String parameter2) {
this.parameter2 = parameter2;
}

模拟数据 这里我是直接写的为了方便  最好是建表配置在数据库 从数据库得到相应的集合list

public static List<ClassData> getlist(){
List<ClassData> list = new ArrayList<ClassData>();
ClassData c = new ClassData();

  

c.setPackages("com.cq.test.clas.ClassVo");
      c.setClassName("getNameVal");
      c.setParameter1("java.lang.String");
      c.setNames("name");
      ClassData c1 = new ClassData();
      c1.setPackages("com.cq.test.clas.ClassVo");
      c1.setClassName("getAgeVal");
      c1.setParameter1("java.lang.String");
      c1.setParameter2("java.lang.String");
      c1.setNames("age");
      ClassData c2 = new ClassData();
      c2.setPackages("com.cq.test.clas.ClassVo");
      c2.setClassName("getDateVal");
      c2.setParameter1("java.lang.String");
      c2.setNames("date");
      ClassData c3 = new ClassData();
      c3.setPackages("com.cq.test.clas.ClassVo");
      c3.setClassName("getMyVal");
      c3.setParameter1("java.lang.String");
      c3.setNames("my");

      list.add(c);

        list.add(c1);
list.add(c2);
list.add(c3);
return list;
}

  

  user.java

package com.cq.test.vo;

import java.math.BigDecimal;
import java.util.Date; public class User { private String name; private Integer age; private Date date; private BigDecimal my; public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Integer getAge() {
return age;
} public void setAge(Integer age) {
this.age = age;
} public Date getDate() {
return date;
} public void setDate(Date date) {
this.date = date;
} public BigDecimal getMy() {
return my;
} public void setMy(BigDecimal my) {
this.my = my;
} @Override
public String toString() {
return "User [name=" + name + ", age=" + age + ", date=" + date
+ ", my=" + my + "]";
} }

ClassVo.java 这是上面配置的java类

package com.cq.test.clas;

import java.math.BigDecimal;
import java.util.Date; public class ClassVo { public String getNameVal(String name){ System.out.println("名字:"+ name); return "名字"+name;
} public Integer getAgeVal(String age){ System.out.println("年龄:"+ (age)); return 18;
} public Date getDateVal(String date){ System.out.println("时间:"+ date); return new Date();
} public BigDecimal getMyVal(String bigg){
BigDecimal big = new BigDecimal(100);
System.out.println("钱:"+ big); return big;
} }

 给实体类属性赋值的方法

public static <T> T modelTrim(T model,String names,Object invoke){
Class<T> clazz = (Class<T>) model.getClass();
//获取所有的bean中所有的成员变量
Field[] fields = clazz.getDeclaredFields();
for(int j=0;j<fields.length;j++){
if(fields[j].getName().equals(names)){
//获取所有的bean中变量类型为String的变量
//if("Integer".equals(fields[j].getType().getSimpleName())){
try {
if(invoke != null && !"".equals(invoke)){
//获取set方法名
String setMethodName = "set"+fields[j].getName().substring(0, 1).toUpperCase()
+fields[j].getName().replaceFirst("\\w", "");
//得到get方法的Method对象,带参数
Method setMethod = clazz.getDeclaredMethod(setMethodName,fields[j].getType());
setMethod.setAccessible(true);
//赋值
setMethod.invoke(model, (Object)(invoke));
}
} catch (Exception e) {
e.printStackTrace();
}
//}
}
}
System.out.println("model--"+model.toString());
return model;
}

 执行main方法

public class MethodTest {
public static void main(String[] args){ List<ClassData> getlist = getlist();
int i = 1;
User user = new User(); System.out.println(user.getName());
for(ClassData cd:getlist){
try {
Class<?> userClass = Class.forName(cd.getPackages());
Object object = userClass.newInstance();
Method refTest1 = userClass.getDeclaredMethod(cd.getClassName(),Class.forName(cd.getParameter1()));
Object invoke = refTest1.invoke(object, "1");
i++;
System.out.println("执行前"+user.toString()+"invoke:"+invoke);
modelTrim(user,cd.getNames(),invoke); System.out.println("执行后"+user.toString()+"\n"); } catch (Exception e) {
e.printStackTrace();
}
}
}
}

得到结果

至此通过Java反射执行方法就完成了,是不是相当灵活,对以后提供了便捷。

使用Java反射优化多个方法调用的更多相关文章

  1. Java 反射 Method的invoke回调调用任意方法

    Java 反射 Method的invoke回调调用任意方法 @author ixenos 关键子:Method.Field.invoke方法指针/函数指针.回调函数 invoke回调流程示例 0.由C ...

  2. Atitit.通过null 参数 反射  动态反推方法调用

    Atitit.通过null 参数 反射  动态反推方法调用 此时,直接使用java  apache的ref工具都失效了.必须要自己实现了. 如果调用接口方法的话,就不能使用apache的ref工具,可 ...

  3. java反射构建对象和方法的反射调用

    Java反射技术应用广泛,其能够配置:类的全限定名,方法和参数,完成对象的初始化,设置是反射某些方法.可以增强java的可配置性. 1.1 通过反射构建对象(无参数): 例如我们使用 ReflectS ...

  4. java反射之获取所有方法及其注解(包括实现的接口上的注解),获取各种标识符备忘

    java反射之获取类或接口上的所有方法及其注解(包括实现的接口上的注解) /** * 获取类或接口上的所有方法及方法上的注解(包括方法实现上的注解以及接口上的注解),最完整的工具类,没有现成的工具类 ...

  5. 深入理解java虚拟机(十一) 方法调用-解析调用与分派调用

    方法调用过程是指确定被调用方法的版本(即调用哪一个方法),并不包括方法执行过程.我们知道,Class 文件的编译过程中并不包括传统编译中的连接步骤,一切方法调用在 Class 文件调用里面存储的都只是 ...

  6. Java APi 之 RMI远程方法调用

    一.什么是RPC RPC全称是remote procedure call,即远程过程调用.它是一种协议,用于从远程计算机上请求服务. 例如有两台服务器A和B,A上的应用想要调用B上应用的方法,但是他们 ...

  7. Java反射机制的使用方法

    Java的反射机制同意你在程序执行的过程中获取类定义的细节.有时候在程序执行的时候才得知要调用哪个方法,这时候反射机制就派上用场了. 获取类 类的获取方法有下面几种: forName().通过Clas ...

  8. java反射与多态(父类调用子类)的代码演示

    package Test0817; import java.lang.reflect.InvocationTargetException;import java.lang.reflect.Method ...

  9. JAVA反射优化

    ****************** 转自 https://my.oschina.net/19921228/blog/3042643 *********************** 比较反射与正常实例 ...

随机推荐

  1. CDN边缘节点容器调度实践(下)

    5月27日,OSC 源创会在上海成功举办.又拍云系统开发高级工程师黄励博在大会分享了<CDN 边缘节点容器调度的实践>.主要介绍又拍云自主开发的边缘节点容器调度方案,从 0 到 1 ,实现 ...

  2. Linux 文件权限于目录配置

    用户与用户组 我們以王三毛為例,王三毛這個『檔案』的擁有者為王三毛,他屬於王大毛這個群組, 而張小豬相對於王三毛,則只是一個『其他人(others)』而已. 不過,這裡有個特殊的人物要來介紹的,那就是 ...

  3. JavaScript的事件及异常捕获

    事件处理 [onClick]单击事件.[onMouseOver]鼠标经过事件.[onMouseOut]鼠标移出事件.[onChange]文本内容改变事件.[onSelect]文本被框选事件.[onFo ...

  4. Python之文件和目录操作

    1.文件基本操作 python内置了打开文件的函数open(),使用规则如下:   File_object=open(filename[,access_mode][,buffering]) Filen ...

  5. 第2章 术语 - Identity Server 4 中文文档(v1.0.0)

    规范.文档和对象模型等都使用特定的术语来表述. 2.1 IdentityServer IdentityServer是OpenID Connect提供程序 - 它实现OpenID Connect和OAu ...

  6. Docker搭建MongoDB

    1. Docker搭建Mongodb 1.1 获取docker镜像 docker pull mongo 1.2 创建mongodb容器 docker run --name my-mongo -p 27 ...

  7. python-IO编程,文件读写

    一.文件读写 1.打开文件 函数:open(name[. mode[. buffering]]) 参数: name:必须:文件的文件名(全路径或执行文件的相对路径.)) mode:可选:对文件的读写模 ...

  8. JAVA_接口_默认方法&静态方法

    1.小结(注意): 1.接口中无法定义成员变量,但是可以定义常量,其值不可以改变,默认使用public static final修饰 2.接口中,没有构造方法,不能创建对象 3.接口中,没有静态代码块 ...

  9. paas saas iaas 区别

    最近在公司里面经常听到一些paas saas iaas云服务的名词,把我自己都听蒙圈啦,所以就各种找资料终于对这三个名词有了一定的了解 首先上一张图如下: IAAS(nfrastructure as ...

  10. testlib.h从入门到入坟

    学了这么久OI连个spj都不会写真是惭愧啊... 趁着没退役赶紧学一波吧 配置 github下载地址 我是直接暴力复制粘贴的.. 然后扔到MingW的目录里 直接引用就好啦 基本语法 引用testli ...