java linux 调用32位so_并发系列-Java调用native函数过程原理实践

写在前面

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

e6f902075ea6dbcc4c8f6c675314cafd.png

线程启动方法

3d4aa250ab32f7cded3fee2152238cbd.png

调用start0方法

6a14c233233ec37498391fb76e0c08ea.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++直接调用,这个函数有四个参数

65092a563458c19fb603570f892c91bf.png

pthread_create参数

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

8237b2c4f7c0079f85c74b32ace356cc.png

启动线程

编译C文件

gcc thread.c -o thread.out -pthread./thread.out
415026a5bcd6fbe1c861d36e74eabedb.png

编译后目录

531a74722fb56a976fbc21bddb04f684.png

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

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

52cd388c51016da3dafd2beecef0eddc.png
531b73d8b789c156cca84f22741dcbba.png

yum -y install wget httpd-tools vim

721b29eff0c3b72ae9b14a4ac4565d39.png

安装完gcc,验证版本

db74b93ca2320e11ebe776e44c29f152.png

再次运行编译,打印结果

d1e8be9c6aa5fd81edb146416fa0af79.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,顺便记录下安装过程

6f1bfd2d91360ca0ddfc46f594690660.png
yum -y list java*yum -y list java* yum install -y java-1.8.0-openjdk.x86_64
afdbb6b6b2d2f1726874b173f5f08179.png

默认的安装目录

/usr/lib/jvm/java-1.8.0-openjdk-1.8.0.242.b08-0.el7_7.x86_64
b4d013a39fc7132487ffc80e1e0067d5.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
c827b2dc61d178e020a8c360bf15b231.png
source /etc/profile

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

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

yum install java-devel

亲测确实可用

87632c726475f8d98f5c32e3acdd87d9.png

生成.h头文件

javah org.xinhua.cbcloud.util.ThreadTest

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

355422faf08664a92bdd8d3dd73968ec.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
94e5ffc4b785a0b85e3b4246f83a4893.png

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

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/lyfeifei

现在直接运行java程序

dfa3c1b842015e840b4aa0a074ec8d60.png

运行结果:

a0de0c523c1d28e79f3f35ac3394fc8e.png

写在最后

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值