Click here to Skip to main content
15,881,898 members
Articles / Desktop Programming / Win32

How to Call Java Functions from C Using JNI

Rate me:
Please Sign up or sign in to vote.
4.47/5 (35 votes)
12 Jan 2008CPOL3 min read 429.6K   8K   46   84
This article covers calling Java functions from C using JNI. It also covers passing/returning simple parameters, arrays, and structure arrays in Java functions.

Introduction

This article describes the methodology to use Java code in C/C++. I have not only discussed simple parameter passing and returning, but complex data structures such as structures and structure arrays to Java functions as well.

Background

A few days ago, my manager asked me to write code in C/C++ that will use Java classes in other projects. It took me a lot of time to grab some information on how to call simple functions as well as pass/return complex data types. I found many articles describing C/C++ function calls in Java using JNI, but very few discussing calling Java functions in C/C++ using JNI. At that time, I decided to write an article so that other people could get help from it.

Using the code

The CTest.cpp file contains the code that calls different functions from Java classes. Java classes that are used in this project are given here under:

  • HelloWorld.java
  • ControlDetail.java
  • WorkOrder.java
  • ReturnData.java

HelloWorld.java contains the functions that will be called from CTest.cpp. The other three Java classes are simply used in place of structures in Java. As there is no structure concept in Java, we can use classes for that purpose. That is what the other three .java files contain.

HelloWorld.java contains the following functions that will be called from C/C++ code:

Java
public static void main(String args[])
{
}

public static void TestCall(String szArg)
{
} 

public static int DisplayStruct(ControlDetail ctrlDetail)
{
} 

public static void DisplayStructArray(WorkOrder ArrWO[])
{
} 

public static Object ReturnObjFunc()
{
}

To call these functions from C/C++, first you need to load the JVM using the following function:

Java
JNIEnv* create_vm(JavaVM ** jvm) {
    
    JNIEnv *env;
    JavaVMInitArgs vm_args;

    JavaVMOption options; 
    //Path to the java source code     
    options.optionString = "-Djava.class.path=D:\\Java Src\\TestStruct"; 
    vm_args.version = JNI_VERSION_1_6; //JDK version. This indicates version 1.6
    vm_args.nOptions = 1;
    vm_args.options = &options;
    vm_args.ignoreUnrecognized = 0;
    
    int ret = JNI_CreateJavaVM(jvm, (void**)&env, &vm_args);
    if(ret < 0)
        printf("\nUnable to Launch JVM\n");       
    return env;
}

Kindly note that to use this code, you will have to modify the options.optionString variable. You will have to set the path of the Java code: where the Java classes are placed. Currently, it being set to D:\Java Src\TestStruct. You can modify it to you situation. You will also need to modify the JDK version information in the above code, as shown below:

vm_args.version = JNI_VERSION_1_6; //JDK version. This indicates version 1.6

Modify it if you have another JDK version installed.

To call a specific Java function from C, you need to do the following:

  1. Obtain the class reference using the FindClass(,,) method.
  2. Obtain the method IDs of the functions of the class that you want to call using the GetStaticMethodID and GetMethodID function calls.
  3. Call the functions using CallStaticVoidMethod, CallStaticIntMethod, and CallStaticObjectMethod.

One important thing to be noted here is specifying the function signatures while obtaining the method IDs.

To obtain the correct method signature, you can use the following Java command:

javap -s -p HelloWorld

It will display you the signature of each function in the HelloWorld class. These signatures can be used to obtain the method IDs. The result of the above command can be seen below:

D:\Java Src\TestStruct>javap -s -p HelloWorld
Compiled from "HelloWorld.java"
public class HelloWorld extends java.lang.Object{
public HelloWorld();
  Signature: ()V
public static void main(java.lang.String[]);
  Signature: ([Ljava/lang/String;)V
public static void TestCall(java.lang.String);
  Signature: (Ljava/lang/String;)V
public static int DisplayStruct(ControlNEDetail);
  Signature: (LControlNEDetail;)I
public static void DisplayStructArray(WorkOrder[]);
  Signature: ([LWorkOrder;)V
public static java.lang.Object ReturnObjFunc();
  Signature: ()Ljava/lang/Object;
}

Kindly note that while specifying the method name in the GetMethodID function, if the method is a constructor, then its method name will be <init>.

Prerequisites

Before traveling down a difficult path, it is important to understand the basic concepts and to have various frameworks and tools installed on your computer.

  1. You will need the Sun Java Developer Kit (JDK). I recommend Java 1.6.0.
  2. Any C/C++ compiler installed.

How to run

To use this code, follow the instructions below:

  • Compile the *.java files using the javac command.
  • Compile the CTest.cpp file using any C++ compiler; I used MSVC++ 6.

Converting this code to pure C from C++

The attached code is written in C++. To convert this code into pure C, you will have to modify the following in the CTest.cpp file:

Use:

C++
(*env)->FunctionName(env,args,..);

instead of:

C++
env->FunctionName(args,..);

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)


Written By
Software Developer (Senior)
United Kingdom United Kingdom
Ahmad Qarshi is an IT Consultant and have good expertise in different development technologies. He loves to code in .NET, JavaFX, Delphi, XSLT/Schematron. Above all very keen to learn new stuff Smile | :)

