Unity与Android交互(双端通信)

news/2024/12/29 1:51:00/

前言

最近小编开始做关于手部康复的项目,需要Android集成Unity,以Android为主,Unity为辅的开发;上一篇给大家分享了Unity嵌入Android的操作过程,所以今天想给大家分享一下双端通信的知识;

一. Android与Unity哪个为主?

一般情况下,根据需求来决定Android与Unity的轻重,可以总结为以下两种

1.  将Unity作为Android程序中的一部分进行开发 ,将Unity3D场景当成一个界面或者说是界面的一部分

2.  将Android作为Unity程序中的一部分进行开发,将Android当作一个插件

小编这里由于项目需要,我是以Android为主,Unity辅助

二. Unity端调用Android端方法

实例化AndroidJavaClass对象,用于拿到Android的UnityClass和Activity要用到的接口

实例化AndroidJavaObject对象,是Unity发送消息给Android的关键

//获取Android中com.unity3d.player.UnityPlayer,这个类在Unity3d导出工程的unity-class.jar中
//这是通过反射的机制
AndroidJavaClass androidJavaClass = new AndroidJavaClass("com.unity3d.player.UnityPlayer");//获取Android当前正在运行的Activity
AndroidJavaObject androidJavaObject = androidJavaClass.GetStatic<AndroidJavaObject>("currentActivity");//调用Activity中的无返回值普通方法
androidJavaObject.Call("Android方法名");//调用Activity中的无返回值普通带参方法
androidJavaObject.Call("Android方法名","参数");//调用Activity中返回值String类的普通方法
androidJavaObject.Call<string>("Android方法名");//调用Activity中返回值String类的普通带参方法
androidJavaObject.Call<string>("Android方法名","参数");//调用Activity中返回值int类的普通方法
androidJavaObject.Call<int>("Android方法名");//调用Activity中返回值int类的普通带参方法
androidJavaObject.Call<int>("Android方法名","参数");//调用Activity中的无返回值静态方法
androidJavaObject.GetStatic("Android方法名");//调用Activity中的无返回值静态带参方法
androidJavaObject.GetStatic("Android方法名","参数");//调用Activity中返回值String类的静态方法
androidJavaObject.GetStatic<string>("Android方法名");//调用Activity中返回值String类的静态带参方法
androidJavaObject.GetStatic<string>("Android方法名","参数");//调用Activity中返回值int类的静态方法
androidJavaObject.GetStatic<int>("Android方法名");//调用Activity中返回值int类的静态带参方法
androidJavaObject.GetStatic<int>("Android方法名","参数");

这里列举了一些Unity调用Android的一些初级方法,记住,这种方式的调用最多只能传递一个参数,下一篇将为大家分享高级用法

三. Android端调用Unity端方法 

UnityPlayer.UnitySendMessage("Unity场景物体名","Unity脚本函数名","Unity脚本函数所需参数名");

UnityPlayer为Unity导出包,可以直接引用
UnitySendMessage()方法必须要传递三个参数
第一个参数为Unity场景中脚本挂载的游戏物体名

第二个参数为Unity脚本中的函数名
第三个参数为Unity脚本中函数所需要的参数,可以为空,但必须要有

这里只列举了双端通信的函数调用,可以满足Android与Unity之间参数不多的情况,如果需要大量参数传递的话要采用实现接口的方式在进行,双端通信接口之间的实现下一篇给大家分享

四. 双端通信实战演练

我给大家演示一个Android端控制Unity场景物体移动,并且Unity反过来控制Android端的一个Demo,希望大家能快速喜欢上Unity与Android的乐趣

1. Unity端

