JAVA反射自定义工具类

1.问题场景

前端业务需要调用后端方法!

1.引用的类(可以是其他任意类)

这可以是任意希望能被引用的类!

package com.test.item;

public class jsTextUtil {
    private int count = 0;

    public int test(String i){
        System.out.println(count+" | "+i);
        return 10;
    }
    public int test(){
        System.out.println("无参方法执行");
        return 0;
    }

    public int test(int a,int b){
        System.out.println("2参方法执行 2int");
        return 0;
    }

    public int test(int a,String b){
        System.out.println("2参方法执行 int+string");
        return 0;
    }

    public int test(double a,String b){
        System.out.println("2参方法执行 double+string");
        return 0;
    }
}

2.反射工具类(核心方法)

这个工具类可以直接复制去用!但这个工具与一个弊端,就是值只识别基本数据类型的同名重写方法,并对同参数数量方法进行最优转换!

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
 * 反射操作工具
 */
public class ReflexUtil {

    Object instance;

    public ReflexUtil(){}


    /**
     * 参数直接传入
     * 1.传参数量: 严格符合目标方法参数
     * 2.传参顺序: 严格按照目标方法传参顺序
     * 3.传参类型为:字符串,和基础数据类型
     * 4.传参数据等级:string<char<boole < double<long<float<int<short<byte (数值顺序优先)
     * @param className     反射方法路径
     * @param methodName    反射方法
     * @param args          传入方法参数
     * @return
     */
    public Object execute(String className,String methodName,String args) {
        Class clazz = null;
        try {
            clazz = Class.forName(className);
            instance = clazz.newInstance();//方法实例
            String[] args0 = args.split(",");
            Method[] methods = clazz.getMethods();//获取所有方法
            List<Method> overrideMethods = new ArrayList<>();//重写(同名方法)方法集合
            for (Method method0 : methods){
                if (!methodName.equals(method0.getName()) || method0.getParameterCount() != args0.length){
                    continue;
                }
                overrideMethods.add(method0);
            }
            if (overrideMethods.size() > 0){
                Map<String,Method> methodPower = new HashMap<>();//权限-方法 k-v暂存
                Map<String,Object[]> argsPower = new HashMap<>();//权限-传参 k-v暂存
                int power = methodMatching(overrideMethods,args0,methodPower,argsPower);// 获取最佳匹配方法

                System.out.println("\n============= 匹配方法 start =============");
                for (String k:methodPower.keySet()){
                    System.out.println(k+" "+methodPower.get(k));
                }
                System.out.println("============= 匹配方法 end =============");

                System.out.println("============= 匹配参数组 start =============");
                for (String k:argsPower.keySet()){
                    for (Object obj:argsPower.get(k)){
                        System.out.print(obj.getClass().getTypeName()+":"+obj+"  ");
                    }
                    System.out.println();
                }
                System.out.println("============= 匹配参数组 end =============\n");
                System.out.println("最优方法权值:"+power);
                Method method = methodPower.get(String.valueOf(power));//最优权限
                Object[] args00 = argsPower.get(String.valueOf(power));
                System.out.print("最终执行方法:"+method+"\n\r入参:");
                for (Object object:args00){
                    System.out.print(object.getClass().getTypeName()+":"+object+" ");
                }
                System.out.println("\n\n执行方法打印:");
                method.invoke(instance,args00);
            }
        } catch (ClassNotFoundException e) {
            throw new RuntimeException("未找到类异常,请检查类地址: " + className, e);
        } catch (InstantiationException e) {
            throw new RuntimeException("实例化异常,请检查实例对象: "+ className,e);
        } catch (IllegalAccessException e) {
            throw new RuntimeException("访问权限的异常,请检查实例对象: "+ className+" 权限", e);
        } catch (InvocationTargetException e) {
            throw new RuntimeException("调用对象异常",e);
        }
        return instance;
    }


    /**
     * 排序方法
     * @param args 排序数组
     * @return
     */
    private int rank(int[] args){
        for (int i = 0;i < args.length-1;i++){
            for (int j = i+1;j < args.length;j++) {
                if (args[i] > args[j]) {
                    args[i] = args[i] + args[j];
                    args[j] = args[i] - args[j];
                    args[i] = args[i] - args[j];
                }
            }
        }
        return args[0];
    }

