删除未使用的Java类文件

package io.github.linwancen.code.modify;

import java.io.File;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.*;
import java.util.concurrent.ConcurrentSkipListSet;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.Collectors;

/**
 * 删除未使用的类
 * <br>仅依赖 JDK
 */
public class DeleteNotUsage {
    /**
     * 多线程电脑会很卡,约快 3 倍
     */
    private static final boolean PARALLEL = true;
    private static final boolean DELETE = false;
    private static final String CLASS_PATH = ClassLoader.getSystemResource("").getPath();
    /**
     * 扫描根目录
     * <br>根据需要添加.getParentFile()
     */
    private static final File ROOT;
    static {
        if (CLASS_PATH.endsWith("test-classes")) {
            // maven
            ROOT= new File(CLASS_PATH) // target/test-classes
                    .getParentFile() // target
                    .getParentFile(); //
        } else if (CLASS_PATH.endsWith("main")) {
            // gradle
            ROOT = new File(CLASS_PATH) // main
                    .getParentFile() // java
                    .getParentFile() // classes
                    .getParentFile() // build
                    .getParentFile();
        } else {
            ROOT = new File("");
        }
    }
    private static final int ROOT_LEN = ROOT.getPath().length();

    private static final Pattern EXT_PATTERN = Pattern.compile("\\.(?:java|xml|properties|conf|yml)|services$");
    private static final Pattern EXCLUDE_PATTERN = Pattern.compile("target|.git");
    /**
     * 添加自行定义的会被调用到的注解或关键字
     */
    private static final Pattern USED_PATTERN = Pattern.compile(
            "@(?:Service|Repository|Component|Test|Controller|Configuration)|main\\(String");

    private static final AtomicInteger N = new AtomicInteger();

    private static final class Data {
        final String pathString;
        final Set<String> usages = new ConcurrentSkipListSet<>();
        final File file;

        public Data(String pathString, File file) {
            this.pathString = pathString;
            this.file = file;
        }
    }

    public static void main(String[] args) throws Exception {
        List<Path> paths = Files.walk(ROOT.toPath())
                .filter(path -> path.toFile().isFile())
                .filter(path -> {
                    String s = path.toString();
                    return EXT_PATTERN.matcher(s).find() && !EXCLUDE_PATTERN.matcher(s).find();
                })
                .collect(Collectors.toList());
        int fileCount = paths.size();
        System.out.println("walk complete:" + fileCount);

        Map<String, Data> usageMap = new HashMap<>();
        StringBuilder patternStr = new StringBuilder("\\b(?:");
        for (Path path : paths) {
            filterClassSimpleName(usageMap, patternStr, path);
        }
        System.out.println("filter complete:" + usageMap.size());

        int len = patternStr.length();
        if (len == 0) {
            return;
        }
        patternStr.delete(len - 1, len);
        patternStr.append(")\\b");
        Pattern pattern = Pattern.compile(patternStr.toString());
        System.out.println("pattern compile complete:" + patternStr.length());

        long start = System.currentTimeMillis();
        if (PARALLEL) {
            paths.parallelStream().forEach(path ->
                    check(fileCount, usageMap, pattern, start, N.incrementAndGet(), path));
        } else {
            for (int i = 0; i < fileCount; i++) {
                Path path = paths.get(i);
                check(fileCount, usageMap, pattern, start, i, path);
            }
        }
        long useTime = (System.currentTimeMillis() - start) / 1000;
        System.out.println("check complete, size:{}" + fileCount + " use " + useTime / 60 + ":" + useTime % 60);

        long count = usageMap.entrySet().stream()
                .filter(entry -> {
                    if (!entry.getValue().usages.isEmpty()) {
                        return false;
                    }
                    String symbol = "-";
                    if (DELETE) {
                        symbol = entry.getValue().file.delete() ? "√" : "×";
                    }
                    String path = entry.getValue().pathString.substring(ROOT_LEN);
                    System.out.println(symbol + "\t.(" + entry.getKey() + ".java:0)\t" + path);
                    return true;
                })
                .count();
        System.out.println("check print complete, " + count + "/" + usageMap.size());
    }

    private static void filterClassSimpleName(Map<String, Data> usageMap, StringBuilder patternStr, Path path) {
        String s = path.toString();
        if (!s.endsWith(".java")) {
            return;
        }
        try {
            byte[] bytes = Files.readAllBytes(path);
            String code = new String(bytes, StandardCharsets.UTF_8);
            if (USED_PATTERN.matcher(code).find()) {
                return;
            }
        } catch (IOException e) {
            e.printStackTrace();
            return;
        }
        String classSimpleName = s.substring(s.lastIndexOf(File.separator) + 1, s.length() - 5);
        usageMap.put(classSimpleName, new Data(s, path.toFile()));
        patternStr.append(classSimpleName).append("|");
    }

    private static void check(int fileCount, Map<String, Data> usageMap, Pattern pattern, long start, int i, Path p) {
        if (i != 0 && (i % 1000) == 0) {
            long use = System.currentTimeMillis() - start;
            long remain = use / i * (fileCount - i);
            use = use / 1000;
            remain = remain / 1000;
            System.out.println("check " + i + "/" + fileCount
                    + ", use " + use / 60 + ":" + use % 60
                    + ", remain " + remain / 60 + ":" + remain % 60);
        }
        String pathStr = p.toString();
        try {
            byte[] bytes = Files.readAllBytes(p);
            String code = new String(bytes, StandardCharsets.UTF_8);
            Matcher m = pattern.matcher(code);
            while (m.find()) {
                String s = m.group();
                Data data = usageMap.get(s);
                if (!data.pathString.equals(pathStr)) {
                    data.usages.add(pathStr);
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,692评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,482评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,995评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,223评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,245评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,208评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,091评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,929评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,346评论 1 311
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,570评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,739评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,437评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,037评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,677评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,833评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,760评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,647评论 2 354

推荐阅读更多精彩内容