Tool zum Lesen und Anzeigen von Java .class-Versionen

115

Kennt jemand von euch ein Tool, das nach .class-Dateien sucht und dann deren kompilierte Versionen anzeigt?

Ich weiß, dass Sie sie einzeln in einem Hex-Editor anzeigen können, aber ich muss viele Klassendateien überprüfen (etwas in meiner riesigen Anwendung wird aus irgendeinem Grund auf Java6 kompiliert).

SCdF
quelle
1
Das populärere Duplikat stackoverflow.com/questions/1096148/… hat einige nützliche Tools beantwortet, die hier nicht erwähnt werden.
Vadzim

Antworten:

142

Verwenden Sie das mit dem JDK gelieferte Javap- Tool. Die -verboseOption druckt die Versionsnummer der Klassendatei.

> javap -verbose MyClass
Compiled from "MyClass.java"
public class MyClass
  SourceFile: "MyClass.java"
  minor version: 0
  major version: 46
...

So zeigen Sie nur die Version an:

WINDOWS> javap -verbose MyClass | find "version"
LINUX  > javap -verbose MyClass | grep version
Mitarbeiter
quelle
2
Version major.minor = JDK / JavaSE; 45,3 = JDK1,1; 46,0 = JDK1,2; 47,0 = JDK1,3; 48,0 = JDK1,4; 49,0 = JavaSE5 (1,5); 51,0 = JavaSE7 (1,7); 50,0 = JavaSE6 (1,6); 52,0 = JavaSE8 (1,8); 53,0 = JavaSE9; 54,0 = JavaSE10; 55,0 = JavaSE11; 56,0 = JavaSE12; 57,0 = JavaSE13; 58,0 = JavaSE14;
Neffe vom
45

Es ist einfach genug, die Signatur der Klassendatei zu lesen und diese Werte ohne eine API eines Drittanbieters abzurufen. Sie müssen lediglich die ersten 8 Bytes lesen.

