使用Java反射优化多个方法调用
有段时间没来写博客了,心里一直念叨空了来,今天有时间来记录一篇。前段时间领导提出优化部分系统模块,根据业务要求系统中有很多产品,产品下面有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反射优化多个方法调用的更多相关文章
- Java 反射 Method的invoke回调调用任意方法
Java 反射 Method的invoke回调调用任意方法 @author ixenos 关键子:Method.Field.invoke方法指针/函数指针.回调函数 invoke回调流程示例 0.由C ...
- Atitit.通过null 参数 反射 动态反推方法调用
Atitit.通过null 参数 反射 动态反推方法调用 此时,直接使用java apache的ref工具都失效了.必须要自己实现了. 如果调用接口方法的话,就不能使用apache的ref工具,可 ...
- java反射构建对象和方法的反射调用
Java反射技术应用广泛,其能够配置:类的全限定名,方法和参数,完成对象的初始化,设置是反射某些方法.可以增强java的可配置性. 1.1 通过反射构建对象(无参数): 例如我们使用 ReflectS ...
- java反射之获取所有方法及其注解(包括实现的接口上的注解),获取各种标识符备忘
java反射之获取类或接口上的所有方法及其注解(包括实现的接口上的注解) /** * 获取类或接口上的所有方法及方法上的注解(包括方法实现上的注解以及接口上的注解),最完整的工具类,没有现成的工具类 ...
- 深入理解java虚拟机(十一) 方法调用-解析调用与分派调用
方法调用过程是指确定被调用方法的版本(即调用哪一个方法),并不包括方法执行过程.我们知道,Class 文件的编译过程中并不包括传统编译中的连接步骤,一切方法调用在 Class 文件调用里面存储的都只是 ...
- Java APi 之 RMI远程方法调用
一.什么是RPC RPC全称是remote procedure call,即远程过程调用.它是一种协议,用于从远程计算机上请求服务. 例如有两台服务器A和B,A上的应用想要调用B上应用的方法,但是他们 ...
- Java反射机制的使用方法
Java的反射机制同意你在程序执行的过程中获取类定义的细节.有时候在程序执行的时候才得知要调用哪个方法,这时候反射机制就派上用场了. 获取类 类的获取方法有下面几种: forName().通过Clas ...
- java反射与多态(父类调用子类)的代码演示
package Test0817; import java.lang.reflect.InvocationTargetException;import java.lang.reflect.Method ...
- JAVA反射优化
****************** 转自 https://my.oschina.net/19921228/blog/3042643 *********************** 比较反射与正常实例 ...
随机推荐
- Webpack+Vue如何导入Jquery和Jquery的第三方插件
创建一个jquery-vendor.js文件 import $ from 'jQuery'; console.log($); window.$ = $; window.jQuery = $; expo ...
- IdentityServer4之Client Credentials(客户端凭据许可)
IdentityServer4之Client Credentials(客户端凭据许可) 参考 项目创建:0_overview,1_client_credentials 概念:客户端凭据许可 认证服务端 ...
- 使用dotnet build时复制引用dll到生成目录
默认配置下dotnet build只会输出项目代码的dll,依赖用的是dotnet缓存中的dll,只有dotnet publish才会把依赖的dll一起输出到生成目录. 在项目csproj文件中添加以 ...
- transient和synchronized的使用
transient和synchronized这两个关键字没什么联系,这两天用到了它们,所以总结一下,两个关键字做个伴! transient 持久化时不被存储,当你的对象实现了Serializable接 ...
- 【Java基础】【19异常&IO(File类)】
19.01_异常(异常的概述和分类) A:异常的概述 异常就是Java程序在运行过程中出现的错误. B:异常的分类 通过API查看Throwable Error 服务器宕机,数据库崩溃等 Except ...
- angr初使用(1)
angr是早在几年前就出来了的一款很好用的工具,如今也出了docker,所以想直接安个docker来跑一跑. docker pull angr/angr .下载下来以后,进入docker ,这时并没有 ...
- [五]基础数据类型之Short详解
Short 基本数据类型short 的包装类 Short 类型的对象包含一个 short 类型的字段 原文地址:[五]基础数据类型之Short详解 属性简介 值为 215-1 ...
- python 中 *args 和 **kwargs 的区别
在 python 中,*args 和 **kwargs 都代表 1个 或 多个 参数的意思.*args 传入tuple 类型的无名参数,而 **kwargs 传入的参数是 dict 类型.下文举例说明 ...
- 服务注册中心之ZooKeeper系列(二) 实现一个简单微服务之间调用的例子
上一篇文章简单介绍了ZooKeeper,讲了分布式中,每个微服务都会部署到多台服务器上,那服务之间的调用是怎么样的呢?如图: 1.集群A中的服务调用者如何发现集群B中的服务提供者呢? 2.集群A中的服 ...
- Python使用WMI模块获取Windows系统的硬件信息,并使用pyinstaller库编译打包成exe的可执行文件
由于公司现阶段大多数应用软件都是基于Windows系统开发和部署,很多软件安装部署都是在windows server 2012.windows server 2008之类的服务器上,部门同事每次测试一 ...