【问题标题】:Read bytes from a Class file within a Jar file从 Jar 文件中的 Class 文件中读取字节
【发布时间】:2015-07-17 18:56:43
【问题描述】:

我有一个.jar 文件,其中包含.class 文件和.java 文件。我想将特定.class 文件的内容加载为byte[] 数组。

static byte[] getBytes(String javaFileName, String jar) throws IOException {
  try (JarFile jarFile = new JarFile(jar)) {
    Enumeration<JarEntry> entries = jarFile.entries();
    while (entries.hasMoreElements()) {
      JarEntry entry = entries.nextElement();

      // We are only interested in .class files spawned by compiling the argument file.
      if (entry.getName().endsWith(".class") &&
          entry.getName().contains(Files.getNameWithoutExtension(javaFileName))) {
        try (InputStream inputStream = jarFile.getInputStream(entry)) {
          return Preconditions.checkNotNull(ByteStreams.toByteArray(inputStream));
        } catch (IOException ioException) {
          System.out.println("Could not obtain class entry for " + entry.getName());
          throw ioException;
        }
      }
    }
  }
}

但是,当我运行此代码时,会返回一个空字节数组 (new byte[0])。关于我在这里可能做错的任何想法?

编辑:此代码运行良好!错误出现在问题的另一部分。无论如何,为了知识而将问题保留在这里:)

【问题讨论】:

    标签: java guava bytestream


    【解决方案1】:

    这对我有用:

    public class Main {
        public static void main(String[] args) {
            try {
                byte[] bytes = getBytes("MeasurePrinter.class", "~/BundleSensorAlarmLib.jar");
                System.out.println(bytesToHex(bytes));
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }
    
        private static byte[] getBytes(String javaFileName, String jar) throws IOException {
            // ... inputs check omitted ...
            try (JarFile jarFile = new JarFile(jar)) {
                Enumeration<JarEntry> entries = jarFile.entries();
                while (entries.hasMoreElements()) {
                    JarEntry entry = entries.nextElement();
    
                    // We are only interested in .class files spawned by compiling the argument file.
                    if (entry.getName().endsWith(".class") &&
                            entry.getName().contains(javaFileName.substring(0, javaFileName.lastIndexOf(".")))) {
                        try (InputStream inputStream = jarFile.getInputStream(entry)) {
                            return getBytes(inputStream);
                        } catch (IOException ioException) {
                            System.out.println("Could not obtain class entry for " + entry.getName());
                            throw ioException;
                        }
                    }
                }
            }
            throw new IOException("File not found");
        }
    
        private static byte[] getBytes(InputStream is) throws IOException {
            try (ByteArrayOutputStream os = new ByteArrayOutputStream();)
            {
                byte[] buffer = new byte[0xFFFF];
                for (int len; (len = is.read(buffer)) != -1;)
                    os.write(buffer, 0, len);
                os.flush();
                return os.toByteArray();
            }
        }
    
        private static char[] hexArray = "0123456789ABCDEF".toCharArray();
        public static String bytesToHex(byte[] bytes) {
            char[] hexChars = new char[bytes.length * 2];
            for ( int j = 0; j < bytes.length; j++ ) {
                int v = bytes[j] & 0xFF;
                hexChars[j * 2] = hexArray[v >>> 4];
                hexChars[j * 2 + 1] = hexArray[v & 0x0F];
            }
            return new String(hexChars);
        }
    }
    

    从某种意义上说,返回的字节数组不为空。

    【讨论】:

    • 显然上面的代码(我发布的那个也有效!我在程序的另一部分搞砸了。不过感谢您的回答!:D
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2011-10-11
    • 1970-01-01
    • 2012-10-09
    • 2014-05-24
    • 1970-01-01
    • 2011-01-17
    相关资源
    最近更新 更多