Comments and Discussions

 
GeneralRe: My application crashes while calling JNI_CreateJavaVM function Pin
ajalilqarshi23-Feb-10 2:13
ajalilqarshi23-Feb-10 2:13 
GeneralRe: My application crashes while calling JNI_CreateJavaVM function [modified] Pin
Khalid Noor23-Feb-10 19:27
Khalid Noor23-Feb-10 19:27 
GeneralRe: My application crashes while calling JNI_CreateJavaVM function Pin
Member 41280638-Mar-11 0:20
Member 41280638-Mar-11 0:20 
GeneralRe: My application crashes while calling JNI_CreateJavaVM function Pin
pratikmota9-May-11 22:55
pratikmota9-May-11 22:55 
GeneralSOLVED: Error message: "Can't find dependent libraries" [modified] Pin
tsjakkaa29-Jan-10 2:25
tsjakkaa29-Jan-10 2:25 
Generalhelp me to sort out the error [modified] Pin
shariqmohd26-Jan-10 17:06
shariqmohd26-Jan-10 17:06 
GeneralRe: help me to sort out the error Pin
ajalilqarshi27-Jan-10 13:09
ajalilqarshi27-Jan-10 13:09 
QuestionNot working for me ...... pls help!!! Pin
kumarmukt20-Jan-10 3:20
kumarmukt20-Jan-10 3:20 
hi friend ... I got a compile time error. its saying ....

/tmp/cc06USvM.o: In function `create_vm':
CTest.cFrown | :( text+0x3a): undefined reference to `JNI_CreateJavaVM'
collect2: ld returned 1 exit status

I am using your c version code(after making the suggested change ).

even for c++ code ... getting similar type of error. This is what i am getting in c++ ...

