首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >读取和显示Java .class版本的工具

读取和显示Java .class版本的工具
EN

Stack Overflow用户
提问于 2008-08-26 06:52:03
回答 6查看 63K关注 0票数 118

你们当中有没有人知道有什么工具可以搜索.class文件,然后显示它们的编译版本?

我知道您可以在十六进制编辑器中单独查看它们,但我有很多类文件需要查看(我的大型应用程序中的某些文件由于某种原因正在编译为Java6 )。

EN

Stack Overflow用户

发布于 2020-08-01 10:17:18

最简单的方法是使用这里的许多答案扫描类文件,这些答案读取类文件的魔术字节。

但是,有些代码是打包在JAR或其他归档格式(如WAR和EAR )中的,其中一些包含其他归档或类文件,再加上您现在拥有多版本的JAR文件-请参阅JEP-238,它对每个JAR使用不同的JDK编译器。

此程序从文件+文件夹列表中扫描类,并打印每个组件的java类文件版本摘要,包括WAR/EARs中的每个JAR:

代码语言:javascript
运行
复制
public static void main(String[] args) throws IOException {
    var files = Arrays.stream(args).map(Path::of).collect(Collectors.toList());
    ShowClassVersions v = new ShowClassVersions();
    for (var f : files) {
        v.scan(f);
    }
    v.print();
}

扫描的输出示例:

代码语言:javascript
运行
复制
Version: 49.0 ~ JDK-5
   C:\jars\junit-platform-console-standalone-1.7.1.jar
Version: 50.0 ~ JDK-6
   C:\jars\junit-platform-console-standalone-1.7.1.jar
Version: 52.0 ~ JDK-8
   C:\java\apache-tomcat-10.0.12\lib\catalina.jar
   C:\jars\junit-platform-console-standalone-1.7.1.jar
Version: 53.0 ~ JDK-9
   C:\java\apache-tomcat-10.0.12\lib\catalina.jar
   C:\jars\junit-platform-console-standalone-1.7.1.jar

扫描仪:

代码语言:javascript
运行
复制
public class ShowClassVersions {
    private TreeMap<String, ArrayList<String>> vers = new TreeMap<>();
    private static final byte[] CLASS_MAGIC = new byte[] { (byte) 0xca, (byte) 0xfe, (byte) 0xba, (byte) 0xbe };
    private final byte[] bytes = new byte[8];

    private String versionOfClass(InputStream in) throws IOException  {
        int c = in.readNBytes(bytes, 0, bytes.length);
        if (c == bytes.length && Arrays.mismatch(bytes, CLASS_MAGIC) == CLASS_MAGIC.length) {
            int minorVersion = (bytes[4] << 8) + (bytes[4] << 0);
            int majorVersion = (bytes[6] << 8) + (bytes[7] << 0);
            return ""+ majorVersion + "." + minorVersion;
        }
        return "Unknown";
    }

    private Matcher classes = Pattern.compile("\\.(class|ear|war|jar)$").matcher("");

    // This code scans any path (dir or file):
    public void scan(Path f) throws IOException {
        try (var stream = Files.find(f, Integer.MAX_VALUE,
                (p, a) -> a.isRegularFile() && classes.reset(p.toString()).find())) {
            stream.forEach(this::scanFile);
        }
    }

    private void scanFile(Path f) {
        String fn = f.getFileName().toString();
        try {
            if (fn.endsWith(".ear") || fn.endsWith(".war") || fn.endsWith(".jar"))
                scanArchive(f);
            else if (fn.endsWith(".class"))
                store(f.toAbsolutePath().toString(), versionOfClass(f));
        } catch (IOException e) {
            throw new UncheckedIOException(e);
        }
    }

    private void scanArchive(Path p) throws IOException {
        try (InputStream in = Files.newInputStream(p)) {
            scanArchive(p.toAbsolutePath().toString(), Files.newInputStream(p));
        }
    }

    private void scanArchive(String desc, InputStream in) throws IOException {
        HashSet<String> versions = new HashSet<>();
        ZipInputStream zip = new ZipInputStream(in);
        for (ZipEntry entry = null; (entry = zip.getNextEntry()) != null; ) {
            String name = entry.getName();
            // There could be different compiler versions per class in one jar
            if (name.endsWith(".class")) {
                versions.add(versionOfClass(zip));
            } else if (name.endsWith(".jar") || name.endsWith(".war")) {
                scanArchive(desc + " => " + name, zip);
            }
        }
        if (versions.size() > 1)
            System.out.println("Warn: "+desc+" contains multiple versions: "+versions);

        for (String version : versions)
            store(desc, version);
    }

    private String versionOfClass(Path p) throws IOException {
        try (InputStream in = Files.newInputStream(p)) {
            return versionOfClass(in);
        }
    }

    private void store(String path, String jdkVer) {
        vers.computeIfAbsent(jdkVer, k -> new ArrayList<>()).add(path);
    }

    // Could add a mapping table for JDK names, this guesses based on (JDK17 = 61.0)
    public void print() {
        for (var ver : vers.keySet()) {
            System.out.println("Version: " + ver + " ~ " +jdkOf(ver));
            for (var p : vers.get(ver)) {
                System.out.println("   " + p);
            }
        }
    }

    private static String jdkOf(String ver)  {
        try {
            return "JDK-"+((int)Float.parseFloat(ver)-44);
        }
        catch(NumberFormatException nfe)
        {
            return "JDK-??";
        }
    }
}
票数 0
EN
查看全部 6 条回答
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/27065

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档