手写一个自定义注解(不详细你来揍我版)

1、首先了解注解是什么,参考如下文章,同时了解自定义注解。

自定义注解实现概括版

在Java中,配置文件(Configuration Files)是注解出现之前用于传递元数据的另一种方式。配置文件是一种将程序配置信息存储在外部文件中的方法,以便在程序运行时读取和使用这些配置信息。

在一些情况下,配置文件可以用于传递元数据,例如在配置框架、应用程序设置、数据库连接等方面。通过读取配置文件,程序可以获取一些必要的信息,从而影响其行为和功能。

然而,配置文件也有一些局限性,例如:

格式不统一: 配置文件的格式可能因文件类型、工具、框架而异,导致解析和处理变得复杂。

易于出错: 配置文件容易出现错误,例如拼写错误、格式错误等,这些错误可能难以定位和修复。

无法与代码关联: 配置文件通常与代码分离,难以直接与代码关联,使得代码和配置之间的联系不够紧密。

注解的引入解决了这些问题,使得元数据能够更紧密地与代码关联,更加可读和可维护。通过在代码中使用注解,开发者可以在代码级别上提供元数据,从而增加了代码的可读性、维护性和灵活性。这与配置文件的思路相似,但注解能够更加直接地嵌入到代码中,并且在编译期和运行期都能够被直接处理。
注解(Annotation)是一种元数据,用于为程序代码提供信息,不会直接影响程序的执行,但可以提供编译器、工具和框架额外的信息。注解最早在Java SE 5.0版本中引入,它们为开发者提供了一种在源代码中添加元数据的方式,以便在编译期、运行期或者其他工具中处理这些信息。注解的引入使得代码的描述更加丰富,同时也方便了代码的生成、分析和管理。

自定义注解的编写:

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

// 定义一个自定义注解
(RetentionPolicy.RUNTIME) // 指定注解保留到运行时
(ElementType.METHOD) // 指定注解适用于方法
public  MyCustomAnnotation {
    String value(); // 声明注解成员
}

在这个示例中,我们创建了一个名为 MyCustomAnnotation 的自定义注解,它具有一个成员 value。@Retention 注解指定了注解的保留策略,这里我们选择在运行时保留,以便在运行期间访问注解信息。@Target 注解指定了注解可以应用的目标,这里我们指定了它适用于方法。

你可以在代码中使用自定义注解,并为注解成员提供值:

public class MyClass {
    (value = "Hello, Custom Annotation!")
    public void myMethod() {
        // 方法内容
    }
}

在实际应用中,你可以通过反射来获取注解信息并根据注解信息进行不同的处理。这种方式在编写框架、库、测试工具等方面非常有用。

请注意,虽然这只是一个简单的例子,但自定义注解在实际应用中可以非常强大。你可以根据实际需求定义更复杂的自定义注解,并在项目中使用它们来提供更多的元数据信息。

在Java中,配置文件(Configuration Files)是注解出现之前用于传递元数据的另一种方式。配置文件是一种将程序配置信息存储在外部文件中的方法,以便在程序运行时读取和使用这些配置信息。

在一些情况下,配置文件可以用于传递元数据,例如在配置框架、应用程序设置、数据库连接等方面。通过读取配置文件,程序可以获取一些必要的信息,从而影响其行为和功能。

然而,配置文件也有一些局限性,例如:

格式不统一: 配置文件的格式可能因文件类型、工具、框架而异,导致解析和处理变得复杂。

易于出错: 配置文件容易出现错误,例如拼写错误、格式错误等,这些错误可能难以定位和修复。

无法与代码关联: 配置文件通常与代码分离,难以直接与代码关联,使得代码和配置之间的联系不够紧密。

注解的引入解决了这些问题,使得元数据能够更紧密地与代码关联,更加可读和可维护。通过在代码中使用注解,开发者可以在代码级别上提供元数据,从而增加了代码的可读性、维护性和灵活性。这与配置文件的思路相似,但注解能够更加直接地嵌入到代码中,并且在编译期和运行期都能够被直接处理。

