cvi怎么调用lib_并发系列-Java调用native函数过程原理实践

本文探讨了Java中线程启动的start方法,详细解析了如何通过JNI在Linux环境下调用pthread_create函数创建线程。通过编写C程序并将其编译为动态链接库,最终成功在Java中调用了C程序,实现了Java native方法调用。
摘要由CSDN通过智能技术生成

写在前面

Java中有很多native方法,今天主要分享线程启动的start方法。

f1d55f5c65cb0b37a3530f7b7cf92f92.png

线程启动方法

e4d562b37adc24e619d10810daecc9e1.png

调用start0方法

2c6fac70ad03169103eec4e925d76e96.png

start0是一个native方法

那么start0方法到底是怎么执行的?接下来我们自己写一个native方法,通过一系列执行流程来进行解释。

Linux创建线程

所谓native方法就行Java调用操作系统来运行的方法,也就是我们启动的线程是由linux系统来创建的,linux下创建线程函数

int pthread_create(pthread_t *thread, const pthread_attr_t *attr,void *(*start_routine) (void *), void *arg);

这个函数是linux系统的函数,可以用C或者C++直接调用,这个函数有四个参数

e500808a8bfbe6514a4c5e7d911e8a6c.png

pthread_create参数

在linux上启动一个线程的代码:

9152e3e6d0a56a9545f4bb2817c8d32f.png

启动线程

编译C文件

gcc thread.c -o thread.out -pthread./thread.out
322a5d3660b44a3df3d90a4d0023af58.png

编译后目录

033713ac8591b3427cfcd864e31585f4.png

刚装的虚拟机,缺少gcc,先安装

yum -y install gcc gcc-c++ autoconf pcre pcre-devel make automake

d33a86fbdbae88c4b9d4519a03413fb5.png
32c0cdd9d0a3c932bcb2b67628845ce5.png

yum -y install wget httpd-tools vim

7ab98e7931611834c88f1857a14bf27b.png

安装完gcc,验证版本

bc3668a97611b4fcac10dc410d17b1c0.png

再次运行编译,打印结果

6159fea5119049b1af9c1f64f1f92b5f.png

采用JNI用java程序去调用这个C文件

编写java程序

package org.xinhua.cbcloud.util;public class ThreadTest {    static {        // 装载库,保证JVM在启动的时候就会装载,故而一般是也给static        System.loadLibrary( "ThreadTestNative" );    }    public static void main(String[] args) {        ThreadTest threadTest =new ThreadTest();        threadTest.start0();    }    private native void start0();}

编译java文件发现才想起来没有装jdk,顺便记录下安装过程

19ced03b4b27de09bd3ac1953a1aa4ab.png
yum -y list java*yum -y list java* yum install -y java-1.8.0-openjdk.x86_64
4d5c3281c5190d5d01bf74f01c63f7f1.png

默认的安装目录

/usr/lib/jvm/java-1.8.0-openjdk-1.8.0.242.b08-0.el7_7.x86_64
bad051f8b31e69ac1c773a3574728274.png

配环境变量,去vim /etc/profile 下面加一下就行

export JAVA_HOME=/usr/lib/jvm/java-1.8.0-openjdk-1.8.0.242.b08-0.el7_7.x86_64export CLASSPATH=.:$JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar:$JAVA_HOME/jre/lib/rt.jarexport PATH=$PATH:$JAVA_HOME/bin
018d1a052c597ef0de1a85d739ef440c.png
source /etc/profile

测试java -version可用,javac不可用

第一次遇到这个情况,网上大神说用

yum install java-devel

亲测确实可用

fa6c2184394e5dc07919a737de2798ab.png

生成.h头文件

javah org.xinhua.cbcloud.util.ThreadTest

注意:执行javac和javah的目录是不一样的

5233f924822f97b36a2908a1af94112a.png

定义一个方法Java_org_xinhua_cbcloud_util_ThreadTest_start0,在方法中启动一个子线程,代码如下

#include #include #include "org_xinhua_cbcloud_util_ThreadTest.h"//记得导入刚刚编译的那个.h文件pthread_t pid;void* thread_entity(void* arg){       while(1){usleep(100);    printf("I am new Thread");    }}JNIEXPORT void JNICALL Java_org_xinhua_cbcloud_util_ThreadTest_start0(JNIEnv *env, jobject c1){pthread_create(&pid,NULL,thread_entity,NULL);    while(1){usleep(100);printf("I am  main");}}int main(){    return 0;}

解析类,把这个thread2.c编译成为一个动态链接库,这样在java代码里会被laod到内存

libLubanThreadNative这个命名需要注意libxx,xx就等于你java那边写的字符串

gcc -fPIC -I /usr/lib/jvm/java-1.8.0-openjdk/include -I /usr/lib/jvm/java-1.8.0-openjdk/include/linux -shared -o libThreadTestNative.so thread2.c
28fc2ebbabafa5328ca91df5b14be7d1.png

做完这一系列事情之后需要把这个.so文件加入到path,这样java才能load到

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/lyfeifei

现在直接运行java程序

9add4ee6bfc302b4c40707c2a22c411c.png

运行结果:

4b18b687f424c571b7c2b7eed18ca01e.png

写在最后

已经成功的调用到了我们写的C程序,结果不重要。最好是亲自动手试试,印象会比较深刻一些。本章的主要目的是通过Java方法调用C程序,那么启动之后我们的run方法是怎么执行的呢?下一篇会分享C程序如何回调我们的Java程序。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值