11

Java 本机接口 (JNI) 的一个组成部分是通过 C 标头桥接 JVM 代码和本机代码。生成这些头文件的方法过去非常简单:只需javah在类文件上调用命令行实用程序。然后,此过程将为任何标有native修饰符的方法生成原型。

然而,从 Java 10 开始,该javah实用程序已被删除,建议将其替换为 javac 的新标志“-h”。如果有可用的 Java 源文件,则替换工作正常,但是在只有编译的类文件可用的情况下就不足了。(引发这个问题的问题是我正在尝试从 Scala 源代码生成 JNI 绑定。我目前的方法是先编译它们,然后在生成的类文件上运行 javah。)

在只有编译的类文件可用的情况下,有没有办法生成 C 头文件,类似于以前的方式javah

4

3 回答 3

7

你总是可以通过javap. 我知道我知道。它很丑,有很多假设,但如果你迫切需要为大量文件生成标题,它可能是唯一的选择。

#!/bin/bash

# FIRST_ARG - full class name (with package)
# SECOND_ARG - class path

CLASS_NAME=`javap -cp $2 $1 | \
  grep -v "Compiled from" | \
  grep "public class" | \
  cut -f3 -d" " | \
  awk -F"." '{ print $NF }'`

PACKAGE_NAME=`javap -cp $2 $1 | \
  grep -v "Compiled from" | \
  grep "public class" | \
  cut -f3 -d" " | \
  sed s/\.${CLASS_NAME}$//`

DIR_NAME=`echo $PACKAGE_NAME | sed 's|\.|/|g'`
mkdir -p java_jni/${DIR_NAME}

JAVA_FILE_NAME="java_jni/${DIR_NAME}/${CLASS_NAME}.java"

echo "package ${PACKAGE_NAME};" > ${JAVA_FILE_NAME}
echo "public class ${CLASS_NAME} {" >> ${JAVA_FILE_NAME}

javap -cp $2 $1 | grep "native" | while read line; do
  param=0
  comma=`echo $line | grep "," | wc -l`
  while [ $comma -gt 0 ]; do
    line=`echo $line | sed "s/,/ param_${param}|/"`
    let param=param+1
    comma=`echo $line | grep "," | wc -l`
  done
  line=`echo $line | sed "s/)/ param_${param})/" | sed 's/|/,/g'`
  echo "  $line" >> ${JAVA_FILE_NAME}
done

echo "}" >> ${JAVA_FILE_NAME}

mkdir -p c_header
javac -h c_header ${JAVA_FILE_NAME}

我敢打赌它可以变得更漂亮。

对我来说,现在,当我慢慢开始考虑不可避免地转向 Java 10 时,以及所有这些情况下,我可能会对不存在的 Java 源代码感到惊讶,我认为拥有一些可供我使用的工具并不是一个坏主意。以防万一。

于 2018-03-27T11:49:17.120 回答
6

我们可以使用gjavah来生成 JNI 头文件。

于 2018-04-11T07:08:25.917 回答
-5

我认为最好的解决方案就是安装一个jdk8。而且不需要卸载jdk10,修改环境变量即可。

于 2018-06-26T04:23:59.830 回答