在Java中,你可以通过创建自定义注解来为代码添加元数据,这可以帮助你在编写工具、框架或库时提供更多的信息。下面是一个简单的示例,展示如何编写一个自定义注解:

2、完全手写一个自定义注解(遍历版和map版)

公共部分代码

package LengthImpl1;


import lombok.Data;

/**
 * @ClassName Person
 * @Description TODO
 * @Author zy
 * @Date 2022/10/14 18:56
 * @Version 1.0
 */

@Data
public class Person {
    //声明字符串类型的属性id
    private String id;

    @Length(min=3, max =8, errorMsg = "姓名长度必须3-8之间")
    private String name;

    private Integer age;

}

package LengthImpl1;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

//表示自定义注解标在字段上
@Target({ElementType.FIELD})
//表示自定义注解可保留到运行时
@Retention(RetentionPolicy.RUNTIME)
//自定义注解,注解名字是Length
public @interface Length {
    // 允许的字符串长度最小值
    int min();
    // 的允许字符串长度最大值
    int max();
    // 自定义错误提示
    String errorMsg();
}


遍历版

package LengthImp;

import java.io.FileNotFoundException;
import java.io.IOException;

/**
 * @BelongsProject: JAVAtest
 * @BelongsPackage: LengthImp
 * @Author: GuoYuan.Zhao
 * @CreateTime: 2023-02-25  11:38
 * @Description: TODO
 * @Version: 1.0
 */
public class Client {
    public static void main(String[] args) throws IOException, NoSuchFieldException, ClassNotFoundException {
        LengthImpl length = new LengthImpl();
        length.traverseAllFields();
        System.out.println(length.min());
        System.out.println(length.max());
        System.out.println(length.errorMsg());

    }
}
package LengthImp;

import java.io.*;
import java.lang.annotation.Annotation;
import java.util.*;

/**
 * @BelongsProject: 09_annotation_source
 * @BelongsPackage:
 * @Author: 赵媛
 * @Description: 描述什么人干什么事儿
 * @CreateTime: 2023-02-21 11:10
 * @Version: 1.0
 */

public class LengthImpl implements Length {

    @Override
    public int min() {
        return Integer.valueOf(minInterger);
    }

    @Override
    public int max() {
        return Integer.valueOf(maxInteger);
    }

    @Override
    public String errorMsg() {
        return errorInfo;
    }

    String maxInteger ;
    //获取min后边的数值
    String minInterger ;
    //获取error后边的字符串
    String errorInfo ;

    public   void  traverseAllFields() throws IOException, ClassNotFoundException, NoSuchFieldException {
        String fileName = "";
        String methodName = "";
        List<String> lineLists = new ArrayList<>();
        String lineListsYes ;
        int index = 0;
        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入文件路径");
        fileName = scanner.next();
        System.out.println("请输入要查阅的属性名称");
        methodName = scanner.next();
        //首先声明一个list放置每行的文本
        Scanner scannerLine = new Scanner(new FileReader(fileName));
        while (scannerLine.hasNextLine()){  //按行读取字符串
            String line = (scannerLine.nextLine());
            lineLists.add(line);
        }
//        System.out.println(lineLists);
        //遍历这个list里边的值,哪一行里边有属性name关键字
        for(String everyLineString : lineLists){
            if(everyLineString.contains(methodName)){
                //获取当前索引
                 index = lineLists.indexOf(everyLineString);
            }
        }
        //找到上一行的位置
         lineListsYes  = lineLists.get(index-1);
//        System.out.println(lineListsYes);
        //读取length后的关键字,可以进行直接输出,也可以放到map里边
        String annotationValue = lineListsYes.replace("(", ",");
        String annotationValue1 = annotationValue.replace(")", ",");
        String[] fields = annotationValue1.split(",");

        for (int i = 0; i < fields.length; i++) {
            if(fields[i].contains("min")){
                minInterger = fields[i].split("=")[1];

            }
            if(fields[i].contains("max")){
                maxInteger = fields[i].split("=")[1];
            }
            if(fields[i].contains("errorMsg")){
                errorInfo = fields[i].split("=")[1];
            }
        }


    }





