UnityMirror学习笔记(3):服务器与客户端间的函数调用,互发控制指令:Command,ClientRpc,TargetRpc

这篇具有很好参考价值的文章主要介绍了UnityMirror学习笔记(3):服务器与客户端间的函数调用,互发控制指令:Command,ClientRpc,TargetRpc。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

Mirror是一个简单高效的开源的Unity多人游戏网络框架。
官方文档链接:
https://mirror-networking.gitbook.io/docs

从客户端发送指令由服务器调用

API

Mirror提供了一个特性Command来进行从客户端到服务器的远程控制指令,
它的核心逻辑就是,当客户端的脚本调用此特性修饰的函数时,
服务端对应的对象执行此函数。
这意味着函数中的代码不会在客户端执行。

基本的用法是在需要进行远程调用的函数前编辑特性标签:

[Command]
void cmdRemoteFunction(){}

此方法是可以传参的,只要参数支持(可自动序列化与反序列化)。当然自定义的数据类型也可以通过自定义序列化与反序列化方式作为参数传递,这以后再学

使用

接下来用一个例子展示此API:

修改上一节写的Cube的控制脚本,使得每按一次空格,就会将自己的颜色改变:
这里为了更好地看到自己的Cube,所有用了Cinemachine的一些控制,可以随便删除。

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Mirror;
using Cinemachine;
public class PlayerController : NetworkBehaviour
{
    [SyncVar(hook = nameof(ColorChanged))]
    Color PlayerColor = Color.white;

    [Command]
    void CmdChangeColor(Color color){
        PlayerColor = color;
    }

    Rigidbody rb;
    MaterialPropertyBlock prop;
    CinemachineVirtualCamera cv;

    void ColorChanged(Color oldColor, Color newColor){
        Debug.Log("Color Changed");
        prop.SetColor("_Color" ,newColor);
        GetComponent<Renderer>().SetPropertyBlock(prop);
    }

    private void Start() {
        rb = GetComponent<Rigidbody>();
        prop = new MaterialPropertyBlock();
        GetComponent<Renderer>().GetPropertyBlock(prop);
        if(isLocalPlayer){
            cv = GameObject.FindGameObjectWithTag("VCAM").GetComponent<CinemachineVirtualCamera>();
            cv.Follow = this.transform;
            cv.LookAt = this.transform;
        }
    }

    private void Update() {
        if(!isLocalPlayer) return;
        if(Input.GetKeyDown(KeyCode.Space)){
            CmdChangeColor(PlayerColor == Color.white?Color.black:Color.white);
        }
    }
}
测试

如图,一个做主机一个做客户端,
当在其中一个客户端按下空格键,大家都能看到对应的角色颜色变了。
UnityMirror学习笔记(3):服务器与客户端间的函数调用,互发控制指令:Command,ClientRpc,TargetRpc

注意点

这里隐含一个条件,那就是只有属于自己的角色,向服务器传送执行方法指令,服务器才会执行对应的角色脚本。
否则上例中,按下空格,两个方块的颜色都会改变。
我们可以通过修改CommandrequiresAuthority参数,实现这个“否则”:
[Command(requiresAuthority = false)]

从服务器发送指令由客户端调用

API

Mirror提供了一个特性ClientRpc来进行从服务器向客户端的远程控制指令,
它的核心逻辑就是,当服务器的脚本调用此特性修饰的函数时,
客户端对应的对象执行此函数。
这意味着函数中的代码不会在服务器执行。

基本的用法是在需要在客户端执行的方法前编辑特性标签:

[ClientRpc]
void RpcRemoteFunction(){}

从服务器发送指令由指定的客户端调用

上面的Rpc会发送给所有客户端的对应对象执行,
有的时候需要发送给指定的客户端的对应对象执行。

一个简单的例子,
玩家A打了玩家B一下,玩家B的模型上,
跳出了自己掉的血的数量的文字UI,
但是我们不希望玩家A看到这个文字UI的显示,
那么就只可以通过只发送给玩家B的客户端上的B实体,
显示扣血文字的指令实现。
API
[TargetRpc]
void TargetRemoteDamaged(NetworkConnection target, int damage){}
//第一个参数可忽略,默认发给自己,也就是调用此方法的客户端实体

在上面那个例子中,当A打到B时,
显然可以获取到挂载在B对象上的NetworkIdentity
可以通过此获取到B客户端与服务器的连接的通道NetworkIdentity.connectionToClient
将其作为参数TargetRpc的第一个参数由服务器执行,即可只发送给B客户端。

当然不携带此参数,就默认发给自己。

实例

上面那个例子我也是看官方文档才理解这种做法的必要性的,
所有实例也就拿官方文档解释一下八:

public class Player : NetworkBehaviour
{
    public int health;

    [Command]
    void CmdMagic(GameObject target, int damage)
    {
        target.GetComponent<Player>().health -= damage;

        NetworkIdentity opponentIdentity = target.GetComponent<NetworkIdentity>();
        TargetDoMagic(opponentIdentity.connectionToClient, damage);
    }

    [TargetRpc]
    public void TargetDoMagic(NetworkConnection target, int damage)
    {
        // This will appear on the opponent's client, not the attacking player's
        Debug.Log($"Magic Damage = {damage}");
    }

    // Heal thyself
    [Command]
    public void CmdHealMe()
    {
        health += 10;
        TargetHealed(10);
    }

