文章目录

  • 类型映射
  • 环境准备
  • 创建目标与代理
  • 目标Java方法
  • PB代理类
  • 调用流程
  • 加载JVM
  • 创建PB代理对象
  • 调用代理对象方法
  • 示例
  • 示例1:调用Java Class字节码
  • 示例2:调用依赖JAR包的Java Class字节码
  • 资源下载


类型映射

Java与PB中方法的参数和返回值类型映射关系如下表:

Java类型

PB类型

byte

byte

short

int

int

long

long

longlong

float

real

double

double

char

char

boolean

boolean

java.lang.String

string

以上类型数组

对应类型数组

环境准备

PB12.5开发环境
D:\Java目录拷贝32位的jre1.8
编译Java代码需要安装jdk1.8
PB安装目录或PB项目目录下拷贝动态库pbjni.dll,右键pbl选择Import PB Extension…导入动态库

创建目标与代理

目标Java方法

创建目录D:\test,新建HelloPBJNI.java文件,代码如下:

package test;

public class HelloPBJNI {
    public String hello(String name) {
        return "Hello " + name;
    }
}

编译Java文件生成字节码:

javac -parameters test\HelloPBJNI.java

PBKDF java pbe 指定 java调用pb模型_powerbasic


获取目标方法hello的签名:

javap -s test.HelloPBJNI

PBKDF java pbe 指定 java调用pb模型_java_02


上图中红框部分就是hello方法的签名:

(Ljava/lang/String;)Ljava/lang/String;

PB代理类

PB项目中创建Custom Class,命名为n_hello_pbjni

PBKDF java pbe 指定 java调用pb模型_Java_03


编辑n_hello_pbjni源代码,用以下代码覆盖:

global type n_hello_pbjni from nonvisualobject
end type
global n_hello_pbjni n_hello_pbjni

forward prototypes
public:
function string hello( string name) alias for "hello,(Ljava/lang/String;)Ljava/lang/String;"
end prototypes

PB方法定义的入参和返回值需要使用PB中对应的类型,alias for 的内容就是Java的目标方法名和方法签名。

调用流程

加载JVM

窗口实例变量中增加定义:

PBKDF java pbe 指定 java调用pb模型_Java_04


窗口open事件加载JVM:

string  classpath = "d:\;"
jvm = create cjavavm
jvm.createjavavm("D:\Java\jre\bin\client\jvm.dll", classpath)

PBKDF java pbe 指定 java调用pb模型_jni_05

创建PB代理对象

long r

n_hello_pbjni hellopbjni
r = jvm.createjavaobject(hellopbjni, "test/HelloPBJNI", "n_hello_pbjni")

调用代理对象方法

string s
s = hellopbjni.hello("PBJNI")

调用效果如下:

PBKDF java pbe 指定 java调用pb模型_jni_06

示例

示例1:调用Java Class字节码

Java代码:

package test;

public class PBJNIDemo1 {
    
    public String helloByList(String[] nameList) {
        StringBuilder result = new StringBuilder();
        for (String name : nameList) {
            result.append("Hello ");
            result.append(name);
            result.append("! ");
        }
        return result.toString();
    }
    
    public String repeatString(String s, int times) {
        StringBuilder result = new StringBuilder();
        for (int i = 0; i < times; i++) {
            result.append(s);
        }
        return result.toString();
    }
	
	public String[] helloReturnList(String[] nameList) {
		for (int i = 0; i < nameList.length; i++) {
			nameList[i] = "Hello " + nameList[i] + "!";
		}
		return nameList;
	}
}

编译并获取方法签名:

PBKDF java pbe 指定 java调用pb模型_java_07

创建PB代理类:

global type n_demo1 from nonvisualobject
end type
global n_demo1 n_demo1

forward prototypes
public:
function string helloByList( string nameList[]) alias for "helloByList,([Ljava/lang/String;)Ljava/lang/String;"
function string repeatString( string s, long times) alias for "repeatString,(Ljava/lang/String;I)Ljava/lang/String;"
function any helloReturnList( string nameList[]) alias for "helloReturnList,([Ljava/lang/String;)[Ljava/lang/String;"
end prototypes

PB中调用:

PBKDF java pbe 指定 java调用pb模型_Java_08

示例2:调用依赖JAR包的Java Class字节码

调用bouncycastle的Base64编码解码:

package test;

import org.bouncycastle.util.encoders.Base64;

public class PBJNIDemo2 {

    public String base64Encode(byte[] bytes) {
        return new String(Base64.encode(bytes));
    }

    public byte[] base64Decode(String base64) {
        return Base64.decode(base64);
    }
}

编译时classpath中要添加依赖包

编译并获取方法签名:

PBKDF java pbe 指定 java调用pb模型_Java_09


创建PB代理类:

global type n_demo2 from nonvisualobject
end type
global n_demo2 n_demo2

forward prototypes
public:
function string base64Encode( byte bytes[]) alias for "base64Encode,([B)Ljava/lang/String;"
function any base64Decode( string base64) alias for "base64Decode,(Ljava/lang/String;)[B"
end prototypes

加载JVM时需要把引用的jar包添加到classpath中:

PBKDF java pbe 指定 java调用pb模型_jni_10


PB中调用:

PBKDF java pbe 指定 java调用pb模型_powerbasic_11

资源下载

pbjni.dll