    @Override
    public Class<? extends Annotation> annotationType() {
        return null;
    }
}

map版

package LengthImpl1;

import java.io.IOException;

/**
 * @BelongsProject: JAVAtest
 * @BelongsPackage: LengthImp
 * @Author: GuoYuan.Zhao
 * @CreateTime: 2023-02-25  11:38
 * @Description: TODO
 * @Version: 1.0
 */
public class Client {
    public static void main(String[] args) throws IOException, NoSuchFieldException, ClassNotFoundException {
//        LengthImpl length = new LengthImpl();
//        length.traverseAllFields();
        MapImpl map = new MapImpl();
        map.readFileInfo();
//        System.out.println(length.min());
//        System.out.println(length.max());
//        System.out.println(length.errorMsg());
        System.out.println(map.min());
        System.out.println(map.max());
        System.out.println(map.errorMsg());

    }
}
package LengthImpl1;

import javassist.bytecode.AttributeInfo;
import javassist.bytecode.ClassFile;
import javassist.bytecode.FieldInfo;

import java.io.*;
import java.lang.annotation.Annotation;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
 * @BelongsProject: JAVAtest
 * @BelongsPackage: LengthImpl1
 * @Author: GuoYuan.Zhao
 * @CreateTime: 2023-02-26  17:21
 * @Description: TODO
 * @Version: 1.0
 */
public class MapImpl implements Length {
    @Override
    public int min()  {
//        String min = this.readFile("min");
//        System.out.println("最小值为:"+min);
//        return Integer.valueOf(min) ;
        String minValue = this.getMethedValue(Thread.currentThread().getStackTrace()[1].getMethodName());
        return Integer.valueOf(minValue);
    }

    @Override
    public int max() {
//        String max = this.readFile("max");
//        System.out.println("最大值为:"+max);
//        return Integer.valueOf(max) ;

        String maxValue = this.getMethedValue(Thread.currentThread().getStackTrace()[1].getMethodName());
        return Integer.valueOf(maxValue);
    }

    @Override
    public String errorMsg() {
//        String errorMessage = this.readFile("errorMsg");
//        System.out.println("错误信息:"+  errorMessage);
//        return errorMessage;

        String errorMsgValue = this.getMethedValue(Thread.currentThread().getStackTrace()[1].getMethodName());
        return errorMsgValue;
    }


    @Override
    public Class<? extends Annotation> annotationType() {
        return null;
    }
    boolean state=true;
    private String getMethedValue(String name){

        if (state==true){
            state=false;
            methedValue= this.readFileInfo();
        }
        if(name.equals("errorMsg")){
            System.out.println("错误信息:"+  methedValue.get(name));
            return methedValue.get(name);
        }else if(name.equals("min")){
            return methedValue.get(name);
        }else if(name.equals("max")){
            System.out.println("最大值为:"+methedValue.get(name));
            return methedValue.get(name);
        }
        return null;
    }

