前言:之前一直想熟练自定义注解,但当时没有实际的应用需求,也就是自己写了个实例,就搁置下来了。 这回,是在做一个工程的时候,需要根据变化,注入新的实例到工厂。 为了方便,也是代码整洁,就用到了自定义注解。 

首先,有几个点需要说明:

1,我需要监控指定路径的变化,比如:文件的删除、文件的增加、修改等。 当文件删除时,我需要销毁工厂中的实例,并发出预警,做好备份。 当文件新增时(可能是class文件,也可能是jar包等)我需要逐层扫描注解,将新的实例注入到工厂备用 等等

2,目前做的实现,只写了class文件的递归扫描, 而且文件的监控,还没有写具体的响应方法

3,基本步骤:首先:定义注解;其次:解析注解

一、定义注解

import java.lang.annotation.*;

/**
 * 自定义excel模板注入注解
 */
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE,ElementType.METHOD})
@Documented
public @interface ExcelTemplate {

    //类名—必须录入
    String name();

    //提供当前模板解析的人
    String author() default "Angelina";

    //当前模板被提供的日期
    String supportedDate();

}

如果是必须录入的属性,就不要提供默认值,其余的一些属性,就自己查查吧     PS:最开始还以为这个玩意儿和之前写的一个权限注解一样的,原来当时用的Jeddict里面,提供了一些框架整合的东西,尴尬了。

二、注解解析

    /**
     * 解析注解
     *
     * @param packageName 包名
     * @throws Exception
     */
    public void parsingAnnotation(String packageName) throws Exception {
        List<Class<?>> classes = getAllClass(packageName);
        if (!classes.isEmpty()) {
            AbstractExcelFactory excelFactory = AbstractExcelFactory.getInstance();
            for (Class _class : classes) {
                if (_class.isAnnotationPresent(ExcelTemplate.class)) {
                    ExcelTemplate excelTemplate = (ExcelTemplate) _class.getAnnotation(ExcelTemplate.class);
                    //获取注解中的类名称
                    String name = excelTemplate.name();
                    String strBeanName=_class.getName();
                    // 注册
                    excelFactory.registryExcelReaderBean(name, strBeanName);
                }
            }
        }
    }

解析注解,其实就很简单,就跟实现一个接口一样:确定当扫描到此注解时,需要执行的工作。 比如说,此处:当我扫描到ExcelTemplate注解时,将其类实例注入到工厂。

三、辅助查找所有包下的class文件方法

    /**
     * 获取包里所有的class文件
     *
     * @return
     * @throws IOException
     */
    public List<Class<?>> getAllClass(String packageName) throws IOException {
        Enumeration<URL> enumeration = Thread.currentThread().getContextClassLoader().getResources(packageName);
        List<Class<?>> classes = new ArrayList<Class<?>>();
        while (enumeration.hasMoreElements()) {
            URL url = enumeration.nextElement();
            String protocol = url.getProtocol();
            if ("file".equals(protocol)) {
                String filePath = URLDecoder.decode(url.getFile(), "UTF-8");
                getAllFile(packageName, filePath, classes);
            }
        }
        return classes;
    }


    /**
     * 获取目录下的所有文件
     *
     * @param packageName 包名
     * @param packagePath 包的路径
     * @param classes     包下的所有类文件
     */
    public void getAllFile(String packageName, String packagePath, List<Class<?>> classes) {
        File fileDirector = new File(packagePath);
        File[] files = fileDirector.listFiles(new FileFilter() {
            //自定义过滤规则
            @Override
            public boolean accept(File file) {
                return (true && file.isDirectory()) || (file.getName().endsWith(".class"));
            }
        });
        for (File file : files) {
            if (file.isDirectory()) {
                String newPackageName = packageName + "." + file.getName();
                String newPackagePath = file.getAbsolutePath();
                getAllFile(newPackageName, newPackagePath, classes);
            } else {
                String className = file.getName().substring(0, file.getName().length() - 6);
                try {
                    classes.add(Class.forName(packageName + '.' + className));
                } catch (ClassNotFoundException e) {
                    throw new ExcelException(ExceptionEnum.FAILDFINDTEMPLATE.getCode(),
                            ExceptionEnum.FAILDFINDTEMPLATE.getName());
                }
            }
        }
    }

这里只是处理了.class文件,如果要解析jar包,则需要做响应的判断和读取,如 if("file".equals(protocol)) 类似的判断,随后做相应的逻辑处理

四、总结

书到用时方恨少,但什么时候学习都不晚。 注解,是通过统一集中处理,从而给开发带来方便,也使得代码更为整洁。 但怎么说呢,凡事看需求吧,因为有时候注解是很方便,但在一些逻辑识别的时候,也有一定的阻碍。  

有时候我就在想,我们平时用到的一系列注解,到底都做了什么?它是怎么做到的?它的这种做法,是不是更好的?

在做这个注解的过程中,一个很大的体会: 既然要利用我提供的便利,那就得遵守我定下的规则!  一切都是取舍平衡


附录:文件监控

1,定义文件变动时事件

import org.apache.commons.io.monitor.FileAlterationListenerAdaptor;

import java.io.File;

/**
 * 文件监听类—新增、修改、删除excel模板
 */
public class FileListener extends FileAlterationListenerAdaptor {
    @Override
    public void onFileCreate(File file) {
        System.out.println("[新建]:" + file.getAbsolutePath());
        System.out.println("此时,调用注入方法,注入新添加的excel模板解析类");
    }

    @Override
    public void onFileChange(File file) {
        System.out.println("[修改]:" + file.getAbsolutePath());
        System.out.println("日志记录,警告");
        System.out.println("此时,销毁旧的excel模板解析类,重新加载新的,但命名一定要能区分出更改的文件");
    }

    @Override
    public void onFileDelete(File file) {
        System.out.println("[删除]:" + file.getAbsolutePath());
        System.out.println("此时,发布异常");
    }
}

2,实施监控

import org.apache.commons.io.filefilter.FileFilterUtils;
import org.apache.commons.io.monitor.FileAlterationMonitor;
import org.apache.commons.io.monitor.FileAlterationObserver;

import java.util.concurrent.TimeUnit;

/**
 * 容器监听—根据文件监听的变化而变化
 */
public class ContainerListener {
    public static void main(String[] args) throws Exception{
        // 监控目录  
        String rootDir = "example";
        // 轮询间隔 7 天—视情况而定 
        //long interval = TimeUnit.DAYS.toDays(7);
        long interval=TimeUnit.SECONDS.toMillis(10);
        FileAlterationObserver observer = new FileAlterationObserver(
                rootDir,
                //设置监控过滤
                FileFilterUtils.and(
                        FileFilterUtils.fileFileFilter(),
                        FileFilterUtils.suffixFileFilter(".java")),
                null);
        observer.addListener(new FileListener());
        FileAlterationMonitor monitor = new FileAlterationMonitor(interval, observer);
        // 开始监控  
        monitor.start();
    }
}




posted on 2017-11-30 23:06  何红霞  阅读(248)  评论(0编辑  收藏  举报