java传入方法

简介: java传入方法

背景介绍

在日常代码中有时候近乎避免不了的使用魔法值,但是如果使用传入方法这种方式可以极大的降低魔法值出现的频率并且不用创建静态值。该方法主要参考了mybatisPlus,并在此基础上进行了扩展。

相关代码

FnConverter

package com.example.demo.bean;

import com.example.demo.entity.ValidatedRequestVO;

/**
 * @author seal 876651109@qq.com
 * @description
 * @date 7/12/2020 7:20 PM
 */
public class FnConverter<F, T> {
    /**
     * 传入方法返回字段名
     *
     * @param fn 方法
     * @return 字段名
     * @author seal 876651109@qq.com
     * @date 7/12/2020 7:32 PM
     */
    public String fnToFieldName(IFn<F, T> fn) {
        return Reflections.fnToFieldName(fn);
    }

    /**
     * 传入方法返回方法名
     *
     * @param fn 方法
     * @return 方法名
     * @author seal 876651109@qq.com
     * @date 7/12/2020 7:32 PM
     */
    public String fnToFnName(IFn<F, T> fn) {
        return Reflections.fnToFnName(fn);
    }

    /**
     * 传入方法返回注解
     *
     * @param fn 方法
     * @return mongo注解
     * @author seal 876651109@qq.com
     * @date 7/12/2020 7:32 PM
     */
    public String fnToMongoName(IFn<F, T> fn) {
        return Reflections.fnToMongoName(fn);
    }

    public static void main(String[] args) {
        FnConverter<ValidatedRequestVO, Object> fnConverter = new FnConverter();
        String fieldName = fnConverter.fnToFieldName(ValidatedRequestVO::getStr);
        System.out.println("字段名:" + fieldName);
        String fnName = fnConverter.fnToFnName(ValidatedRequestVO::getStr);
        System.out.println("方法名:" + fnName);

        FnConverter<String, Object> fnConverter2 = new FnConverter();
        String fieldName2 = fnConverter2.fnToFieldName(new ValidatedRequestVO()::setStr);
        System.out.println("字段名:" + fieldName2);
        String fnName2 = fnConverter2.fnToFnName(new ValidatedRequestVO()::setStr);
        System.out.println("方法名:" + fnName2);

        FnConverter<ValidatedRequestVO, Object> fnConverter3 = new FnConverter();
        String fieldName3 = fnConverter3.fnToMongoName(ValidatedRequestVO::getStartDate);
        System.out.println("字段名:" + fieldName3);
    }
}

IFn

package com.example.demo.bean;

import java.io.Serializable;

/**
 * F 传入类型,T返回类型
 * @author seal
 */
@FunctionalInterface
public interface IFn<F, T> extends Serializable {
    T apply(F source);
}

Reflections

package com.example.demo.bean;

import lombok.extern.slf4j.Slf4j;
import org.springframework.data.mongodb.core.mapping.Field;

import java.beans.Introspector;
import java.lang.invoke.SerializedLambda;
import java.lang.reflect.Method;

/**
 * 传入方法后的实现逻辑
 *
 * @author seal 876651109@qq.com
 * @date 7/12/2020 7:20 PM
 */
@Slf4j
public class Reflections {
    private Reflections() {
    }

    public static String fnToFieldName(IFn fn) {
        try {
            Method method = fn.getClass().getDeclaredMethod("writeReplace");
            method.setAccessible(Boolean.TRUE);
            SerializedLambda serializedLambda = (SerializedLambda) method.invoke(fn);
            String getter = serializedLambda.getImplMethodName();
            String fieldName = "";
            if (getter.startsWith("get")) {
                fieldName = Introspector.decapitalize(getter.replace("get", ""));
            } else {
                fieldName = Introspector.decapitalize(getter.replace("set", ""));
            }
            return fieldName;
        } catch (ReflectiveOperationException e) {
            log.warn(String.format("%s:%s",
                    Thread.currentThread().getStackTrace()[1].getMethodName(), e.getMessage()), e);
        }
        return "";
    }