    Map<String,String> methedValue = new HashMap<>();
    public Map<String, String> readFileInfo()  {
        try{
            BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
            System.out.println("请输入要读取文件的路径");
            String fileName=br.readLine();
            DataInputStream csanner = new DataInputStream(new FileInputStream(new File(fileName)));
            ClassFile classFile = new ClassFile(csanner);
            List<FieldInfo> fieldsList = classFile.getFields();
            for (FieldInfo field:fieldsList){
                for (AttributeInfo attributeInfo : field.getAttributes()) {
                    System.out.println(attributeInfo);
                    String annotationValue = attributeInfo.toString().replace("(", ",");
                    String annotationValue1 = annotationValue.replace(")", ",");
                    String[] fields = annotationValue1.split(",");
                    for (int i = 1; i < fields.length; i++) {
                        String[] fieldSplit = fields[i].split("=");
                        methedValue.put(fieldSplit[0].trim(),fieldSplit[1].trim());
                        System.out.println(methedValue);
                    }
                }

            }
        }catch (Exception e){
            System.out.println("获取文件失败"+e);
        }
        return methedValue;
    }
}
### 回答1: 首先,感谢您的提问。关于手写一个 Spring 框架,这是一个非常复杂的任务,需要深入了解 Spring 框架的原理和设计思想。如果您想要手写一个 Spring 框架,建议您先学习 Spring 框架的基本原理和核心组件,例如 IoC 容器、AOP、MVC 等。然后,您可以根据自己的需求和业务场景,逐步实现自己的 Spring 框架。当然,这需要您具备较强的 Java 编程能力和系统设计能力。希望我的回答能够帮助到您。 ### 回答2: 手写一个Spring框架是一项复杂且庞大的任务,需要深入理解Spring框架的核心概念和原理。以下是一个简要的步骤指南: 1. 了解Spring框架的核心概念和原理,包括依赖注入、面向切面编程、控制反转等。 2. 创建一个独立的IoC容器,用于管理对象的创建和依赖注入。可以通过反射机制实现对象的自动注入,同时考虑循环依赖的情况。 3. 实现一个简单的AOP框架,通过动态代理机制实现方法拦截和增强。可以使用JDK自带的动态代理或者CGLIB库等。 4. 设计和实现一个MVC框架,用于处理Web请求和返回结果。可以使用Servlet作为底层技术,设计并实现自己的DispatcherServlet。 5. 创建一个配置文件解析器,用于解析Spring配置文件中的Bean定义和依赖关系。可以使用DOM、SAX、XML解析器等。 6. 编一个Bean工厂,负责管理Bean的生命周期和依赖关系。可以在Bean创建时对其进行初始化和销毁的处理。 7. 实现一个简单的事务管理器,用于处理数据库事务。可以使用JDBC或者类似的持久化框架。 手写一个Spring框架需要深入理解Java反射、动态代理、设计模式等知识,并具备一定的编程和架构能力。以上只是一个简要的步骤指南,实际的开发过程中还需要考虑到更多的细节和边界情况。因此,对于初学者来说,手写一个完整的Spring框架可能是一个挑战,建议先熟悉和使用官方提供的Spring框架,通过实践逐步深入理解其原理和内部实现,再考虑手写一个简化的本。 ### 回答3: 手写一个Spring框架是一项庞大而复杂的任务,需要深入理解Spring的设计原理和内部机制。以下是手写一个简化Spring框架的高级步骤: 第一步:实现IoC容器 IoC(控制反转)是Spring框架的核心概念,通过实现一个简单的IoC容器,来管理对象的创建和依赖注入。容器可以采用单例模式,在初始化时读取配置文件,通过反射实例化并注入依赖。容器会根据配置文件中的Bean定义,创建并管理对象之间的依赖关系。 第二步:实现依赖注入 依赖注入是Spring框架的另一个重要概念,通过实现依赖注入功能,实现对象之间的解耦。可以通过反射遍历对象的属性,扫描注解或配置文件,将依赖的对象自动注入到属性中。 第三步:实现AOP功能 AOP(面向切面编程)是Spring框架中提供的另一个重要功能,通过实现AOP功能,能够在不修改原有代码的情况下,对业务逻辑进行增强。可以使用动态代理等机制,在目标方法的前后增加额外的操作。 第四步:实现Web MVC功能 Spring框架还提供了强大的Web MVC功能,通过实现简化的请求处理、路由和视图解析等功能,来实现一个基本的Web应用。可以通过Servlet或者自定义HttpHandler来处理请求,并使用模板引擎等技术进行视图解析和渲染。 第五步:实现其他特性 除了上述核心功能外,Spring框架还提供了诸如事务管理、数据访问、安全等一系列扩展功能。可以根据实际需要,逐步实现这些特性。 最后,为了保持代码的可维护性和可扩展性,建议使用设计模式、模块化的方式来实现框架的各个功能。此外,持续学习和了解Spring框架的最新发展和技术变化,可以帮助优化和提升手写框架的质量和性能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Circ.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值