环境 Android studio 3.4 + unity2018.3

1,android studio 新建空工程,一切默认,完成。
这个空工程只是个壳,它的所有参数都没什么用,它存在的意义是为了后面能创建 module。
因为JAVA不像C++,可以直接创建类库。
2,新建模块
file->new module->android library
application name 和 module name 都不重要,没有实际用处(这里取名mylibrary),重要的是包名: package name。
这里取名为 com.x.y
包名用到哪些地方?
(1,AndroidManifest.xml里的 package
(2,每个java文件的开始处,形如 package com.x.y,表示此java文件被放入了工程的com/x/y目录下
(3,unity 的build Settings里的 package name

这三处必须使用同一个包名,如果不一致则启动时闪退。如果想修改包名,就要修改这三处就行了。

3,在mylibrary模块下的src/java/com.x.y/添加java文件,随便取名,就叫MainActivity吧
修改其源码如下:

 



package com.x.y;

import android.content.Intent;
import com.unity3d.player.UnityPlayerActivity;
import android.app.Activity;
import android.os.Bundle;

public class MainActivity extends UnityPlayerActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //setContentView(R.layout.activity_main);
        activity = this;
    }

    //public  static  MainActivity activity;
    public static MainActivity activity;
    public int Calc(int x, int y){
        return  x + y;
    }

    public static float Calcf(float x, float y){
        return  x + y;
    }

    public void restartApplication() {
        new Thread(){
            public void run(){
                Intent launch=getBaseContext().getPackageManager().getLaunchIntentForPackage(getBaseContext().getPackageName());
                launch.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
                startActivity(launch);
                android.os.Process.killProcess(android.os.Process.myPid());
            }
        }.start();

        finish();
    }
}



 

这个类的目的:
(1,测试普通成员函数和静态函数的调用
(2,测试重启功能

4,修改mylibary模块的AndroidManifest.xml,这一步放在最后也行,此文件是拿给UNITY用的,内容如下:



<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.x.y">

<application android:allowBackup="true" android:label="@string/app_name">
<activity android:name=".MainActivity" android:label="@string/app_name">
<intent-filter>
  <action android:name="android.intent.action.MAIN"/>
  <category android:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
</application>

</manifest>



 

注意一定要有下面两行

<intent-filter>
 <action android:name="android.intent.action.MAIN"/>
 <category android:name="android.intent.category.LAUNCHER"/>
 </intent-filter>


学过安卓开发的都知道这是标记此活动是主活动,没有主活动的应用是不会被运行的。
默认情况下UNITY会创建自己AndroidManifest.xml,一旦我们在Assets/Plugins/Android/下放置了我们自定的AndroidManifest.xml,
UNITY就以我们的配置为主,并将buildsettings里的相关配置合并进来,如安装位置,持久化路径等。
因此,如果我们生成了自己的AndroidManifest.xml,却又不配置为主活动,安装后桌面没有图标,因为它不是一个可执行程序。

5,修改mylibrary模块的 build.gradle(注意不是修改工程的),为编译做准备,在末尾添加:



task exportJar(type: Copy) {
//必须先删除旧的包,否则新包不会生成,这个目录对应mylibrary/build/libs/AndroidUtils.jar
delete 'build/libs/AndroidUtils.jar'

//生成的目录在这里
from('build/intermediates/packaged-classes/release/')

//生成build/libs目录,并将build/intermediates/packaged-classes/release/下的classes.jar复制到此目录
into('build/libs/')

//指定我们只在build/intermediates/packaged-classes/release/下寻找classes.jar(放入到build/libs/目录
include('classes.jar')

//重命名下,起个个性的名字,不要与别人混淆
rename('classes.jar', 'AndroidUtils.jar')
}

//这里会触发真正对mylibaray的编译
exportJar.dependsOn(build)



 

然后点击工具栏上的 run anything:

,或双击ctrl,在弹出的输入窗口中输入 gradlew exprotJar,回车,等待编译完成。

如果没问题,会提示 BUILD SUCCESSFUL

如果出了错,仔细看前面的输出LOG。

这里的注意点:
(1,from('build/intermediates/packaged-classes/release/') 这里的路径在不同版本的 Android Studio中不同。
在低版本的AS中路径为 build/intermediates/bundles/release
如果编译中报错 exportJar: NO-SOURCE,就说明是此路径错误了,这时就去磁盘上的build/intermediates/目录搜索classes.jar文件,
找到它的路径,写入到from里就对了。

6,生成完成后,得到我们的AndroidUtils.jar包,新建UNITY工程,package name取名:com.x.y

在Assets下新建如下目录 Plugins/Android,然后将AndroidManifest.xml放入到此目录下,然后在创建 Plugins/Android/libs目录,
将AndroidUtils.jar放入此目录。
注意这里的目录结构要求:AndroidManifest.xml必须与libs同目录,AndroidUtils.jar必须在libs目录中,名字不能错。
但libs和XML不一定直接在Android下,可以是多层目录下,如:Android/abc/libs也可以

7,添加脚本,随便挂在什么上,如相机上,测试代码如下:


using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;

public class calljava : MonoBehaviour
{
    AndroidJavaObject jo;
    Text info;

    void Start()
    {
        //方式一,使用基类,这种比较通用
        var jc = new AndroidJavaClass("com.unity3d.player.UnityPlayer");
        jo = jc.GetStatic<AndroidJavaObject>("currentActivity");
        info = GameObject.Find("TextInfo").GetComponent<Text>();
        info.text += "start \n";

        //方式二,使用具体类,不通用,当类名改了就要改这里的字符串
        //var jc = new AndroidJavaClass("com.example.mylibraryx.MainActivity");
        //jo = jc.GetStatic<AndroidJavaObject>("activity");

        info.text += "get java object " + (jo == null ? "null\n" : "ok\n");

        //var btn = GetComponent<Button>();
        //btn.onClick.AddListener(() =>
        //{
            if (jo != null)
            {
                //注意1,参数类型必须传对,否则运行到这里就出错了
                var ret = jo.Call<int>("Calc", 10, 20);
                info.text += "call ret : " + ret + " \n";

                //调用静态函数
                var retf = jo.CallStatic<float>("Calcf", (float)10.2, (float)20.2);
                info.text += "call retf : " + retf + " \n";


                //注意2,由于Calcf是个静态函数,非静态调用会造成参数错位,结果为retfx = 10.2
                var retfx = jo.Call<float>("Calcf", (float)10.2, (float)20.2);
                info.text += "call retfx : " + retfx + " \n";

            }

        //});
    }

    // Update is called once per frame
    void Update()
    {
        
    }
}


 

8,打包APK,到安卓上运行。在UNITY上运行会出错,提示jo为空.