    public static String fnToFnName(IFn fn) {
        try {
            Method method = fn.getClass().getDeclaredMethod("writeReplace");
            method.setAccessible(Boolean.TRUE);
            SerializedLambda serializedLambda = (SerializedLambda) method.invoke(fn);
            return serializedLambda.getImplMethodName();
        } catch (ReflectiveOperationException e) {
            log.warn(String.format("%s:%s",
                    Thread.currentThread().getStackTrace()[1].getMethodName(), e.getMessage()), e);
        }
        return "";
    }

    public static String fnToMongoName(IFn fn) {
        try {
            Method method = fn.getClass().getDeclaredMethod("writeReplace");
            method.setAccessible(Boolean.TRUE);
            SerializedLambda serializedLambda = (SerializedLambda) method.invoke(fn);
            String getter = serializedLambda.getImplMethodName();
            String fieldName = "";
            if (getter.startsWith("get")) {
                fieldName = Introspector.decapitalize(getter.replace("get", ""));
            } else {
                fieldName = Introspector.decapitalize(getter.replace("set", ""));
            }
            Field field = Class.forName(serializedLambda.getImplClass().replace("/", ".")).getDeclaredField(fieldName).getAnnotation(Field.class);
            return field == null ? fieldName : field.value();
        } catch (ReflectiveOperationException e) {
            log.warn(String.format("%s:%s",
                    Thread.currentThread().getStackTrace()[1].getMethodName(), e.getMessage()), e);
        }
        return "";
    }
}

举例应用

  1. 如视频中所展现的可以取方法/字段的注解
  2. 利用反射实现伪代理
  3. 最普遍的应用即mybatisPlus的应用,可以动态传入需要的字段和不需要的字段而不用改动sql,以此优化性能

相关连接

视频
github
如果有帮助到你的话请顺手点个赞,这对我真的很重要

相关文章
|
11天前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
51 4
|
22天前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
43 17
|
16天前
|
Java 测试技术 Maven
Java一分钟之-PowerMock:静态方法与私有方法测试
通过本文的详细介绍,您可以使用PowerMock轻松地测试Java代码中的静态方法和私有方法。PowerMock通过扩展Mockito,提供了强大的功能,帮助开发者在复杂的测试场景中保持高效和准确的单元测试。希望本文对您的Java单元测试有所帮助。
32 2
|
1月前
|
算法 Java Linux
java制作海报二:java使用Graphics2D 在图片上合成另一个照片,并将照片切割成头像,头像切割成圆形方法详解
这篇文章介绍了如何使用Java的Graphics2D类在图片上合成另一个照片,并将照片切割成圆形头像的方法。
48 1
java制作海报二:java使用Graphics2D 在图片上合成另一个照片,并将照片切割成头像,头像切割成圆形方法详解
|
24天前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
18 3
|
26天前
|
Java 大数据 API
别死脑筋,赶紧学起来!Java之Steam() API 常用方法使用,让开发简单起来!
分享Java Stream API的常用方法,让开发更简单。涵盖filter、map、sorted等操作,提高代码效率与可读性。关注公众号,了解更多技术内容。
|
24天前
|
Java 开发者
在Java多线程编程中,选择合适的线程创建方法至关重要
【10月更文挑战第20天】在Java多线程编程中,选择合适的线程创建方法至关重要。本文通过案例分析,探讨了继承Thread类和实现Runnable接口两种方法的优缺点及适用场景,帮助开发者做出明智的选择。
16 2
|
24天前
|
安全 Java
Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧
【10月更文挑战第20天】Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧,包括避免在循环外调用wait()、优先使用notifyAll()、确保线程安全及处理InterruptedException等,帮助读者更好地掌握这些方法的应用。
15 1
|
24天前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
28 1
|
24天前
|
Java
在Java多线程编程中,`wait()`和`notify()`方法的相遇如同一场奇妙的邂逅
在Java多线程编程中,`wait()`和`notify()`方法的相遇如同一场奇妙的邂逅。它们用于线程间通信,使线程能够协作完成任务。通过这些方法,生产者和消费者线程可以高效地管理共享资源,确保程序的有序运行。正确使用这些方法需要遵循同步规则,避免虚假唤醒等问题。示例代码展示了如何在生产者-消费者模型中使用`wait()`和`notify()`。
24 1