我在Unity的场景中创建一个Cube物体,在物体上挂载Move脚本,然后在场景中新建Canvas,Canvas下四个Button按钮,如下图所示

 Move脚本:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;public class Move : MonoBehaviour
{public Button btn_up;public Button btn_down;public Button btn_left;public Button btn_right;private AndroidJavaClass androidJavaClass;private AndroidJavaObject androidJavaObject;private void Start(){androidJavaClass = new AndroidJavaClass("com.unity3d.player.UnityPlayer");androidJavaObject = androidJavaClass.GetStatic<AndroidJavaObject>("currentActivity");btn_up.onClick.AddListener(AndroidUp);btn_down.onClick.AddListener(AndroidDown);btn_left.onClick.AddListener(AndroidLeft);btn_right.onClick.AddListener(AndroidRight);}public void Up(){transform.Translate(0, 0.5f, 0);Debug.Log("Android端调用了Unity场景Cube物体上的Move脚本中的Up方法");}public void Down(){transform.Translate(0, -0.5f, 0);Debug.Log("Android端调用了Unity场景Cube物体上的Move脚本中的Down方法");}public void Left(){transform.Translate(-0.5f, 0, 0);Debug.Log("Android端调用了Unity场景Cube物体上的Move脚本中的Left方法");}public void Right(){transform.Translate(0.5f, 0 , 0);Debug.Log("Android端调用了Unity场景Cube物体上的Move脚本中的Right方法");}public void AndroidUp(){androidJavaObject.Call("Up");Debug.Log("Unity端调用了Android端的Up方法");}public void AndroidDown(){androidJavaObject.Call("Down");Debug.Log("Unity端调用了Android端的Down方法");}public void AndroidLeft(){androidJavaObject.Call("Left");Debug.Log("Unity端调用了Android端的Left方法");}public void AndroidRight(){androidJavaObject.Call("Right");Debug.Log("Unity端调用了Android端的Right方法");}}

2. Android端

AndroidStudio新建空项目,按照我上一篇集成Unity的文章来操作,然后新建一个类,让这个类继承UntyPlayerActivity,具体步骤如下

MainActivity.java:

package com.example.test;import androidx.appcompat.app.AppCompatActivity;import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;public class MainActivity extends AppCompatActivity {private Button btn;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);btn = (Button) findViewById(R.id.btn);btn.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {Intent intent = new Intent(MainActivity.this, AndroidToUnityActivity.class);startActivity(intent);}});}
}

activity_main.xml:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"tools:context=".MainActivity"><Buttonandroid:id="@+id/btn"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="进入Unity"android:textAllCaps="false"app:layout_constraintBottom_toBottomOf="parent"app:layout_constraintEnd_toEndOf="parent"app:layout_constraintStart_toStartOf="parent"app:layout_constraintTop_toTopOf="parent" /></androidx.constraintlayout.widget.ConstraintLayout>

AndroidToUnityActivity.java:

package com.example.test;import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.RelativeLayout;import com.unity3d.player.UnityPlayer;public class AndroidToUnityActivity extends UnityPlayerActivity{/*** 声明变量*/private RelativeLayout Unity_View;private Button up_btn,left_btn,right_btn,down_btn;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_androidtounity);/*** 为变量绑定控件*/Unity_View = (RelativeLayout) findViewById(R.id.UnityView);up_btn = (Button) findViewById(R.id.btn1);left_btn = (Button) findViewById(R.id.btn2);right_btn = (Button) findViewById(R.id.btn3);down_btn = (Button) findViewById(R.id.btn4);/*** 将Unity视图添加到RelativeLayout中*/Unity_View.addView(mUnityPlayer);up_btn.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {UnityPlayer.UnitySendMessage("Cube","Up","");}});left_btn.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {UnityPlayer.UnitySendMessage("Cube","Left","");}});right_btn.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {UnityPlayer.UnitySendMessage("Cube","Right","");}});down_btn.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {UnityPlayer.UnitySendMessage("Cube","Down","");}});}private void Up(){UnityPlayer.UnitySendMessage("Cube","Up","");}private void Down(){UnityPlayer.UnitySendMessage("Cube","Down","");}private void Left(){UnityPlayer.UnitySendMessage("Cube","Left","");}private void Right(){UnityPlayer.UnitySendMessage("Cube","Right","");}
}