ClassFile {
    u4 magic;
    u2 minor_version;
    u2 major_version;

Für die Klassendateiversion 51.0 (Java 7) lauten die öffnenden Bytes:

CA FE BA BE 00 00 00 33

... wobei 0xCAFEBABE die magischen Bytes sind, 0x0000 die Nebenversion und 0x0033 die Hauptversion.

import java.io.*;

public class Demo {
  public static void main(String[] args) throws IOException {
    ClassLoader loader = Demo.class.getClassLoader();
    try (InputStream in = loader.getResourceAsStream("Demo.class");
        DataInputStream data = new DataInputStream(in)) {
      if (0xCAFEBABE != data.readInt()) {
        throw new IOException("invalid header");
      }
      int minor = data.readUnsignedShort();
      int major = data.readUnsignedShort();
      System.out.println(major + "." + minor);
    }
  }
}

Das Gehen von Verzeichnissen ( Datei ) und Archiven ( JarFile ) nach Klassendateien ist trivial.

In Joe Darcys Blog von Oracle werden die Klassenversionen für JDK-Versionszuordnungen bis Java 7 aufgelistet :

Target   Major.minor Hex
1.1      45.3        0x2D
1.2      46.0        0x2E
1.3      47.0        0x2F
1.4      48.0        0x30
5 (1.5)  49.0        0x31
6 (1.6)  50.0        0x32
7 (1.7)  51.0        0x33
8 (1.8)  52.0        0x34
9        53.0        0x35
McDowell
quelle
Denken Sie auch daran, dass assert nur ausgeführt wird, wenn es beim Starten von Java aktiviert ist, sodass Sie Junk-Dateien lesen können, wenn Sie nicht IllegalArgumentException (zum Beispiel) verwenden
jontejj
21

Auf Unix-ähnlich

Datei /path/to/Thing.class

Gibt auch den Dateityp und die Version an. So sieht die Ausgabe aus:

kompilierte Java-Klassendaten, Version 49.0

Phunehehe
quelle
(vereinfacht aus der Antwort von WMR)
Phunehehe
Dies ist viel einfacher als die anderen Lösungen
mmuller
9

Wenn Sie sich auf einem Unix-System befinden, können Sie einfach eine

find /target-folder -name \*.class | xargs file | grep "version 50\.0"

(Meine Version der Datei sagt "kompilierte Java-Klassendaten, Version 50.0" für Java6-Klassen).

WMR
quelle
Unter macOS (mindestens 10.12.6) ist die Ausgabe noch hilfreicher: file *.class Erzeugt: ClassName.class: compiled Java class data, version 50.0 (Java 1.6)
Gary
5

Noch eine Java-Versionsprüfung

od -t d -j 7 -N 1 ApplicationContextProvider.class | head -1 | awk '{print "Java", $2 - 44}'
ich gehe
quelle
5

In Eclipse, wenn Sie keine Quellen angehängt haben. Beachten Sie die erste Zeile nach der Schaltfläche zum Anhängen der Quelle.

// Kompiliert aus CDestinoLog.java ( Version 1.5: 49.0, Superbit )

Geben Sie hier die Bildbeschreibung ein

PbxMan
quelle
2

Vielleicht hilft das auch jemandem. Anscheinend gibt es eine einfachere Möglichkeit, die JAVA-Version zum Kompilieren / Erstellen von .class zu verwenden. Diese Methode ist nützlich, um die Anwendung / Klasse in der JAVA-Version selbst zu überprüfen.

Ich habe die JDK-Bibliothek durchgesehen und diese nützliche Konstante gefunden: com.sun.deploy.config.BuiltInProperties.CURRENT_VERSION . Ich weiß nicht, wann es in JAVA JDK ist.

Wenn ich diesen Code für mehrere Versionskonstanten versuche, erhalte ich das folgende Ergebnis:

src:

System.out.println("JAVA DEV       ver.: " + com.sun.deploy.config.BuiltInProperties.CURRENT_VERSION);
System.out.println("JAVA RUN     v. X.Y: " + System.getProperty("java.specification.version") );
System.out.println("JAVA RUN v. W.X.Y.Z: " + com.sun.deploy.config.Config.getJavaVersion() ); //_javaVersionProperty
System.out.println("JAVA RUN  full ver.: " + System.getProperty("java.runtime.version")  + " (may return unknown)" );
System.out.println("JAVA RUN       type: " + com.sun.deploy.config.Config.getJavaRuntimeNameProperty() );

Ausgabe:

JAVA DEV       ver.: 1.8.0_77
JAVA RUN     v. X.Y: 1.8
JAVA RUN v. W.X.Y.Z: 1.8.0_91
JAVA RUN  full ver.: 1.8.0_91-b14 (may return unknown)
JAVA RUN       type: Java(TM) SE Runtime Environment

Im Klassenbytecode ist wirklich eine Konstante gespeichert - siehe rot markierten Teil von Main.call - Konstante, die im Klassenbytecode gespeichert ist

Die Konstante wird in der Klasse verwendet, um zu überprüfen, ob die JAVA-Version veraltet ist (siehe Wie Java überprüft, ob sie veraltet ist ) ...

Radoslav Kastiel
quelle
1

Eine Java-basierte Lösung mit magischen Versionszahlen . Darunter wird es vom Programm selbst verwendet, um seine Bytecode-Version zu erkennen.

import java.io.IOException;
import java.io.InputStream;
import java.util.HashMap;
import java.util.Map;
import org.apache.commons.codec.DecoderException;
import org.apache.commons.codec.binary.Hex;
import org.apache.commons.io.IOUtils;
public class Main {
    public static void main(String[] args) throws DecoderException, IOException {
        Class clazz = Main.class;
        Map<String,String> versionMapping = new HashMap();
        versionMapping.put("002D","1.1");
        versionMapping.put("002E","1.2");
        versionMapping.put("002F","1.3");
        versionMapping.put("0030","1.4");
        versionMapping.put("0031","5.0");
        versionMapping.put("0032","6.0");
        versionMapping.put("0033","7");
        versionMapping.put("0034","8");
        versionMapping.put("0035","9");
        versionMapping.put("0036","10");
        versionMapping.put("0037","11");
        versionMapping.put("0038","12");
        versionMapping.put("0039","13");
        versionMapping.put("003A","14");

        InputStream stream = clazz.getClassLoader()
            .getResourceAsStream(clazz.getName().replace(".", "/") + ".class");
        byte[] classBytes = IOUtils.toByteArray(stream);
        String versionInHexString = 
            Hex.encodeHexString(new byte[]{classBytes[6],classBytes[7]});
        System.out.println("bytecode version: "+versionMapping.get(versionInHexString));
    }
}
Marinos An
quelle
0

Diese Java-Klasse durchsucht den Inhalt aller WAR-Inhalte und JARs, die in der Liste der Verzeichnisse aufgeführt sind, und druckt eine Zusammenfassung der Java-Klassendateiversionen für jede Komponente, einschließlich jeder JAR in WARs:

public class ShowClassVersions {
    private static final byte[] CLASS_MAGIC = new byte[] {(byte)0xca, (byte)0xfe, (byte)0xba, (byte)0xbe};
    private final byte[] bytes = new byte[8];
    private TreeMap<String,ArrayList<String>> vers = new TreeMap<>();

    private void scan(Path f) throws IOException {
        if (Files.isDirectory(f)) {
            Pattern pattern = Pattern.compile("\\.[wj]ar$"); // or |\\.class
            try(var stream = Files.find(f, Integer.MAX_VALUE, (p,a) -> a.isRegularFile() && pattern.matcher(p.toString()).find())) {
                stream.forEach(this::scanFile);
            }
            return;
        }
        scanFile(f);
    }
    private void scanFile(Path f) {
        String fn = f.getFileName().toString();
        try {
            if (fn.endsWith(".jar"))
                scanArchive(f);
            else if (fn.endsWith(".war"))
                scanArchive(f);
            else if (fn.endsWith(".class"))
                record(f, 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(), in);
        }
    }
    private String scanArchive(String desc, InputStream in) throws IOException {
        String version = null;
        ZipInputStream zip = new ZipInputStream(in);
        ZipEntry entry = null;
        while ((entry = zip.getNextEntry()) != null) {
            String name = entry.getName();
            if (version == null && name.endsWith(".class"))  {
                version = versionOfClass(zip);
            }
            else if (name.endsWith(".jar"))  {
                scanArchive(desc+" ==>> "+name, zip);
            }
        }
        if (version != null)
            record(desc, version);
        return version;
    }

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

    private String versionOfClass(InputStream in) throws IOException {
        String version = null;
        int c = in.read(bytes);
        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);
            version = ""+majorVersion + "." + minorVersion;
        }
        return version;
    }
    private void record(String p, String v) {
        vers.computeIfAbsent(String.valueOf(v), k -> new ArrayList<String>()).add(p);
    }
    private void record(Path p, String v) {
        record(p.toAbsolutePath().toString(), v);
    }
    public static void main(String[] args) throws IOException {
        ShowClassVersions v = new ShowClassVersions();
        var files = Arrays.stream(args).map(Path::of).collect(Collectors.toList());
        for (var f : files) {
            v.scan(f);
        }
        for (var ver : v.vers.keySet()) {
            System.out.println("Version: "+ver);
            for (var p : v.vers.get(ver)) {
                System.out.println("   "+p);
            }
        };
    }
}
DuncG
quelle