質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.50%
JNI

JNI(Java Native Interface)は、Javaプラットフォームにおいて、Javaで記述されたプログラムと、他の言語で書かれたネイティブコードを連携するためのインタフェース仕様である

Android

Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

COCOS2D-X

COCOS2D-Xは、 2Dゲームを手軽に開発できるフレームワークのことです。 iPhone(iOS)向け、Android等に対応しており、 実質ワンソースで開発が可能です。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Android Studio

Android Studioは、 Google社によって開発された、 Androidのネイティブアプリケーション開発に特化した統合開発ツールです。

Q&A

解決済

1回答

3034閲覧

jniを使ってandroidからcocosへcallbackする処理を記述してるのですがエラーが出ます。

kaji

総合スコア648

JNI

JNI(Java Native Interface)は、Javaプラットフォームにおいて、Javaで記述されたプログラムと、他の言語で書かれたネイティブコードを連携するためのインタフェース仕様である

Android

Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

COCOS2D-X

COCOS2D-Xは、 2Dゲームを手軽に開発できるフレームワークのことです。 iPhone(iOS)向け、Android等に対応しており、 実質ワンソースで開発が可能です。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Android Studio

Android Studioは、 Google社によって開発された、 Androidのネイティブアプリケーション開発に特化した統合開発ツールです。

0グッド

0クリップ

投稿2017/04/06 09:27

編集2017/04/10 05:25

jniを使ってandroidからcocosへcallbackする処理を記述してるのですがエラーが出ます。

cpp