activity_androidtounity.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><RelativeLayoutandroid:id="@+id/UnityView"android:layout_width="match_parent"android:layout_height="600dp"></RelativeLayout><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:gravity="center_horizontal"><Buttonandroid:id="@+id/btn1"android:text="上移"android:layout_width="wrap_content"android:layout_height="wrap_content"/></LinearLayout><LinearLayoutandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_gravity="center_horizontal"android:orientation="horizontal"><Buttonandroid:id="@+id/btn2"android:text="左移"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginHorizontal="20dp"/><Buttonandroid:id="@+id/btn3"android:text="右移"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_marginHorizontal="20dp"/></LinearLayout><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:gravity="center_horizontal"><Buttonandroid:id="@+id/btn4"android:text="上移"android:layout_width="wrap_content"android:layout_height="wrap_content"/></LinearLayout></LinearLayout>

到此为止,以上就是我的全部源码,大家可以参考一下,要是有问题,我们可以评论区讨论一下

3. 演示效果

Android与Unity双端通信

如果有致力于Android+Unity的小伙伴私信联系我,我们一起交流平时项目中的疑问,一起拓展这片领域


http://www.ppmy.cn/news/969525.html

相关文章

ChatGPT微调分类示例

我们将微调 ada 分类器以区分两种运动&#xff1a;棒球和曲棍球。 from sklearn.datasets import fetch_20newsgroups import pandas as pd import openaicategories [rec.sport.baseball, rec.sport.hockey] sports_dataset fetch_20newsgroups(subsettrain, shuffleTrue, …

【极简,亲测,解决】Too many levels of symbolic links

前言&#xff08;与内容无关&#xff09; 帖子看多了&#xff0c;让我产生一种错觉&#xff0c;就是生产这些帖子的人都是机器人吗&#xff1f;是活着的吗&#xff1f;乱七八糟的转载和明显错误的结论太多了。 原因 原因是 链接的层数过多&#xff0c;已经产生了回路。 大概…

Foxbase和Visual FoxPro已退出舞台,新数据库管理系统崭露头角

你知道Foxbase吗&#xff1f; 你知道Foxbase吗&#xff1f; 一不小心暴露年纪了吧&#xff01; Foxbase作为上世纪80年代初面世的数据库管理系统&#xff0c;是由Fox Software公司于1984年推出。面世之后一直深受计算机专业人士的喜爱。因为在那个时候使用计算机人群并不是很…

让ChatGPT无路可走

&#x1f482; 个人主页: 沐尘而生&#x1f3c6;今日踽踽独行&#xff0c;他日化蝶而去&#x1f4ac;不要忘了那个奔跑的自己&#xff0c;我们一起加油&#xff01; 最近 ChatGPT 火&#x1f525;翻了&#xff0c;在申请通过后&#xff0c;给大家展现一下我看到的效果&#xff…

1.3 - 操作系统 - firewalld防火墙iptables防火墙

「作者简介」:CSDN top100、阿里云博客专家、华为云享专家、网络安全领域优质创作者 「订阅专栏」:此文章已录入专栏《网络安全入门到精通》 Linux防火墙 Frewalld1、常用操作2、开放/关闭服务3、开放/关闭端口4、IP访问端口规则5、安全域Iptables1、常用操作2、四表五链cent…

再见操作系统!ChatGPT和Windows直接在一起了

来源&#xff1a;水木学堂 二十世纪初&#xff0c;微软曾发起过一场“浏览器战争”&#xff0c;用 IE 4.0 成功打赢了浏览器市场份额超过 90% 以上的网景浏览器。当时&#xff0c;微软用的就是“将 IE 放在 Windows 98 ”这样的“禁忌武器”&#xff0c;还因此遭到了日后的反垄…

UNIX操作系统中,文件的索引结构放在( )

UNIX操作系统中&#xff0c;文件的索引结构放在&#xff08; &#xff09; A. 超级块 B. 索引结点 C. 目录项 D. 空闲块 UNIX采用树形目录结构&#xff0c;文件信息存放在索引结点中。 超级块是用来面熟文件系统的。

前端面试必备技能(持续更新)

js 1.let 、const 、var 的区别 答:var会挂载到window&#xff0c;会存在全局变量污染&#xff0c;会存在变量提升&#xff0c;而let、const不会 let、const同一作用域内不可声明同变量、var可以 (特别说明&#xff0c;let其实也会变量提升&#xff0c;但是提升之后会存在一…