    /**
     * @methodMatching  同名同参数数量方法匹配
     * @param overrideMethods   重新方法
     * @param args0             切割参数
     * @param methodPower       权限-方法
     * @param argsPower         权限-传参
     * string<char<boole < double<long<float<int<short<byte
     */
    private int methodMatching(List<Method> overrideMethods, String[] args0,Map<String,Method> methodPower,Map<String,Object[]> argsPower){
        int methodCount = 0;
        int[] power = new int[overrideMethods.size()]; //方法权限
        Object[] args3;
        for (Method method:overrideMethods){ //合法方法遍历
            int i = 0;
            int methodPower2 = 0;
            args3 = new Object[args0.length];
            try{
                Class[] mentodClass = method.getParameterTypes();
                for (Class mc:mentodClass){//方法参数遍历配型
                    switch (mc.getName()){
                        case "byte"://字节型
                            args3[i] = Byte.parseByte(args0[i]);
                            if (args0[i].toString().equals(String.valueOf(Byte.parseByte(args0[i].toString())))){
                                methodPower2+=19; //字节型
                            } else if (args0[i].toString().equals(String.valueOf(Short.parseShort(args0[i].toString())))){
                                methodPower2+=29;
                            } else if (args0[i].toString().equals(String.valueOf(Integer.parseInt(args0[i].toString())))){
                                methodPower2+=38;
                            } else if (args0[i].toString().equals(String.valueOf(Long.parseLong(args0[i].toString())))){
                                methodPower2+=56;
                            } else {
                                methodPower2+=9999;
                            }
                            break;
                        case "short"://短整型
                            args3[i] = Short.parseShort(args0[i]);
                            if (args0[i].toString().equals(String.valueOf(Short.parseShort(args0[i].toString())))){
                                methodPower2+=28;
                            }else if (args0[i].toString().equals(String.valueOf(Integer.parseInt(args0[i].toString())))){
                                methodPower2+=38;
                            } else if (args0[i].toString().equals(String.valueOf(Long.parseLong(args0[i].toString())))){
                                methodPower2+=56;
                            } else {
                                methodPower2+=9999;
                            }
                            break;
                        case "int"://整型
                            args3[i] = Integer.parseInt(args0[i]);
                            if (args0[i].toString().equals(String.valueOf(Integer.parseInt(args0[i].toString())))){
                                methodPower2+=37;
                            } else if (args0[i].toString().equals(String.valueOf(Long.parseLong(args0[i].toString())))){
                                methodPower2+=56;
                            } else {
                                methodPower2+=9999;
                            }
                            break;
                        case "float"://单精度浮点型
                            args3[i] = Float.parseFloat(args0[i]);
                            if (args0[i].toString().equals(String.valueOf(Float.parseFloat(args0[i].toString())))){
                                methodPower2+=46;
                            } else if (args0[i].toString().equals(String.valueOf(Double.parseDouble(args0[i].toString())))){
                                methodPower2+=65;
                            } else {
                                methodPower2+=9999;
                            }
                            break;
                        case "login"://常整形
                            args3[i] = Long.parseLong(args0[i]);
                            methodPower2+=55;
                            break;
                        case "double"://双精度浮点型
                            args3[i] = Double.parseDouble(args0[i]);
                            methodPower2+=64;
                            break;
                        case "boolean":
                            args3[i] = Boolean.parseBoolean(args0[i]);
                            methodPower2+=73;
                            break;
                        case "char"://字符型
                            args3[i] = args0[i].toCharArray()[0];
                            methodPower2+=82;
                            break;
                        default:
                            args3[i] = String.valueOf(args0[i]);
                            methodPower2+=91;
                    }
                    i++;
                }
                power[methodCount] = methodPower2;//权限匹配
                methodPower.put(String.valueOf(methodPower2),method);//方法权限
                argsPower.put(String.valueOf(methodPower2),args3);//参数权限
            }catch (Exception e){
                System.out.println("类型转换异常!\n异常方法:"+method);
                power[methodCount] = 9999;//权限匹配
            }
            methodCount++;
        }
        return rank(power);
    }
}

3.启动类,调用方法

import org.junit.Test;

public class TestMain {
    @Test
    public void test(){
        try {
            System.out.println("------");
            new ReflexUtil().execute("com.test.item.jsTextUtil","test","11.2,12.1")
            System.out.println("------");
        } catch (Exception e) {
            System.out.println("反射获取对象错误");
            throw new RuntimeException(e);
        }
    }
}

4.测试打印

------
类型转换异常!
异常方法:public int com.test.item.jsTextUtil.test(int,int)
类型转换异常!
异常方法:public int com.test.item.jsTextUtil.test(int,java.lang.String)

============= 匹配方法 start =============
155 public int com.test.item.jsTextUtil.test(double,java.lang.String)
137 public int com.test.item.jsTextUtil.test(float,java.lang.String)
128 public int com.test.item.jsTextUtil.test(double,double)
92 public int com.test.item.jsTextUtil.test(float,float)
============= 匹配方法 end =============
============= 匹配参数组 start =============
java.lang.Double:11.2  java.lang.String:12.1  
java.lang.Float:11.2  java.lang.String:12.1  
java.lang.Double:11.2  java.lang.Double:12.1  
java.lang.Float:11.2  java.lang.Float:12.1  
============= 匹配参数组 end =============

最优方法权值:92
最终执行方法:public int com.test.item.jsTextUtil.test(float,float)
入参:java.lang.Float:11.2 java.lang.Float:12.1 

执行方法打印:
参方法执行 float:11.2 float:12.1
------