/tmp/cc1Vr5Ol.o: In function `create_vm(JavaVM_**)':
CTest.cppFrown | :( text+0x3a): undefined reference to `JNI_CreateJavaVM'
/tmp/cc1Vr5Ol.oFrown | :( eh_frame+0x11): undefined reference to `__gxx_personality_v0'
collect2: ld returned 1 exit status

I have jdk1.6.0_18. I compile the code with given command in linux os.

gcc -I /usr/java/jdk1.6.0_18/include -I /usr/java/jdk1.6.0_18/include/linux CTest.c

here is the modified code ......


#include "StdAfx.h"
#include <stdio.h>
#include "jni.h"
#include <string.h>

#define PATH_SEPARATOR ';' /* define it to be ':' on Solaris */
#define USER_CLASSPATH "." /* where Prog.class is */

struct ControlDetail
{
int ID;
char Name[100];
char IP[100];
int Port;
};

struct WorkOrder
{
char sumSerialId[20];
char accessNumber[18];
char actionType[4];
char effectiveDate[24];
char fetchFlag[2];
char reason[456];
char accessSource[100];
};


JNIEnv* create_vm(JavaVM ** jvm) {

JNIEnv *env;
JavaVMInitArgs vm_args;
JavaVMOption options;
// options.optionString = "-Djava.class.path=D:\\Java Src\\TestStruct"; //Path to the java source code
options.optionString = "-Djava.class.path=/home/mukesh/c_room/Java Src/TestStruct"; //Path to the java source code
vm_args.version = JNI_VERSION_1_6; //JDK version. This indicates version 1.6
vm_args.nOptions = 1;
vm_args.options = &options;
vm_args.ignoreUnrecognized = 0;

int ret = JNI_CreateJavaVM(jvm, (void**)&env, &vm_args);
if(ret < 0)
printf("\nUnable to Launch JVM\n");
return env;
}

int main(int argc, char* argv[])
{
JNIEnv *env;
JavaVM * jvm;
env = create_vm(&jvm);
if (env == NULL)
return 1;

struct ControlDetail ctrlDetail;
ctrlDetail.ID = 11;
strcpy(ctrlDetail.Name,"HR-HW");
strcpy(ctrlDetail.IP,"10.32.164.133");
ctrlDetail.Port = 9099;

printf("Struct Created in C has values:\nID:%d\nName:%s\n IP:%s\nPort:%d\n",ctrlDetail.ID,ctrlDetail.Name,ctrlDetail.IP,ctrlDetail.Port);

/********************************************************/
struct WorkOrder WO[2];
strcpy(WO[0].sumSerialId,"2000");
strcpy(WO[0].accessNumber,"2878430");
strcpy(WO[0].actionType,"04");
strcpy(WO[0].effectiveDate,"25-12-2007 12:20:30 PM");
strcpy(WO[0].fetchFlag, "0");
strcpy(WO[0].reason,"Executed Successfully");
strcpy(WO[0].accessSource,"PMS");
strcpy(WO[1].sumSerialId,"1000");
strcpy(WO[1].accessNumber,"2878000");
strcpy(WO[1].actionType,"T4");
strcpy(WO[1].effectiveDate,"25-12-2007 11:20:30 PM");
strcpy(WO[1].fetchFlag,"0");
strcpy(WO[1].reason,"");
strcpy(WO[1].accessSource,"RMS");


jclass clsH=NULL;
jclass clsC = NULL;
jclass clsW = NULL;
jclass clsR = NULL;
jmethodID midMain = NULL;
jmethodID midCalling = NULL;
jmethodID midDispStruct = NULL;
jmethodID midDispStructArr = NULL;
jmethodID midRetObjFunc = NULL;
jmethodID midCtrlDetConst = NULL;
jmethodID midWoConst = NULL;

jobject jobjDet = NULL;
jobject jobjRetData = NULL;
jobjectArray jobjWOArr = NULL;

//Obtaining Classes
clsH = (*env)->FindClass(env, "HelloWorld");
clsC = (*env)->FindClass(env, "ControlDetail");
clsW = (*env)->FindClass(env, "WorkOrder");

//Obtaining Method IDs
if (clsH != NULL)
{
midMain = (*env)->GetStaticMethodID(env, clsH, "main", "([Ljava/lang/String;)V");
midCalling = (*env)->GetStaticMethodID(env, clsH,"TestCall","(Ljava/lang/String;)V");
midDispStruct = (*env)->GetStaticMethodID(env, clsH,"DisplayStruct","(LControlDetail;)I");
midDispStructArr = (*env)->GetStaticMethodID(env, clsH,"DisplayStructArray","([LWorkOrder;)V");
midRetObjFunc = (*env)->GetStaticMethodID(env, clsH,"ReturnObjFunc","()Ljava/lang/Object;");
}
else
{
printf("\nUnable to find the requested class\n");
}
if(clsC != NULL)
{
//Get constructor ID for ControlDetail
midCtrlDetConst = (*env)->GetMethodID(env, clsC, "<init>", "(ILjava/lang/String;Ljava/lang/String;I)V");
}
else
{
printf("\nUnable to find the requested class\n");
}

if(clsW != NULL)
{
//Get Constructor ID for WorkOrder
midWoConst = (*env)->GetMethodID(env, clsW, "<init>", "(Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;)V");
}
else
{
printf("\nUnable to find the requested class\n");
}

/************************************************************************/
/* Now we will call the functions using the their method IDs */
/************************************************************************/
if(midMain != NULL)
(*env)->CallStaticVoidMethod(env, clsH, midMain, NULL); //Calling the main method.

if (midCalling!=NULL)
{
jstring StringArg = (*env)->NewStringUTF(env, "\nTestCall:Called from the C Program\n");
//Calling another static method and passing string type parameter
(*env)->CallStaticVoidMethod(env, clsH,midCalling,StringArg);
}

printf("\nGoing to Call DisplayStruct\n");
if (midDispStruct!=NULL)
{
if(clsC != NULL && midCtrlDetConst != NULL)
{
jstring StringArgName = (*env)->NewStringUTF(env, ctrlDetail.Name);
jstring StringArgIP = (*env)->NewStringUTF(env, ctrlDetail.IP);

//Creating the Object of ControlDetail.
jobjDet = (*env)->NewObject(env, clsC, midCtrlDetConst, (jint)ctrlDetail.ID, StringArgName, StringArgIP, (jint)ctrlDetail.Port);
}

if(jobjDet != NULL && midDispStruct != NULL)
(*env)->CallStaticIntMethod(env, clsH,midDispStruct,jobjDet); //Calling the method and passing ControlDetail Object as parameter
}
//Calling a function from java and passing Structure array to it.
printf("\n\nGoing to call DisplayStructArray From C\n\n");
if (midDispStructArr!=NULL)
{
if(clsW != NULL && midWoConst != NULL)
{
//Creating the Object Array that will contain 2 structures.
jobjWOArr = (jobjectArray)(*env)->NewObjectArray(env, 2,clsW,
(*env)->NewObject(env, clsW, midWoConst,
(*env)->NewStringUTF(env, ""),
(*env)->NewStringUTF(env, ""),
(*env)->NewStringUTF(env, ""),
(*env)->NewStringUTF(env, ""),
(*env)->NewStringUTF(env, ""),
(*env)->NewStringUTF(env, ""),
(*env)->NewStringUTF(env, "")));
//Initializing the Array
int i = 0;
for(i=0;i<2;i++)
{
(*env)->SetObjectArrayElement(env, jobjWOArr,i,
(*env)->NewObject(env, clsW, midWoConst,
(*env)->NewStringUTF(env, WO[i].sumSerialId),
(*env)->NewStringUTF(env, WO[i].accessNumber),
(*env)->NewStringUTF(env, WO[i].actionType),
(*env)->NewStringUTF(env, WO[i].effectiveDate),
(*env)->NewStringUTF(env, WO[i].fetchFlag),
(*env)->NewStringUTF(env, WO[i].reason),
(*env)->NewStringUTF(env, WO[i].accessSource)));
}
}
//Calling the Static method and passing the Structure array to it.
if(jobjWOArr != NULL && midDispStructArr != NULL)
(*env)->CallStaticVoidMethod(env, clsW,midDispStructArr,jobjWOArr);
}
//Calling a Static function that return an Object
if (midRetObjFunc != NULL)
{
//Calling the function and storing the return object into jobject type variable
//Returned object is basically a structure having two fields (string and integer)
jobjRetData = (jobject)(*env)->CallStaticObjectMethod(env, clsH,midRetObjFunc,NULL);
//Get the class of object
clsR = (*env)->GetObjectClass(env, jobjRetData);
//Obtaining the Fields data from the returned object
jint nRet = (*env)->GetIntField(env, jobjRetData,(*env)->GetFieldID(env, clsR,"returnValue","I"));
jstring jstrLog = (jstring)(*env)->GetObjectField(env, jobjRetData,(*env)->GetFieldID(env, clsR,"Log","Ljava/lang/String;"));
const char *pLog = (*env)->GetStringUTFChars(env, jstrLog,0);

printf("\n\nValues Returned from Object are:\nreturnValue=%d\nLog=%s",nRet,pLog);
//After using the String type data release it.
(*env)->ReleaseStringUTFChars(env, jstrLog,pLog);
}
//Release resources.
int n = (*jvm)->DestroyJavaVM(jvm);
return 0;
}



Thanking you
Mukesh Kumar
GeneralRE: How to Call Java functions from C Using JNI Pin
chinmaya.mishra961-Aug-09 6:46
chinmaya.mishra961-Aug-09 6:46 
GeneralRe: RE: How to Call Java functions from C Using JNI Pin
Mohammed Anees13-Aug-09 23:20
Mohammed Anees13-Aug-09 23:20 
GeneralVS 2008 Pin
thusithagh5-Jul-09 8:47
thusithagh5-Jul-09 8:47 
GeneralMy vote of 2 Pin
thusithagh5-Jul-09 6:54
thusithagh5-Jul-09 6:54 
Questionjvm.dll not found Pin
gdaley9-Jun-09 17:34
gdaley9-Jun-09 17:34 
AnswerRe: jvm.dll not found Pin
gdaley9-Jun-09 18:50
gdaley9-Jun-09 18:50 
GeneralJNI_CreateJavaVM returning -3 Pin
rostronix14-May-09 15:34
rostronix14-May-09 15:34 
GeneralUnknown Error Pin
Ankush M Gupta18-Oct-08 4:47
Ankush M Gupta18-Oct-08 4:47 
QuestionRe: Unknown Error Pin
jakko00724-Feb-09 0:36
jakko00724-Feb-09 0:36 
AnswerRe: Unknown Error Pin
Damos25-Mar-09 7:44
Damos25-Mar-09 7:44 
GeneralRe: Unknown Error Pin
Jagdish Vasani12-Jun-09 4:16
Jagdish Vasani12-Jun-09 4:16 
Generalundeclared identifier.. Pin
bujal1-Jul-08 20:50
bujal1-Jul-08 20:50 
GeneralRe: undeclared identifier.. Pin
ajalilqarshi30-Jul-08 0:22
ajalilqarshi30-Jul-08 0:22 
Questionhow can i do webservice Pin
norazanita30-Jun-08 17:45
norazanita30-Jun-08 17:45 
QuestionHow can i create jvm in dll? Pin
Member 108979216-Jun-08 23:08
Member 108979216-Jun-08 23:08 
AnswerRe: How can i create jvm in dll? Pin
ajalilqarshi30-Jul-08 0:32
ajalilqarshi30-Jul-08 0:32 
AnswerRe: How can i create jvm in dll? Pin
jakko00724-Feb-09 0:46
jakko00724-Feb-09 0:46 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.