AndroidStudio3.4+Unity2018.3,导出JAR包给UNITY使用
环境 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", , );
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为空.
AndroidStudio3.4+Unity2018.3,导出JAR包给UNITY使用的更多相关文章
- Unity 3D与Android Studio安卓交互之-导出jar包
u3d与安卓 jar 包交互 本文提供全流程,中文翻译. Chinar 坚持将简单的生活方式,带给世人!(拥有更好的阅读体验 -- 高分辨率用户请根据需求调整网页缩放比例) Chinar -- 心分享 ...
- Android Studio安卓导出aar包与Unity 3D交互
Unity与安卓aar 包交互 本文提供全流程,中文翻译. Chinar 坚持将简单的生活方式,带给世人!(拥有更好的阅读体验 -- 高分辨率用户请根据需求调整网页缩放比例) Chinar -- 心分 ...
- Android Studio 导出jar包
不像在Eclipse,可以直接导出jar包.AndroidStudio只可以生成aar包. 在网上看到许多朋友问怎么可以像Eclipse一样导出jar包,其实我们只要知道它的原理就可以了. 用jar命 ...
- android studio 使用gradle 导出jar包,并打包assets目录
警告:本文年久失修. 随着android studio的升级 ,gradle的升级,严格按照本文的代码去做可能不会成功,希望依然可以作为解决问题的思路. 最近项目在做一个sdk,供别的开发者使用,所以 ...
- [转]AndroidStudio导出jar包
原文链接:http://blog.csdn.net/hjq842382134/article/details/38538097# 1. 不像在Eclipse,可以直接导出jar包.AndroidStu ...
- [Android Pro] AndroidStudio导出jar包
reference : http://blog.csdn.net/beijingshi1/article/details/38681281 不像在Eclipse,可以直接导出jar包.Android ...
- [转]--android studio 使用gradle 导出jar包,并打包assets目录
转自: http://www.cnblogs.com/wuya/p/android-studio-gradle-export-jar-assets.html 最近项目在做一个sdk,供别的开发者使 ...
- eclipse导出jar包
第一种:普通类导出jar包,我说的普通类就是指此类包含main方法,并且没有用到别的jar包. 1.在eclipse中选择你要导出的类或者package,右击,选择Export子选项: 2.在弹出的对 ...
- Spring Boot导出jar包发布
一:事由 现在的项目组开发项目使用的是Spring Boot的技术,开发的时候是直接通过一个入口主函数来启动项目的.如果将项目交给客户,怎样才能正确的发布运行呢?百度了一下有关的知识,大概了解到是通过 ...
随机推荐
- spring-data-elasticsearch使用出现的一些小问题
问题一failed to load elasticsearch nodes : org.elasticsearch.index.mapper.MapperParsingException: No ty ...
- 004/HyperLedger-Fabric(节点与channel关系)
HyperLedger:超级账本项目 一.Fabric中的节点 1.节点术语 [1].Orderers: 提供共识服务的网络节点,例如,使用Kafka或PBFT [2].Peers: 维护账本 ...
- 数组的includes方法
Array.prototype.includes方法返回一个布尔值,表示某个数组是否包含给定的值,与字符串的includes方法类似.该方法属于 ES7 ,但 Babel 转码器已经支持. [1, 2 ...
- 前端 CSS的继承性
css有两大特性:继承性和层叠性 继承性 继承:给父级设置一些属性,子级继承了父级的该属性,这就是我们的css中的继承. 记住:有一些属性是可以继承下来 : color . font-*. text- ...
- 数组去重,排序,重复次数,两个数组合并,两个数组去重,map(),filter(),reduce()
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- CSRF verification failed. Request aborted.错误解决办法
在Django项目的页面,提交form表单POST请求时,会出现报错:CSRF verification failed. Request aborted. 需要在form表单中加:{% csrf_to ...
- Ribbon远程调用
Ribbon是客户端的负载均衡机制,它有几种负载均衡机制.默认是轮询,我们也可以自定义规则.通过合理的分配网络请求来减小服务器的压力.项目都是注册到eureka服务器上.通过ribbon去调用其他服务 ...
- 正则表达式RegExp对象
3.1 正则表达式对象的创建方式 字面量的方式 var patt = /匹配规则/修饰符; / --> 边界的意思 new关键字 var patt = new RegExp( ...
- 【知识强化】第七章 输入/输出系统 7.3 I/O接口
下面我们进入第七章的第三节,I/O接口. I/O接口呢就是解决了外设和主机之间的一个连接的问题.那么我们这一节就要来看一下I/O接口它有哪些功能,以及它是怎么组成的,还有就是我们主机如何来定位到那样一 ...
- vue的class和style的绑定
<div class="input-search" :class="{input-search-focus : iscur == 1}"> 在原本有 ...