    [TargetRpc]
    public void TargetHealed(int amount)
    {
        // No NetworkConnection parameter, so it goes to owner
        Debug.Log($"Health increased by {amount}");
    }
}

这里就是如果伤害到对方,就让对方的客户端知道自己受伤了掉了一些血,而其它客户端不知道
如果自己回血,那就只有自己知道自己回血了,比如看到绿色粒子动画效果,别人看不到。

最后的注意

[Command]函数由客户端对象调用,内容由服务器对象执行,
[ClientRpc][TargetRpc]函数都是由服务器对象调用,内容被客户端对象执行。
这里一定不能搞混,保持清醒!文章来源地址https://www.toymoban.com/news/detail-465381.html

到了这里,关于UnityMirror学习笔记(3):服务器与客户端间的函数调用,互发控制指令:Command,ClientRpc,TargetRpc的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 《TCP/IP网络编程》阅读笔记--基于TCP的服务器端/客户端

    目录 1--TCP/IP协议栈 2--TCP服务器端默认函数调用顺序 3--TCP客户端的默认函数调用顺序 4--Linux实现迭代回声服务器端/客户端 5--Windows实现迭代回声服务器端/客户端 6--TCP原理 7--Windows实现计算器服务器端/客户端         TCP/IP协议栈共分 4 层,可以理解为数据收发分成了 4 个层

    2024年02月10日
    浏览(43)
  • C语言再学习 -- C语言搭建TCP服务器/客户端

    TCP/UDP讲过~ 参看:UNIX再学习 – TCP/UDP 客户机/服务器 参看:UNIX再学习 – 网络IPC:套接字 这里记录一下可用的TCP服务器和客户端代码。 参看:用C语言搭建TCP服务器/客户端

    2024年01月20日
    浏览(42)
  • 服务器端使用django websocket,客户端使用uniapp 请问服务端和客户端群组互发消息的代码怎么写的参考笔记

    2023/8/29 19:21:11 服务器端使用django websocket,客户端使用uniapp 请问服务端和客户端群组互发消息的代码怎么写 2023/8/29 19:22:25 在服务器端使用Django WebSocket和客户端使用Uniapp的情况下,以下是代码示例来实现服务器端和客户端之间的群组互发消息。 服务器端代码 (使用Django Chann

    2024年02月11日
    浏览(36)
  • unity学习(19)——客户端与服务器合力完成注册功能(1)入门准备

    逆向服务器用了三天的时间,但此时觉得一切都值,又可以继续学习了。 服务器中登录请求和注册请求由command变量进行区分,上一层的type变量都是login。 从注册入手!视频对应的应该是第七讲。 点击注册按钮,输入账号密码,实测可以在服务器收到编码后的字符串,但此时

    2024年02月21日
    浏览(82)
  • netty学习(3):SpringBoot整合netty实现多个客户端与服务器通信

    创建一个SpringBoot工程,然后创建三个子模块 整体工程目录:一个server服务(netty服务器),两个client服务(netty客户端) pom文件引入netty依赖,springboot依赖 NettySpringBootApplication NettyServiceHandler SocketInitializer NettyServer NettyStartListener application.yml Client1 NettyClientHandler SocketInitializ

    2024年02月11日
    浏览(46)
  • unity学习(22)——客户端与服务器合力完成注册功能(4)数据库化

    单纯的账号密码这种非频繁读写,实现起来很简单的,游戏的属性信息到时候也许会比较麻烦。 思路:每次加入有新键值TryAdd,如果加入成功,直接重写账号密码文件即可。 C#JsonConvert.DeserializeObject反序列化与JsonConvert.SerializeObject序列化(一)-CSDN博客 在vs中-项目-NuGet,搜索

    2024年02月19日
    浏览(34)
  • 《TCP/IP网络编程》阅读笔记--基于Windows实现Hello Word服务器端和客户端

    目录 1--Hello Word服务器端 2--客户端 3--编译运行 3-1--编译服务器端 3-2--编译客户端 3-3--运行 运行结果:

    2024年02月10日
    浏览(44)
  • unity的学习,准备搞一款mmo小游戏,服务器和客户端从零学

    先学一下unity,mmo服务器框架到时候在学习一下,暂时服务器简单做一下 如代码所示,简单了解一下。 我个人感觉不要一个放在Awake函数中,一个放在Start中。因为这只适合两个脚本使用,如果多个脚本还是没有办法解决脚本执行的顺序。在这里设置脚本的执行顺序,添加进

    2023年04月21日
    浏览(38)
  • SOAP学习之一:Visual C++创建简单的客户端--使用soap toolkit 3.0获取UTC服务器时间

    初始接触 XML及SOAP第一天,摸不着头绪,看了很多文章,总结一下几点心得,附一个小例子使用VC++控制台程序获取UTC服务器时间。 看到的资料VC++都是使用soap toolkit来使用SOAP,soap toolkit目前看最高版本应该是3.0,但是微软已经不再对soap toolkit提供技术支持及更新,在微软官网

    2024年02月09日
    浏览(54)
  • 服务器异步客户端

    internal class MessageManagerT:SingletonMessageManagerT {     Dictionaryint, ActionT MsgDic = new Dictionaryint, ActionT();     public void OnAddListen(int id,ActionT action)     {         if(MsgDic.ContainsKey(id))         {             MsgDic[id] += action;         }         else         {             MsgDic.Add(id, ac

    2024年04月09日
    浏览(42)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包