Java .class 버전을 읽고 표시하는 도구


115

.class 파일을 검색 한 다음 컴파일 된 버전을 표시하는 도구를 아는 사람이 있습니까?

16 진 편집기에서 개별적으로 볼 수 있다는 것을 알고 있지만 살펴볼 클래스 파일이 많이 있습니다.


1
더 인기있는 중복 stackoverflow.com/questions/1096148/… 여기에 언급되지 않은 몇 가지 편리한 도구가 있습니다.
Vadzim 2014-06-04

답변:


142

JDK와 함께 제공 되는 javap 도구를 사용하십시오 . 이 -verbose옵션은 클래스 파일의 버전 번호를 인쇄합니다.

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

버전 만 표시하려면 :

WINDOWS> javap -verbose MyClass | find "version"
LINUX  > javap -verbose MyClass | grep version

2
버전 major.minor = JDK / JavaSE; 45.3 = JDK1.1; 46.0 = JDK1.2; 47.0 = JDK1.3; 48.0 = JDK1.4; 49.0 = 자바 SE5 (1.5); 51.0 = 자바 SE7 (1.7); 50.0 = 자바 SE6 (1.6); 52.0 = 자바 SE8 (1.8); 53.0 = 자바 SE9; 54.0 = 자바 SE10; 55.0 = 자바 SE11; 56.0 = 자바 SE12; 57.0 = 자바 SE13; 58.0 = 자바 SE14;
nephewtom 19

45

클래스 파일 서명 을 읽고 타사 API없이 이러한 값을 얻는 것은 쉽습니다 . 당신이해야 할 일은 처음 8 바이트를 읽는 것입니다.

ClassFile {
    u4 magic;
    u2 minor_version;
    u2 major_version;

클래스 파일 버전 51.0 (Java 7)의 경우 여는 바이트는 다음과 같습니다.

CA FE BA BE 00 00 00 33

... 여기서 0xCAFEBABE는 매직 바이트, 0x0000은 부 버전, 0x0033은 주 버전입니다.

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);
    }
  }
}

클래스 파일을 찾는 워킹 디렉토리 ( File )와 아카이브 ( JarFile )는 간단합니다.

Oracle의 Joe Darcy의 블로그 에는 Java 7까지의 JDK 버전 매핑에 대한 클래스 버전이 나열되어 있습니다 .

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

또한 assert는 Java를 시작할 때 활성화 된 경우에만 실행되므로 IllegalArgumentException을 사용하지 않는 경우 (예
:)

21

유닉스 계열

/path/to/Thing.class 파일

파일 유형과 버전도 제공합니다. 출력은 다음과 같습니다.

컴파일 된 Java 클래스 데이터, 버전 49.0


(WMR의 답변에서 단순화)
phunehehe 2011-07-07

이것은 다른 솔루션보다 훨씬 간단하다
mmuller

9

유닉스 시스템을 사용하는 경우 다음을 수행 할 수 있습니다.

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

(내 파일 버전은 "컴파일 된 Java 클래스 데이터, 버전 50.0"(java6 클래스의 경우))입니다.


macOS (최소 10.12.6)에서는 출력이 훨씬 더 유용합니다. 다음을 file *.class 생성합니다. ClassName.class: compiled Java class data, version 50.0 (Java 1.6)
Gary

5

또 다른 자바 버전 확인

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

5

이클립스에서 소스가 첨부되지 않은 경우. 소스 첨부 버튼 다음의 첫 번째 줄에 유의하십시오.

// CDestinoLog.java에서 컴파일 됨 ( 버전 1.5 : 49.0, 수퍼 비트 )

여기에 이미지 설명 입력


2

아마도 이것은 누군가에게 도움이 될 것입니다. .class를 컴파일 / 빌드하는 데 사용되는 JAVA 버전을 얻는 더 쉬운 방법이있는 것 같습니다. 이 방법은 JAVA 버전에서 응용 프로그램 / 클래스 자체 검사에 유용합니다.

JDK 라이브러리를 살펴본 결과 com.sun.deploy.config.BuiltInProperties.CURRENT_VERSION 이 유용한 상수를 찾았습니다 . JAVA JDK에서 언제부터인지 모르겠습니다.

여러 버전 상수에 대해이 코드를 시도하면 아래 결과가 나타납니다.

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() );

산출:

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

클래스 바이트 코드에는 실제로 저장된 상수가 있습니다. Main.call의 빨간색으로 표시된 부분을 참조하십시오 . .class 바이트 코드에 저장된 상수

상수는 JAVA 버전이 오래되었는지 확인하는 데 사용되는 클래스에 있습니다 ( Java가 오래된 버전을 확인 하는 방법 참조 ) ...


1

버전 매직 넘버를 사용하는 자바 기반 솔루션 . 아래에서는 프로그램 자체에서 바이트 코드 버전을 감지하는 데 사용됩니다.

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));
    }
}

0

이 Java 클래스는 디렉토리 목록 아래에있는 모든 WAR 컨텐츠 및 JAR의 컨텐츠를 스캔하고 WAR 내의 각 JAR을 포함하여 각 구성 요소에 대한 Java 클래스 파일 버전의 요약을 인쇄합니다.

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);
            }
        };
    }
}
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.