5.总结

其实反射的应用场景一般遇不到,即使遇到也会用其他的技术手段代替,没必要专门去另外写一个方法!但前端业务通过Ajax调用java方法工具的场景,反射就很不错!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Java开发中,我们经常需要使用各种工具类来简化和优化我们的代码。下面是一些常用的Java工具类: 1. StringUtils: StringUtils是Apache Commons Lang库中的一个工具类,提供了许多字符串处理的方法,如判断字符串是否为空、去除空格、截取指定长度等。 2. DateUtils: DateUtils也是Apache Commons Lang库的一个工具类,用于日期和时间的处理。它提供了一些常用的方法,如日期格式化、日期比较、日期计算等。 3. MathUtils: MathUtils是一个自定义工具类,提供了一些数学运算的方法,如求平均数、求最大最小值、四舍五入等。这些方法可以方便地在我们的代码中进行数学计算。 4. FileUtils: FileUtils是Apache Commons IO库中的一个工具类,用于处理文件和目录。它提供了一些常用的方法,如复制文件、删除文件、读取文件内容等。 5. CollectionUtils: CollectionUtils是Apache Commons Collections库的一个工具类,用于集合的操作。它提供了一些方便的方法,如判断集合是否为空、合并集合、过滤集合等。 6. HttpClientUtils: HttpClientUtils是Apache HttpClient库的一个工具类,用于发送HTTP请求。它提供了一些方法,如发送GET请求、发送POST请求、设置请求头等。 7. JsonUtils: JsonUtils是一个自定义工具类,用于处理JSON数据。它提供了一些方法,如将Java对象转换为JSON字符串、将JSON字符串转换为Java对象等。 8. ReflectionUtils: ReflectionUtils是一个自定义工具类,用于反射操作。它提供了一些方法,如获取类的所有字段、调用类的方法、创建类的实例等。 以上是一些常用的Java工具类,它们可以在我们的开发过程中提供便利,帮助我们更高效地编写代码。当然,根据具体的需求,我们还可以自定义一些工具类来满足特定的业务需求。 ### 回答2: 在Java开发中,工具类是经常用到的,可以提高开发效率和代码重用性。以下是一些常用的Java工具类: 1. String工具类:提供常用的字符串操作方法,如字符串判空、大小写转换、字符串拼接、去除空格等。 2. File工具类:提供文件操作的方法,如文件拷贝、文件删除、文件重命名、文件大小获取等。 3. Date工具类:提供日期操作的方法,如日期格式化、日期比较、日期加减操作等。 4. Collection工具类:提供集合操作的方法,如集合判空、集合拷贝、集合反转等。 5. Math工具类:提供数学计算的方法,如绝对值、最大值、最小值、四舍五入等。 6. Random工具类:提供生成随机数的方法,如生成指定范围的随机整数、生成随机字符串等。 7. JSON工具类:提供JSON数据转换的方法,如JSON对象转字符串、字符串转JSON对象等。 8. JDBC工具类:提供数据库连接和操作的方法,如数据库连接、SQL执行、结果集解析等。 9. XML工具类:提供XML数据解析和生成的方法,如解析XML文件、生成XML文件等。 10. 加密解密工具类:提供常用的加密解密算法,如MD5加密、BASE64编码等。 以上是一些常用的Java工具类,可以根据项目需求和具体场景选择使用。这些工具类可以节省开发时间、提高代码质量,是Java开发中必不可少的辅助工具。 ### 回答3: 为了方便开发人员编写Java代码,提高代码的重用性和可维护性,常常会整理收集一些常用的Java工具类。这些工具类涵盖了各种常见的功能,如字符串处理、日期时间操作、文件处理、网络通讯等。 首先,字符串处理是开发中常见的需求之一。收集的Java工具类中包含了字符串的各种操作方法,比如字符串的截取、拼接、替换、查找等。这些方法可以帮助开发人员更方便地操纵字符串,完成各种需求。 其次,日期时间操作是常用的功能之一。Java工具类中包含了日期时间的各种处理方法,如日期格式化、日期比较、日期加减等。这些方法可以帮助开发人员快速处理日期时间相关的需求,提高开发效率。 另外,文件处理也是常见的功能之一。Java工具类中包含了文件的各种处理方法,比如读取文件内容、写入文件内容、复制文件、删除文件等。这些方法可以帮助开发人员更方便地对文件进行操作,提供了文件处理的便利性。 最后,网络通讯也是常见的功能之一。Java工具类中包含了网络通讯的各种方法,如发送HTTP请求、接收HTTP响应、解析JSON等。这些方法可以帮助开发人员更方便地进行网络通讯操作,实现与服务器交互的功能。 综上所述,整理收集的常用Java工具类涵盖了字符串处理、日期时间操作、文件处理、网络通讯等各种常见的功能。这些工具类可以帮助开发人员更快速、方便地完成开发任务,提高代码的质量和效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值