1JNIEXPORT void JNICALL Java_jp_co_sample_SampleActivityBridge_onSampleLoaded(JNIEnv *env, jclass clazz, jobjectArray jsampleInits) 2 { 3jclass jsampleInit = (jclass)env->GetObjectArrayElement(jsampleInits, 0); 4jfieldID var = env->GetFieldID(jsampleInit, "name", "Ljava/lang/String;"); 5jstring jstr = (jstring)env->GeCCtObjectField(jsampleInit, var);

jsampleInitはLjp/co/sample/sampleInit/SampleInit;の型になります。

エラー

JNI WARNING: jclass arg has wrong type (expected Ljava/lang/Class;, got Ljp/co/sample/sampleInit/SampleInit;) (GetFieldID) 04-06 18:14:52.198 26400-26400/jp.co.sample W/dalvikvm: in Ljp/co/sample/SampleActivityBridge;.onSamplesLoaded:([Ljp/co/sample/SampleInit;)V (GetFieldID)

一行目のjsampleInitの型がjclassだとエラーが出るっぽいのですが、

以下のようなコードもネット検索すると見つかり、jclassで問題ないかと思ったのですが、以下コードは問題なくじっこうできます、上記のように書くとエラーになるようです。

jclass cls = env->FindClass("jp/co/sample/SampleInit");

ここのJNI Types一覧から妥当な型はjclassのみでした。
http://stnguyen.com/cocos2d-x/call-cpp-functions-from-java.html

jclassの部分を別の型にすればいいと思うのですが、カスタムのクラスの型がかければいいのですが、どう書くのでしょうか?
知ってる方教えてほしいです

==追記A 2017/4/7 16:06 start==
Cocos側の入れ物として以下を定義しました。
Classes/CCSampleInit.h

cpp

1class CCSampleInit 2{ 3public: 4 const char* name = ""; 5 int valule = 0; 6};

Classes/CCSampleInit.cpp

USING_NS_CC; CCSampleInit* CCSampleInit::create() { return new CCSampleInit(); }

==追記A end==

==追記B 2017/4/7 16:12 start==

public class SampleActivityBridge { private static Activity mActivity; public static native void onSampleInitsLoaded(SampleInit[] sampleInits); @Override public void onSampleInitsLoaded(SampleInit[] sampleInits) { SampleInitActivityBridge.onSampleInitsLoaded(sampleInits); }

==追記B end==

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

jclass jsampleInit = (jclass)env->GetObjectArrayElement(jsampleInit, 0);

この行はタイプミスではないでしょうか。このままではC++でコンパイルエラーになる気がします。

推測ですがおそらくこういうことをしたいのではないでしょうか?

java

1class SampleInit { 2 String name; 3 ... 4} 5... 6 7//このメソッドをJNI側に実装 8void callCocos2D(SampleInit[] elementArray) { 9 SampleInit element = elementArray[0]; // (1) 10 String str = element.name; // (2) 11 ... 12}

C++

1jobjectArray jelementArray = ... // native methodの引数などからとってくる 2 3//(1)の処理 4jobject jelement = env->GetObjectArrayElement(jelementArray, 0); 5//(2)の処理 6jclass jsampleInit = env->FindClass("jp/co/sample/SampleInit"); 7jfieldID var = env->GetFieldID(jsampleInit, "name", "Ljava/lang/String;"); 8jstring jstr = (jstring)env->GetObjectField(jelement, var);

こうした質問の場合、以下も併せて明記すると質問者さんが何をしたいかがよりはっきりわかると思います。

  • nativeメソッドのjava側での宣言
  • nativeメソッドをjava側から呼び出すコード周辺
  • nativeメソッドのC++側のプロトタイプ

jclassの部分を別の型にすればいいと思うのですが、カスタムのクラスの型がかければいいのですが、どう書くのでしょうか?

JNIではjava.lang.String以外の全ての参照型のインスタンスは型jobjectとしてのみ使えます。jclassは特定のクラスとしてフィールドやメソッドにアクセスするために使いますが、その求め方はenv->FindClass("jp/co/sample/SampleInit")のようにすべきと思います。


追記1:ここから
JNIの仕様を眺めていてもう一つ重要なjclassの求め方があることに気づきました。Java側でnativeメソッドがstaticと宣言されている場合、C++で定義された関数の第二引数にクラスオブジェクトが渡されます。ただし、そこに渡されるのはnativeメソッドを定義したクラスであって、引数に渡されるクラスと同じではありませんのでその点注意してください。
追記1:ここまで


Class<?>のインスタンスをJNIへ渡せばjobject => jclassに変換してもよいのでしょうか・・・残念ながらそれができるかどうか自分には分かりません><

投稿2017/04/06 10:48

編集2017/04/07 07:47
KSwordOfHaste

総合スコア18392

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

kaji

2017/04/07 03:40 編集

>この行はタイプミスではないでしょうか。 業務内容がもれないように変数名を変換してたのですが、置換処理をミスしたようです。修正致しました。 https://teratail.com/questions/71545 こちらで全体を質問していて解答がなかったので、部分的に新たに質問したのですが、KSwordOfHasteさんから頂いたコードを試してだめであれば新たに追加したいと思います。
KSwordOfHaste

2017/04/07 03:53

> こちらで全体を質問していて そういうことでしたか。こういう場合、元の質問へのリンクを本質問に書いておくと回答者が気づきやすいと思います。元の質問の方が情報量が多いので回答できるかどうか見てみます。
kaji

2017/04/07 06:54 編集

前投稿の解答もありがとうございます。一旦こちらを返信します。 //(1)の処理 と //(2)の処理 はつながってますか?どちらかの処理という意味でしょうか? どちらかの処理という意味であれば、 1の処理は動作しますが、そこからnameプロパティにアクセスするにはどうすればよいのでしょうか? ためしに以下のようにしましたが・・。 ``` jobject jsampleInit = env->GetObjectArrayElement(jsampleInits, 0); jfieldID var = env->GetFieldID(jsampleInit, "name", "Ljava/lang/String;"); jstring jstr = (jstring)env->GetObjectField(jsampleInit, var); const char* utfstr = env->GetStringUTFChars(jstr, 0); ``` jclassをjobjectに変えたところ以下エラーが出ました。 ``` error: invalid conversion from 'jobject {aka _jobject*}' to 'jclass {aka _jclass*}' [-fpermissive] jfieldID var = env->GetFieldID(jsampleInit, "zoneID", "Ljava/lang/String;"); ``` 2の処理はSampleInitの空データができますよね?androidから渡ってきたデータ配列のSampleInit->nameを取得したいです。
KSwordOfHaste

2017/04/07 06:42 編集

(1),(2)はJava/C++の対応のために書きました。同じ番号どうしが同じ処理を意味しています。 ゆえに(1)と(2)は連続して実行すると考えてください。 jclassの扱いは自分が気づいてない点があったので回答へ追記します。
kaji

2017/04/07 07:00

jclassの扱い確認しました。java側でnativeを定義したクラスの名前はjni側では不要なので第2引数のjclassは使っておりません。
kaji

2017/04/07 07:14 編集

//このメソッドをJNI側に実装 void callCocos2D(SampleInit[] elementArray) { SampleInit element = elementArray[0]; // (1) String str = element.name; // (2) ... } これは見逃しておりました。 Android側Bridgeの意味でしょうか? 追記しました(追記B)
kaji

2017/04/07 07:16

1処理で求めたjelementが 2処理に見当たらないので、どう繋げばよいかわかりませんでした。 どうか、ご教授お願い致します。
KSwordOfHaste

2017/04/07 07:44

> jclassをjobjectに変えたところ以下エラーが出ました。 クラスとオブジェクトの扱いに混乱されているようですが、まずはJNI上でjclass型であるべき変数についてキャストしてはいけないと考えた方が多分間違いがないと思います。 JNIではフィールドやメソッドにアクセスするためにはJavaでのようにストレートには呼べません。必ずそのフィールドやメソッドが宣言されたクラスをenv->FindClass(クラス名)で求めてから、そのクラスに対して「このフィールドのID教えて」「このメソッドのID教えて」とやってから、やっとフィールドやメソッドにアクセスできると考えてください。
KSwordOfHaste

2017/04/07 07:46 編集

jelementは引数の配列の第一要素である「SampleInitのインスタンス」と考えてください。 --- しまった、自分の回答がおかしかったです。回答を修正します。
kaji

2017/04/07 09:34

なんと!いけました〜。2日ぐらい検索&試すを繰り返して、超ドハマリしてたので、助かりました。KSwordOfHasteさんのお陰で先へ進めそうです!ありがとうございました。
KSwordOfHaste

2017/04/07 09:37

おめでとうございます。 多分jclassやjobjectを使ったフィールド・メソッドアクセスが一つ成功すると後は楽ではないでしょうか~
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.50%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問