JAVA案例 | 反射和注解的应用

【小案例】

  将src源文件中加了@Singleton注解的类都在程序启动时以【单例】的形式加载到内存。

首先我们来看一看本案例中创建的各个类(位置可以根据个人喜好随意):

首先 我们来创建一个用来 找出所有class文件,并将路径名转换为全限定名的工具类FileUtil


import java.io.File;
import java.util.List;
import java.util.ArrayList;
import java.util.stream.Collectors;

public class FileUtil {
    public static List<String> getAllClassPath(File file){
        List<String> classPaths = new ArrayList<>();

        findAll(file.getAbsolutePath(),classPaths);

        List<String> list = classPaths.stream().map(path -> {
            return path.replace(file.getAbsolutePath()+"\\","")
                    .replaceAll("\\\\",".")
                    .replace(".class","");
        }).collect(Collectors.toList());
        return list;
    }
    private static void findAll(String path,List<String> list){
        File file = new File(path);
        File[] files = file.listFiles((f,n) -> new File(f,n).isDirectory() || n.contains(".class"));

        //判断数组是否为空或长度是否为0
        if(files == null || files.length == 0){
            return;
        }

        //使用增强for循环迭代
        for(File parent : files){
            //判断是否为文件夹
            if(parent.isDirectory()){
                //是则递归
                findAll(parent.getAbsolutePath(),list);
            } else {
                //不是则直接加入数组
                list.add(parent.getAbsolutePath());
            }
        }
    }
}

 该类的findAllClassPath方法会返回一个泛型为String的List,存储了所有类的全限定名,接下来我们要做的就是通过这些全限定名,利用反射,得到加了@Singleton注解的类,并在程序启动时以单例的形式加载到内存。

我们先来添加一个Singleton.class

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

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Singleton {

}

再创建一个ApplicationContext类来存放最终我们要存放单例的Map映射,只能通过addSingleton和getSingleton来添加或获取单例.

import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

public class ApplicationContext {
    private static final Map<Class<?>,Object> context = new ConcurrentHashMap<>();

    public static void addSingleton(Class<?> clazz, Object entity){
        context.put(clazz,entity);
    }

    public static <T> T getSingleton(Class<T> clazz){
        return (T)context.get(clazz);
    }
}

最后我们写一个处理器SingletonHandler 用来找出加了@Singleton注解的类,并将其和其对应的单例放到我们的context中


import java.util.List;

public class SingletonHandler {

    public static void handler(List<String> classPaths){
        for(String classPath : classPaths){
            Class<?> clazz = null;
            try {
                clazz = Class.forName(classPath);
            } catch (ClassNotFoundException e){
                e.printStackTrace();
            }

            Singleton annotation = null;
            if(clazz != null) {
                annotation = clazz.getAnnotation(Singleton.class);
            }

            if(annotation != null){
                Object instance = null;
                try{
                    instance = clazz.newInstance();
                } catch (IllegalAccessException | InstantiationException e){
                    e.printStackTrace();
                }
                ApplicationContext.addSingleton(clazz,instance);
            }
        }
    }
}

温馨提示:别忘了导包哦~

最后创建一个Bootstrap类,将上面那些工具类的调用放入静态代码块中

import java.net.URL;
import java.io.File;
import java.util.List;
import java.util.ArrayList;

public class Bootstrap {
    static{
        //尝试获取类路径
        //获取当前线程,再获取当前的类加载器,若为空字符串则获取根路径
        final URL resource = Thread.currentThread().getContextClassLoader().getResource("");

        if(resource != null) {
            //findAll(resource.getFile());
            File file = new File(resource.getFile());
            List<String> list = FileUtil.getAllClassPath(file);
            SingletonHandler.handler(list);
        }
    }
    public static void main(String[] args){

    }
}

写到这里,那么恭喜你,小案例已经完成啦!

接下来我们来实验一下我们的成果:

创建一个Dog类,并为它加上@Singleton注解

@Singleton
public class Dog {
    private String name;

    public Dog() {
    }

    private Dog(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "Dog{" +
                "name='" + name + '\'' +
                '}';
    }
}

写个main方法尝试运行一下:

public static void main(String[] args){
        Dog dog = ApplicationContext.getSingleton(Dog.class);
        System.out.println(dog);
    }

我们来看看运行结果

 

成功啦!!

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java反射可以让我们在运行时获取和操作类的信息,包括类的属性、方法、构造函数等。下面是一个使用Java反射的简单案例: 假设我们有一个类Person: ```java public class Person { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } public void sayHello() { System.out.println("Hello, my name is " + name + ", I am " + age + " years old."); } } ``` 现在,我们可以使用反射来获取和调用Person类的构造函数和方法: ```java import java.lang.reflect.Constructor; import java.lang.reflect.Method; public class ReflectionExample { public static void main(String[] args) throws Exception { // 获取Person类的Class对象 Class<Person> personClass = Person.class; // 获取Person类的构造函数 Constructor<Person> constructor = personClass.getConstructor(String.class, int.class); // 使用构造函数创建Person对象 Person person = constructor.newInstance("Tom", 20); // 获取Person类的sayHello方法 Method sayHelloMethod = personClass.getMethod("sayHello"); // 调用sayHello方法 sayHelloMethod.invoke(person); } } ``` 以上代码中,我们首先通过`Person.class`获取了Person类的Class对象,然后使用`getConstructor`方法获取了Person类的构造函数,并使用构造函数创建了一个Person对象。接着,我们使用`getMethod`方法获取了Person类的`sayHello`方法,并使用`invoke`方法调用了该方法。最终,我们输出了`Hello, my name is Tom, I am 20 years old.`。 这只是一个简单的反射案例,实际上Java反射还可以做很多其他的事情,比如动态代理、注解处理等等。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值