Unity3d多人网络

From:http://game.ceeger.com/forum/read.php?tid=428&page=3

 Unity3d多人网络(从圣典拷贝来的,先看看,因为我觉得这位仁兄比我翻译得好,我也对这个教程作一些修改

教程简介

 

我一直认为unity需要一个好一点的多人网络的教程。当我开始用unity网络功能的时候,我感觉unity自带的例子太混乱了;一个好的网络功能的例子应该包括源文件,这样你可以迅速找到你需要的资料。由于这个想法,我决定参加UniKnowledge比赛并且终于完成了一个网络功能的教程,我希望这个教程包括了你所需要的所有的内容。

这个教程介绍了很多案例;从最小的细节一直到真正的FPS游戏。我建议你从头到尾看一遍这个教程,不过如果你学东西很快的话,也可以自己看一下这些案例,如果需要更多细节,再回过头来看一下这个文档。

About the author
这个教程由M2H的MikeHergaarden(Leepo)所写。我们已经使用unity两年多了,不过我们真正用unity进行正规开发只有最近的几个月。我们在最开始就在关注多人游戏的功能。实际上我们的第一个游戏就是多人在线游戏;其实很简单!我们的多人游戏有:Crashdrive3D, Cratemania, Surrounded by Death, Verdun Online还有最近我们正在搞的Hyberon。

希望你能够享受这个教程。如果你你搞出什么名堂来,记得和我们联络哦。

How to use this tutorial
和文档一起的还有一个压缩包,里面是教程中用到的案例的源文件。我们假设你已经知道怎么用unity编辑器和脚本,如果你不熟悉这些,请先去看unity的视频教程。

多人游戏的debug很麻烦,因为你有两个机器在跑(服务器和客户端)这个项目。所以我们建议你在学习这个教程的时候,在编辑器里跑服务器端,在web里跑客户端。

如果你想把教程中的源文件用在自己的项目里,注意这些文件已经针对教程进行了设置。在你自己的项目里,要确保Run inbackground选项被选中,这可以让你把服务器端在后端激活,避免进入睡眠状态。这样的话你就可以再后端跑服务器。不然的话你就没办法在跑客户端的时候同时在后端跑服务器。你可以打开这个选项在:Edit-Projectsettings-Player.

Unity3d多人网络(从圣典拷贝来的,先看看,因为我觉得这位仁兄比我翻译得好,我也对这个教程作一些修改

Tutorial1:Connect &Disconnect


让我们开始吧
1.打开教程的第一个场景:这个场景在:Tutorial1/Tutorial_1. 这个场景包括了一个摄像机,一Unity3d多人网络(从圣典拷贝来的,先看看,因为我觉得这位仁兄比我翻译得好,我也对这个教程作一些修改个游戏物体和它的脚本,还有另一个物体用来显示场景标题。
2.Build一个webplayer然后运行
3.在编辑器里也开始跑同一个场景,然后点击:Start a server(用默认的IP和端口)
4.在webplayer里点击:Connect as client
5.你应该可以在你的两个项目里都看到:Connection status:Client! 还有:Connectionstatus:Server! 恭喜啦,连接上了!



简单吧;幸运的是这个脚本一点都不难。看一下脚本:Tutorial 1/Connect.js .这个例子里用到的所有的代码都在OnGUI()函数里,看下这个函数,然后确定你明白这个函数是怎么工作的。这段代码挺简单的(如果你看的懂代码的话,嘿嘿),不过我们还是大致看一下这部分代码。

var connectToIP :String = "127.0.0.1";
var connectPort : int = 25001;


//Obviously the GUI is for bothclient&servers (mixed!)
function OnGUI ()
{

  if(Network.peerType ==NetworkPeerType.Disconnected)

 {
      //We are currently disconnected: Not a clientorhost
     GUILayout.Label("Connectionstatus: Disconnected");
  
     connectToIP = GUILayout.TextField(connectToIP,GUILayout.MinWidth(100));
     connectPort =parseInt(GUILayout.TextField(connectPort.ToString()));
  
     GUILayout.BeginVertical();
     if (GUILayout.Button ("Connect as client"))
     {
          //Connect to the "connectToIP" and"connectPort" as entered via the GUI
           //Ignore the NAT fornow
   
          Network.Connect(connectToIP,connectPort);
       }
  
      if (GUILayout.Button ("Start Server"))
      {
          //Start a server for 32 clients using the "connectPort" given viathe GUI
          //Ignore the nat fornow 
   
          Network.InitializeServer(32, connectPort);
      }
      GUILayout.EndVertical();
  
     }

     else

    {
         //We've got a connection(s)!
  

        if(Network.peerType ==NetworkPeerType.Connecting)

       {
  
            GUILayout.Label("Connection status: Connecting");
   
       }

         elseif (Network.peerType == NetworkPeerType.Client)

        {
   
            GUILayout.Label("Connectionstatus: Client!");
            GUILayout.Label("Ping toserver: "+Network.GetAveragePing( Network.connections[0] ));  
   
        }

        else if (Network.peerType ==NetworkPeerType.Server)

        {
   
            GUILayout.Label("Connectionstatus: Server!");
            GUILayout.Label("Connections:"+Network.connections.length);
            if(Network.connections.length>=1)     

            {
                 GUILayout.Label("Ping to first player:"+Network.GetAveragePing(  Network.connections[0]) );
               
         }

         if (GUILayout.Button ("Disconnect"))
        {
             Network.Disconnect(200);
         }
    }
 

}

// NONE of the functionsbelow is of any use in this demo, the code below is only used fordemonstration.
// First ensure you understand the code in the OnGUI() functionabove.

//Client functions called byUnity
functionOnConnectedToServer()

{
     Debug.Log("This CLIENT has connected to aserver"); 
}

functionOnDisconnectedFromServer(info :NetworkDisconnection)

{
    Debug.Log("This SERVER OR CLIENT has disconnected from aserver");
}

functionOnFailedToConnect(error:NetworkConnectionError)

{
    Debug.Log("Could not connect to server: "+error);
}


//Server functions called byUnity
functionOnPlayerConnected(player: NetworkPlayer)

{
    Debug.Log("Player connected from: " + player.ipAddress +":" +player.port);
}

functionOnServerInitialized()

{
    Debug.Log("Server initialized and ready");
}

functionOnPlayerDisconnected(player: NetworkPlayer)

{
    Debug.Log("Player disconnected from: " + player.ipAddress+":" +player.port);
}


// OTHERS:
// To have a full overview of all network functions called byunity
// the next four have been added here too, but they can be ignoredfor now

functionOnFailedToConnectToMasterServer(info:NetworkConnectionError)

{
    Debug.Log("Could not connect to master server: "+info);
}

functionOnNetworkInstantiate (info :NetworkMessageInfo)

{
    Debug.Log("New object instantiated by " + info.sender);
}

functionOnSerializeNetworkView(stream : BitStream, info :NetworkMessageInfo)
{
 //Custom code here (your code!)
}

 



脚本最上面的两个参数(connectToIP 和connectPort)是用来对应GUI对话框里的用户输入,当用户点击链接按钮的时候,它们就会被调用。GUI函数分为4个部分:服务器,客户端已连接,客户端连接中,客户端断开。我们直接使用unity提供的状态:Network.peer.Type来查看当前的链接状态。我们调用Network.Connect函数用来把客户端连接到服务器端,这个函数包含IP,端口还有密码(可选项)作为参数。建立一个服务器也差不多,我们调用另一个函数:Network.InitializeServer。这个函数包含端口和允许的最大连接数量作为参数。注意这里,你在服务器运行的时候,总是可以把连接数调低,但是没有办法超过在服务器初始化时所设置的数值。在你连接服务器或者初始化服务器之前,还有一个选项需要注意:Network.useNat你应该能在connection/initializing函数的代码上方看到它。

NATconnection(Network.useNat)


我们设置Network.useNat为false因为我们不想用NetworkAddress Translation(网络地址转换)。NAT在客户端处在路由器之后的时候很有用(内部局域网)。这个网络Demo应该只在局域网中运行;你肯定没办法连接你朋友家(除非你朋友有个无限制的防火墙/路由器)关于NAT的更多信息请看连接:http://unity3d.com/support/documentation/Components/net-MasterServer.html

现在,最后的一段代码;这十来个函数,会被unity自行调用。其实你不需要它们,就算你把它们都删了,这个Demo还是一样能跑。前六个客户端和服务器端的函数应该很好懂;它们只被客户端或者服务器端调用,如果你想调用这些函数传送的参数,自己去查查unity的手册吧。

最后的三个函数不一样,OnFailedToConnectToMasterServer当你不能连接到主服务器的时候被客户端调用,主服务器的信息在后面会提到。OnNetworkInstantiate被实例化的物体调用,这个在后面也会被提到。OnSerializeNetworkView是我们用来在服务器和客户端之间传送信息的两个方法之一。RPC调用你自己定义的网络信息或网络函数。下一个教程里我们会看一下序列化还有RPC调用。

教程的最后看一下这几个函数:Network.Messages Sent,Class Variables 和ClassFunctions
http://unity3d.com/support/documentation/ScriptReference/Network.html

现在你知道在哪里能找到这些参考信息了,恩~用户手册。我们已经大致介绍了大概75%的信息了,爽吧!

 

Tutorial 2:Sending messages

Tutorial 2A:服务器播放,客户端监视,非实例化。
Tutorial 2/Tutorial 2A1

Unity3d多人网络(从圣典拷贝来的,先看看,因为我觉得这位仁兄比我翻译得好,我也对这个教程作一些修改

不要让这些标题吓到了,打开场景:Tutorial 2/Tutorial 2A1.教程1中网络连接的脚本,现在已经放在:Connect物体上了。另外PlayerCube物体被赋予了Tutorial2A1.js脚本和NetworkView组件。每一个物体,只要需要接受或者发送网络信息的,都需要一个NetworkView组件。你可以在整个游戏中只使用一个NetworkView组件,然后用脚本引用它。但是这样太麻烦了,最简单就是给每个需要网络功能的物体都加一个组件。
Tutorial2A1.js脚本:

functionUpdate(){
 
   //Onlyrun this on the server
  if(Network.isServer)

  {
         //Onlythe server can move thecube!   
         varmoveDirection : Vector3 = newVector3(-1*Input.GetAxis("Vertical"),0,Input.GetAxis("Horizontal"));
       var speed : float = 5;
        transform.Translate(speed* moveDirection * Time.deltaTime);//now really move!
   
}
跑一下这个demo,服务器和客户端都打开。客户端应该能看到服务器移动方块物体。神奇吧,其实这一切就是使用了NetworkView组件的observing(观察)参数,它监视了这个方块的移动。现在看一下方块物体上的Tutorial2A1.js脚本。这段代码只能在服务器上跑(因为用了Network.isServer来检查是否为服务器端):当服务器端的玩家移动方块,它就会立刻移动。不过你也能看到客户端上方块的移动特别卡,但是不要担心,我们Unity3d多人网络(从圣典拷贝来的,先看看,因为我觉得这位仁兄比我翻译得好,我也对这个教程作一些修改回头会解决这个问题,现在先讲最基本的内容。

现在,怎么让客户端知道服务器端的物体移动了呢?看一下附加在物体上的NetworkView组件。它检测了物体的transform(变换)属性。也就是说unity会自动发送物体的变换属性(包括了位置,旋转角度和缩放的Vector3数值)。它只会把信息从服务器端发送到客户端,反之就不行,因为服务器端独占了NetworkView的功能。客户端就不能发送信息,只能够接收。



我们看一下NetworkView的其他选项,稍微总结一下。PlayerCube物体的Networkview组件中,Statesynchronization选项,被设定为Reliablecompressed。这说明只有被观察的参数发生改变的时候,它才会发送信息。如果服务器端15分钟都诶有移动方块物体,它就绝对不会发送任何信息,智能吧~。如果设置成Unreliable,无论参数有没有变,它都一致在发送信息。最后一个,如果设置Statesynchronization为Off,会完全停止NetworkView所有的网络同步行为。如果你的NetworkView组件没有在对物体进行监视,你可以把同步选项关闭(不过也不是必须的)。如果你不明白我们为什么需要这么一个关掉了同步选项的NetworkView组件,就这么给你说吧,因为“RemoteProcedure Calls”(远程程序调用)需要一个NetworkView组件,但是并不需要Statesynchronization和observed选项。不过你还是可以把RPC和observed一起用。RPC的内容会在下面的教程2A3里讲到。基本上它就是一个你自己定义的网络信息收发机制。

 

Tutorial2/Tutorial2A2

如果你想让方块物体沿着Y轴移动怎么办,或者你想控制unity同步的具体内容。跑一下教程2/教程2A2.这个游戏应该和之前一摸一样,但是后端的代码已经改变了。PlayerCube上的NetworkView组件现在监测的是“Tutorial2A2.js”脚本。

functionUpdate()

{
 
   if(Network.isServer)

   {
        //Only the server can move thecube!   
        var moveDirection :Vector3 = new Vector3(-1*Input.GetAxis("Vertical"),0,Input.GetAxis("Horizontal"));
       var speed : float = 5;
       transform.Translate(speed * moveDirection *Time.deltaTime);
     }
 
}


functionOnSerializeNetworkView(stream : BitStream, info :NetworkMessageInfo)
{
    if (stream.isWriting)

    {
           //Executed on the owner of the networkview;in this case the Server
           //The server sends it's position over thenetwork
  
          var pos : Vector3 =transform.position;  
          stream.Serialize(pos);//"Encode" it, and sendit
  
            
  
        }

       else

       {
            //Executed on the others;in this case the Clients
            //The clients receive a position and set theobject to it
  
            var posReceive :Vector3 = Vector3.zero;
           stream.Serialize(posReceive);//"Decode" it and receive it
           transform.position=posReceive;
  
            
       }
}

具体就是说,这个NetworkView组件现在在检测脚本内的“OnSerializeNetworkView”函数。看一下这个函数.我们现在明确的指定了我们想要监测的内容。你可以用这个函数来同步具体你需要的内容,再说一次,当你选中Reliabledeltacompressed的时候,只有当参数发生变化的时候才会被发送出去。OnSerializeNetworkView函数有点诡异,它虽然是用来发送和接受数据,但是unity会查看Networkview组件的使用者,然后决定你是不是能够发送数据,如果你是服务器端,就调用“stream.isWriting”部分的代码,你就可以发送数据。如果你是客户端,就调用“else”部分的代码,你就只能接收数据。

 

Tutorial2/Tutorial 2A3


这是我最喜欢的发送信息的方法,也是最后一个方法;Remote ProcedureCalls。我之前提到过这个,你可以去看看Tutorial 2/Tutorial2A3的例子,搞个明白到底是怎么一回事。这个Demo和之前两个实现了一样的功能。不过Networkview不再监视任何物体,同步选项也已经被关闭。秘密就在Tutorial2A3.js这个脚本里,特别是这一行networkView.RPC("SetPosition", RPCMode.Others,transform.position);.

 

Tutorial 2A3.js脚本:

 

private varlastPosition : Vector3;

functionUpdate()


    if(Network.isServer)

   {
        //Only the server can movethecube!   
       var moveDirection : Vector3 = newVector3(-1*Input.GetAxis("Vertical"),0,Input.GetAxis("Horizontal"));
       var speed : float = 5;
       transform.Translate(speed * moveDirection * Time.deltaTime);
  
        //Save some network bandwidth; only send anrpc when the position has moved more thanX
        if(Vector3.Distance(transform.position,lastPosition)>=0.05)

       {
             lastPosition=transform.position;
   
              //Send the position Vector3 over to theothers; in this case allclients
            networkView.RPC("SetPosition", RPCMode.Others,transform.position);
        }
    }
 
}


@RPC
function SetPosition(newPos : Vector3)

{
     //This RPC is in this case always called bythe server,
     // but executed on all clients
 
    transform.position=newPos; 
}

服务器调用了RPC,这个RPC会要求客户端调用“SetPosition”函数,同时这个RPC还包含这一个新的位置信息“transform.position”,然后所有的客户端都调用”SetPosition”这个函数。下面是整个移动的过程:

1.服务器端玩家按下按键,他控制的物体移动。(代码14-18行)
2.服务器用移动的数值和上次更新的数值比较,如果差距大于设置的最小值,就发送一个RPC给出了自己的所有人,这个RPC包含了新的物体位置。(代码20-25行)
3.所有的客户端接收到RPC的设置物体位置命令,并且得到其中包括的新位置参数,然后再让它们本地执行位置移动的代码。
4.现在无论服务器还是客户端,大家的物体都处在相同的位置了~!

如果我们想要使用RPC函数,需要在脚本中这个函数的上面加上“@RPC”(C#里面是”[RPC]”).当发送一个RPC的时候,我们可以指定下列的接收器:

 RPCMode.Server    只发送给服务器
 RPCMode.Others  发送给除了调用者之外的所有人
 RPCMode.OthersBuffered  发送给除了调用者之外的所有人,暂存的内容
 RPCMode.All 发送给包括调用者在内的所有人
 RPCMode.AllBuffered 发送给包括调用者在内的所有人,暂存的内容


暂存的内容,指的是无论何时新玩家连接到服务器,都将会接收到这个信息。一个包含暂存内容的RPC可以用于比如说生成玩家的时候。这个暂存的内容会被服务器记住,然后每个玩家连接到服务器的时候,都会先收到一个生成玩家的RPC,这个RPC会在这个刚连接的新玩家的客户端中,生成其他所有在他之间加入服务器的玩家。

如果你大致已经明白上面讲的所有的内容的话,你已经很牛啦!我们已经讲完了所有的基础内容,现在可以关注一下细节问题了。

 

Tutorial 2B: Server and client(s) play,with instantiating.


我们现在要研究一下FPS游戏的基本细节。我们需要搞一个多人游戏,可以包括服务器端的玩家在内,并且也要可以剔除服务器端的玩家,把服务器放在后台。所以我们决定当新的客户端连接到服务器的时候,再生成玩家,而不是把玩家设置成物体,直接放在场景中。打开场景“Tutorial2/Tutorial2B”,服务器端还是在编辑器中,客户端在web里,都打开。移动一下方块,看看在客户端和服务器端是不是都工作正常。

PlayerCube物体已经从场景中移除了,我们新建了一个Spawnscript物体,并且给它赋予了Spawnscript.js脚本。

public varplayerPrefab : Transform;


functionOnServerInitialized()

{
    Spawnplayer();
}

functionOnConnectedToServer()

{
    Spawnplayer();
}

functionSpawnplayer()

{
 
     var myNewTrans : Transform =Network.Instantiate(playerPrefab,               transform.position, transform.rotation, 0);

}
function OnPlayerDisconnected(player:NetworkPlayer)

{
     Debug.Log("Clean up after player " + player);
     Network.RemoveRPCs(player);
     Network.DestroyPlayerObjects(player);
}

functionOnDisconnectedFromServer(info :NetworkDisconnection)

{
     Debug.Log("Clean up a bit after server quit");
     Network.RemoveRPCs(Network.player);
     Network.DestroyPlayerObjects(Network.player);
 
      
      Application.LoadLevel(Application.loadedLevel);
}

当玩家(包括服务器和客户端)开始的时候,这个生成玩家的脚本会生成我们指定的预设物体(这里就是生成玩家-其实是方块)。生成脚本包含了位置,旋转角度和物体所在小组的信息。生成的物体会复制Spawnscripts物体本身的位置和角度信息,并且设置小组号为0(现在不用关心小组的事儿)。当我们断开连接的时候,会移除所有生成的预设物体。谁调用的Network.Instantiate谁就会自动获得这个函数本次生成的物体。这样我们就可以正确的控制不同的方块物体(服务器端和客户端就不会混在一起)。
“Tutorial_2B_Playerscript.js”脚本使用了Tutorial2AB的代码,不同的地方是,只有物体的所有者的输入才会被监测。

 

Tutorial 3:Authoritative servers
Unity3d多人网络(从圣典拷贝来的,先看看,因为我觉得这位仁兄比我翻译得好,我也对这个教程作一些修改
之前的服务器设置,被称之为“非权威性”服务器;服务器对所有的网络信息没有任何控制权,客户端会和服务器共享物体的位置信息,并且所有的终端都接受并且执行这些信息。在你的FPS游戏里,你肯定不想有玩家能瞬间移动,水上飞什么的。所以一般来说服务器都是“权威性”服务器。设置一个权威性服务器也不需要什么特别难的代码,不过它的确需要你设计代码框架的时候,稍微做些调整。你需要在服务器端完成所有的工作并且检查所有的通讯。

我们回头看一下上个教程B2,怎么才能把它修改成权威性服务器呢。首先,服务器需要生产玩家,玩家不能决定他们被生成的时间和地点。其次,服务器要告诉所有的客户端所有物体的位置,客户端之间无法发送和接受信息。因为只有服务器能够移动物体的位置,客户端的玩家想要移动的话,必须向服务器发送他的所需要的移动信息,然后接收服务器指令才能移动。

我们要发送所有客户端的移动输入命令到服务器端,服务器会处理这些数据,然后送回结果数据(新的位置)到客户端。看一下Tutorial3场景。功能还是和以前一样,但是内部处理机制已经不一样了。移动起来可能比以前感觉更卡一点,但是现在这个暂时不重要。

这个例子里没有新脚本,只有Playerscript脚本和spawnscript脚本的内容有改变。我们先看下

Tutorial_3_Spawnscript.js。

 

public varplayerPrefab : Transform;
public var playerScripts : ArrayList = newArrayList();

functionOnServerInitialized()

{
     //Spawn a player for the serveritself
     Spawnplayer(Network.player);
}

functionOnPlayerConnected(newPlayer: NetworkPlayer)

{
      //A player connected to me(theserver)!
     Spawnplayer(newPlayer);

 
function Spawnplayer(newPlayer :NetworkPlayer)

{
     //Called on the serveronly 
     var playerNumber : int =parseInt(newPlayer+"");
     //Instantiate a new object for this player,remember; the server is therefore theowner.
     var myNewTrans : Transform =Network.Instantiate(playerPrefab, transform.position,transform.rotation,playerNumber);
 
     //Get the networkview of this newtransform
     varnewObjectsNetworkview : NetworkView =myNewTrans.networkView;
 
     //Keep track of this new player so we canproperly destroy it when required.
     playerScripts.Add(myNewTrans.GetComponent(Tutorial_3_Playerscript));
 
     //Call an RPC on this newnetworkview, set the player who controls thisplayer
      newObjectsNetworkview.RPC("SetPlayer",RPCMode.AllBuffered, newPlayer);//Set it on theowner
}

 

functionOnPlayerDisconnected(player: NetworkPlayer)

{
       Debug.Log("Clean up after player " +player);

       for(var script : Tutorial_3_Playerscript inplayerScripts)

       {
             if(player==script.owner)

              {

                   //We found the playersobject

                   //remove the bufferd SetPlayercall
                   Network.RemoveRPCs(script.gameObject.networkView.viewID);

                 //Destroying the GO will destroyeverything

                 Network.Destroy(script.gameObject);                    

                playerScripts.Remove(script);//Remove this player from thelist
                   break;
               }
        }
 
          //Remove the buffered RPCcall for instantiate for thisplayer.
          var playerNumber :int = parseInt(player+"");
         Network.RemoveRPCs(Network.player,playerNumber);
 
 
          // The next destroys will not destroyanything since the players never
          // instantiated anything nor bufferedRPCs
          Network.RemoveRPCs(player);
         Network.DestroyPlayerObjects(player);
}

functionOnDisconnectedFromServer(info :NetworkDisconnection)

{
         Debug.Log("Resetting the scene the easyway.");
         Application.LoadLevel(Application.loadedLevel); 
}

客户端在这个脚本里没有任何操作,每当客户端连接的时候服务器端才开始生成物体。服务器端还会保存一个已连接客户端的列表,列表中还包括了Playerscripts的信息,这样在一个客户端下线的时候,服务器就可以删除正确的玩家物体。这个Spawnscript脚本是一个纯粹的服务器端脚本,和客户端的“OnDisconnectedFromServer”函数没有任何关系。现在我们再看一下

 

Tutorial_3_Playerscript.js脚本:

 

public var owner :NetworkPlayer;

//Last input value, we'resaving this to save networkmessages/bandwidth.
private var lastClientHInput : float=0;
private var lastClientVInput : float=0;

//The input values the serverwill execute on this object
private var serverCurrentHInput : float = 0;
private var serverCurrentVInput : float = 0;


function Awake()

{
     // We are probably not the owner of thisobject: disable this script.
     // RPC's and OnSerializeNetworkView will STILL get trough!
     // The server ALWAYS run this scriptthough
     if(Network.isClient)

     {
          enabled=false;  // disable this script (thisenables Update()); 
      
}


@RPC
function SetPlayer(player : NetworkPlayer)

{
     owner = player;
     if(player==Network.player)

     {
           //Heythats us! We can control this player: enable this script (thisenablesUpdate());
          enabled=true;
      }
}

functionUpdate()


      //Clientcode
      if(owner!=null&&Network.player==owner)

      {
            //Onlythe client that owns this object executes thiscode
           var HInput : float= Input.GetAxis("Horizontal");
         var VInput : float =Input.GetAxis("Vertical");
  
           //Is our input different? Do we need toupdate theserver?
           if(lastClientHInput!=HInput ||lastClientVInput!=VInput )

           {
                 lastClientHInput = HInput;
                 lastClientVInput =VInput;   
   
                  if(Network.isServer)

                 {
                       //Too bad a server can't send an rpc toitself 

                       //using "RPCMode.Server"!...bugged :[

                       SendMovementInput(HInput,VInput);
                  }

                  else if(Network.isClient)

                  {
                       //SendMovementInput(HInput,VInput); //Use this (and line 64) for simple"prediction"
                    networkView.RPC("SendMovementInput", RPCMode.Server, HInput,VInput);
   }
   
  }
 }
 
 //Server movementcode
 if(Network.isServer)//Alsoenable this on the client itself: "||Network.player==owner)

{

      //Actually move the playerusing his/herinput
      var moveDirection :Vector3 = new Vector3(serverCurrentHInput, 0,serverCurrentVInput);
      var speed : float = 5;
      transform.Translate(speed * moveDirection * Time.deltaTime);
 }
 
}

 


@RPC
function SendMovementInput(HInput : float, VInput :float)


       //Called on the server
       serverCurrentHInput= HInput;
      serverCurrentVInput = VInput;
}


functionOnSerializeNetworkView(stream : BitStream, info :NetworkMessageInfo)
{
     if (stream.isWriting)

     {
            //This is executed on theowner of the networkview
            //The owner sends it's position over thenetwork
  
            var pos : Vector3 =transform.position;  
          stream.Serialize(pos);
//"Encode" it, and sendit
    
         }

       else

       {
              //Executedon all non-owners
             //This client receive a position and set the object to it
  
              var posReceive :Vector3 = Vector3.zero;
             stream.Serialize(posReceive);
//"Decode" it and receive it
  
             //We've just recieved thecurrent servers position of this object in'posReceive'.
  
              transform.position =posReceive;  
               //Toreduce laggy movement a bit you could comment the line above anduse position lerping below instead: 
              //transform.position =Vector3.Lerp(transform.position, posReceive, 0.9); //"lerp" to theposReceive by90%
  
        }
}

 

这个脚本现在不只被Networkview所有。因为现在由服务器端来生成所有的物体,所以全部的Networkview都为服务器所有。所以现在我们使用每个终端自己的“所有者”参数来控制,哪一个网络上的玩家会控制哪一个物体。playerscript脚本的所有者会发送移动信息到服务器。服务器执行这个移动信息并且负责移动玩家物体。这样一来,我们就有了一个“权威性”的服务器!

关于卡的问题:在之前的例子里,玩家物体会在按下按键后立即移动,但是当我们使用权威性服务器,我们需要发送移动信息给服务器,然后服务器会处理它,然后再发回一个移动指令,然后我们才能移动物体。我们当然是想让服务器端有所有的控制权,但是我们不想让客户端等太长时间。其实这个问题也很简单,只要让客户端也同时计算移动信息,然后再让服务器端的信息覆盖客户端的计算结果,这样服务器端总是有控制权。很简单吧。Tutorial_3_Playerscript.js这个脚本在客户端调用了“SendMevementInput(HIput,Vinput)”函数。这里你可以发送一个移动信息RPC到服务器(第56行代码)。随后这个SendMovementInputRPC 会调用客户端移动脚本里的Update()函数的最后一部分代码,来更新物体的移动。同时在本地调用这一段代码:“||Network.player == owner)”(第64行代码)。这样就可以确保客户端的移动立刻就能执行,而且让服务器端的计算结果为最终结果。

虽然我们设置了让客户端可以“预测”物体的移动,但是还是有些卡,在代码的第100行这里有一段代码,它合并了当前的物体位置和服务器发送来的位置,并且以服务器的位置为主。你还可以把服务器发送来的位置保存为一个参数,然后用Vector3.Lerp这个命令来进行插值。这样你就可以在Update函数里进行平滑的插值,而不是只能在OnSerializeNetworkView函数里进行一次插值。
Unity3d多人网络(从圣典拷贝来的,先看看,因为我觉得这位仁兄比我翻译得好,我也对这个教程作一些修改
注意一下,其实你不用总是在你的多人游戏里用“权威性”服务器。比如我们公司的Crashdriver3D游戏,就用了非权威性服务器。玩家可以恶意修改他的赛车位置;但是谁在乎呢~这种修改最多也就能让玩家得到很高的分数。之后我们再检查那些高的离谱的得分要容易得多。总而言之:想明白你究竟为什么要用权威性服务器。另外也要知道,如果你直接修改权威性服务器,也能作弊哦。

Further network subjects explained

Unity editor options related to networking

“Edit - Project settings - Network”
Sendrate(发送率):这个选项决定了每秒钟发送多少次网络信息(Unreliable 或者 Reliable deltacompressed).注意,这个选项对RPC信息没有效果。在不影响游戏视觉效果的前提下,尽量把这个数值调低。

Debug level:改变多少debug信息会在编辑器中显示。

“Edit - Project settings - Player”
Run in background: Yes/No 当运行服务器端时,需要打开这个选项以便服务器可以在后台保持通讯。


Limiting traffic: Scoping and grouplimiting

你可以通过限制数据的数量来提高通讯性能。在多人游戏里,玩家不用接受所有的信息。一定距离以外发生的事情,对玩家也就没什么意义了。这里有两种方法,可以让玩家拒收信息:“组”或者”玩家“。
首先所有的NetworkView组件,要设置一个SetScop函数:
function SetScope (player : NetworkPlayer, relevancy : bool) :bool

默认情况下,这个函数为true。你可以设置为false如果某个玩家已经离你足够远,然后你就不会再接收到他的信息。不过很可惜,这个函数只能用于NetworkViwe的observe属性,对RPC不起作用。

网络函数:
static function SetReceivingEnabled (player : NetworkPlayer, group: int, enabled : bool) : void
static function SetSendingEnabled (group : int, enabled : bool) :void
这两个函数可以根据网络组来限制信息的发送和接收。比如说,你可以把地图划分为32份,玩家只会发送/接收玩家周围的8个格子(还有玩家本身的一个,总共9个)。但是很遗憾在unity网络库中,你最多只能有32个组,虽然对FPS这样的游戏来说也够用了,但是对真正的MMO游戏,还是差很远。


Securing the network connection

添加AES加密,CRS,随即加密SYNCookies和RSA加密好像都挺复杂的哈。幸运的是我们可以只用一行代码就搞定这些:
function StartServer ()
{
     Network.InitializeSecurity();
//就是这一行!
     Network.InitializeServer(32, 25000);
}

只是记得在初始化服务器之前调用Network.InitializeSecurity()函数一次,安全系统会让每个信息包增加15比特。


反作弊

就算是有了之前的安全系统,当你设计游戏的时候,还是要考虑到最糟的情况。假设玩家对程序懂的和你一样多,并且他们可以随意修改你的网络信息包,搞出一些离谱的数值来。所以总是要在服务器端检查你接收到的数据。只要设计网络功能的时候巧妙一些,就不用为了反作弊写一大堆额外的代码。


Using aproxy
关于使用代理的事,手册上已经说的很清楚了,自己看下链接吧:
http://unity3d.com/support/documentation/ScriptReference/Network-useProxy.html
虽然我们对使用代理来改善网络链接很有兴趣,但是我们还没有仔细研究这部分功能。

Combat Lag: prediction, extrapolation andinterpolation
(战斗延迟:预测,外推法和插值)

我们已经在Tutorial3里大致提到了这些问题:当你使用权威性服务器来做计算的时候,同时可以让客户端做预测计算来减少延迟。

以下摘至Unity手册:
“我们用来推测玩家行为的方法也可以用来推测敌人的行为。外推法就是根据服务器上一帧所收到的信息,计算一个敌人可能的方向和速度,然后假设敌人会继续朝这个方向移动。

插值是怎么回事呢,当丢包的时候,通常玩家和敌人会突然卡住不动了,然后当下一个包发过来的时候,再跳到新的位置。但是我们可以设置一个延时(通常大约100毫秒)然后把之前的位置和新的位置做一个插值,这样的话,丢包的时候,玩家的移动依然是平滑的。”

在unity的官方例子里可以找到关于插值和外推法的例子,这个教程中的FPS例子里,也有这相关内容。另外你也可以通过提高网络发送率来提高同步的精度。

Manually allocate networkviewID's
(手动分配NetworkViewID)
有时候Network.Instantiate 对权威性服务器的支持不好。如果手动分配网络设置的ID可以获得更好的控制。
代码例子:http://unity3d.com/support/documentation/ScriptReference/Network.AllocateViewID.html


Networkloading
对于网络工作来说,只要网络连接情况良好,无论服务器或者客户端上跑的是什么内容都无关紧要。也就是说你可以在服务器端跑一个游戏场景,而在一个刚连接的新客户端跑游戏大厅。通常都不会出什么问题,除非服务器向所有的客户端发送“缓存的实例化”游戏物体的命令。因为这个原因,你最好在载入游戏的时候暂时关闭网络通讯。你可以在客户端成功连接服务器之后,立即调用下面的代码:“Network.isMessageQueueRunning=false;”这样就可以关闭网络通讯。网络大厅的例子里有这个代码的应用。

告诉你一个秘密,其实一个服务器可以同时跑多个场景/关卡,只要你能巧妙地设置好网络组。只是要小心不同场景中的玩家的碰撞信息。

Real lifeexamples

Example 1:Chatscript

 

public var usingChat : boolean =false; //Canbe used to determine if we need to stop player movement since we'rechatting
var skin :GUISkin;      //Skin
var showChat : boolean=false;   //Show/Hidethe chat

//Private vars used by thescript
private varinputField : String= "";

private var scrollPosition :Vector2;
private var width : int= 500;
private var height : int= 180;
private var playerName : String;
private var lastUnfocusTime : float =0;
private var window :Rect;
 
//Server-onlyplayerlist
private varplayerList = new ArrayList();
class PlayerNode

{
    varplayerName : String;
    varnetworkPlayer : NetworkPlayer;
}

private var chatEntries = newArrayList();
class ChatEntry
{
   var name : String= "";
    var text :String= ""; 
}

functionAwake()

{
   window =Rect(Screen.width/2-width/2, Screen.height-height+5, width,height);
 
   //We getthe name from the masterserver example, if you entered your namethere;).
   playerName =PlayerPrefs.GetString("playerName", "");
   if(!playerName || playerName=="")

   {
       playerName = "RandomName"+Random.Range(1,999);
    
 
}


//Clientfunction
functionOnConnectedToServer()

{
   ShowChatWindow();
   networkView.RPC ("TellServerOurName", RPCMode.Server,playerName);
   // //We could have also announcedourselves:
    //addGameChatMessage(playerName" joined the chat");
    // //Butusing "TellServer.." we build a list of active players which we canuse for other stuff as well.
}

//Serverfunction
functionOnServerInitialized()

{
  ShowChatWindow();
  //I wish Unity supported sending an RPC onthe server to the server itself :(
   // If so; we could use thesame line as in"OnConnectedToServer();"
  var newEntry : PlayerNode = newPlayerNode();
  newEntry.playerName=playerName;
  newEntry.networkPlayer=Network.player;
  playerList.Add(newEntry); 
  addGameChatMessage(playerName+" joined thechat");
}

//A handy wrapper function to get thePlayerNode by networkplayer
function GetPlayerNode(networkPlayer :NetworkPlayer)

{
   for(var entry : PlayerNodein  playerList)

  {
    if(entry.networkPlayer==networkPlayer)

    {
        return entry;
    }
   }
 Debug.LogError("GetPlayerNode: Requested aplayernode of non-existing player!");
 return null;
}


//Server function
function OnPlayerDisconnected(player:NetworkPlayer)

{

   addGameChatMessage("Playerdisconnected from: " + player.ipAddress+":" + player.port);
 
  //Remove player from the serverlist
    playerList.Remove(GetPlayerNode(player) );
}

functionOnDisconnectedFromServer()

{
   CloseChatWindow();
}

//Serverfunction
functionOnPlayerConnected(player: NetworkPlayer)

{
   addGameChatMessage("Playerconnected from: " + player.ipAddress +":" + player.port);
}

@RPC
//Sent bynewly connected clients, recieved byserver
functionTellServerOurName(name : String, info :NetworkMessageInfo)

{
   var newEntry : PlayerNode =new PlayerNode();
  newEntry.playerName=name;
  newEntry.networkPlayer=info.sender;
  playerList.Add(newEntry);
 
   addGameChatMessage(name+"joined the chat");
}

 


function CloseChatWindow ()
{
   showChat =false;
   inputField = "";
   chatEntries = newArrayList();
}

function ShowChatWindow ()
{
   showChat = true;
   inputField = "";
   chatEntries = newArrayList();
}

function OnGUI ()
{
   if(!showChat){
   return;
 }
 
 GUI.skin =skin;  
   
 if (Event.current.type == EventType.keyDown&& Event.current.character == "\n"&& inputField.Length<= 0)
 {
   if(lastUnfocusTime+0.25<Time.time)

   {
      usingChat=true;
      GUI.FocusWindow(5);
      GUI.FocusControl("Chat input field");
    }
  }
 
 window = GUI.Window (5, window, GlobalChatWindow,"");
}


function GlobalChatWindow (id :int)


  GUILayout.BeginVertical();
   GUILayout.Space(10);
  GUILayout.EndVertical();
 
    // Begin a scroll view. All rects are calculatedautomatically -
    // it willuse up any available screen space and make sure contents flowcorrectly.
    // This iskept small with the last two parameters to force scrollbars toappear.
  scrollPosition =GUILayout.BeginScrollView (scrollPosition);

  for (var entry : ChatEntry in chatEntries)
   {
     GUILayout.BeginHorizontal();
     if(entry.name=="")

     {

         //Gamemessage
         GUILayout.Label(entry.text);
     }

     else

     {
        GUILayout.Label (entry.name+": "+entry.text);
     }
    GUILayout.EndHorizontal();
    GUILayout.Space(3);
  
  }
 // End the scrollview we beganabove.
   GUILayout.EndScrollView ();
 
    if(Event.current.type == EventType.keyDown&& Event.current.character == "\n"&& inputField.Length> 0)
    {
       HitEnter(inputField);
    }
    GUI.SetNextControlName("Chat input field");
    inputField = GUILayout.TextField(inputField);
 
 
    if(Input.GetKeyDown("mouse 0"))

    {
       if(usingChat)

       {
           usingChat=false;
           GUI.UnfocusWindow ();//Deselect chat
           lastUnfocusTime=Time.time;
        }
     }
}

function HitEnter(msg :String)

{
    msg = msg.Replace("\n", "");
    networkView.RPC("ApplyGlobalChatText", RPCMode.All, playerName,msg);
    inputField = "";
//Clearline
    GUI.UnfocusWindow ();//Deselect chat
    lastUnfocusTime=Time.time;
    usingChat=false;
}


@RPC
function ApplyGlobalChatText (name : String, msg : String)
{
    var entry =new ChatEntry();
    entry.name =name;
    entry.text =msg;

   chatEntries.Add(entry);
 
    //Remove oldentries
    if (chatEntries.Count > 4)

    {
         chatEntries.RemoveAt(0);
    }

 scrollPosition.y= 1000000; 
}

//Add game messagesetc
functionaddGameChatMessage(str : String)

{
   ApplyGlobalChatText("", str);
   if(Network.connections.length>0)

    {
       networkView.RPC("ApplyGlobalChatText", RPCMode.Others, "",str); 
    
}
Unity3d多人网络(从圣典拷贝来的,先看看,因为我觉得这位仁兄比我翻译得好,我也对这个教程作一些修改


Example1/Example1_Chat基本是就是教程1的代码加上一个聊天脚本。在游戏里添加一个聊天功能简单的要死。你可以重复使用这个脚本只要没有其他特殊要求。只是记得要对应好玩家的名字。现在可以显示4行聊天信息。你要想修改代码让它能显示更多聊天内容的话,可以用yield或者coroutine来删除或者淡出旧的信息。服务器中保存了一个玩家的列表。在真正的游戏中你应该做一个单独的游戏玩家列表,而不是把玩家列表放在聊天脚本里。

Example2: Masterserverexample
Unity3d多人网络(从圣典拷贝来的,先看看,因为我觉得这位仁兄比我翻译得好,我也对这个教程作一些修改


打开场景“Example2/Example2_menu”.这个例子中使用到了masterserver来显示所有正在进行中的游戏。快速游戏的按钮可以让玩家随机加入第一个可以进入的游戏。下面的进阶选项中玩家可以创建一个服务器,填写IP和端口以便别的玩家可以直接连接到他,或者用masterserver的游戏列表直接手动选一个。唯一没有的功能是用密码开房间。这个功能可以简单的加在创建房间和连接的步骤之间,然后再游戏列表上你要加一个输入密码的窗口。

这个例子中的”游戏“只是展示了怎么连接服务器和客户端,你可以轻易地替换游戏内容,网络功能也一样可以正常使用。你只需要设置“Network.isMessageQueueRunning=true;”。我们之前把这个函数关掉了,因为在客户端还在加载的时候,我们要防止从游戏局内发出一些无法识别的网络信息。还有一个事就是在服务器开始游戏之后,记得要在masterserver注册一下游戏。


Example 3:Lobby system
Unity3d多人网络(从圣典拷贝来的,先看看,因为我觉得这位仁兄比我翻译得好,我也对这个教程作一些修改


“Example3/Example3_lobby”:这个例子和第一个例子很像,唯一的不同是它给每个游戏创建了一个大厅,并且有密码选项。在大厅里,只会显示给玩家masterserver的游戏列表,游戏一旦开始就会被从列表中移除。还是一样,你要想用这些功能,直接拷贝代码然后针对你的游戏做点调整就行,只是记得在游戏场景里开启信息队列。


Example4:FPS game
Unity3d多人网络(从圣典拷贝来的,先看看,因为我觉得这位仁兄比我翻译得好,我也对这个教程作一些修改


由于大多数想学unity网络功能的人都想做一个FPS游戏,我决定根据最后一个例子来做一个FPS的游戏。这个FPS例子是用的非权威性服务器,所以如果你愿意,可以重新设计代码,把它改成一个权威性服务器的游戏。

这个例子用了masterserver的代码来连接主菜单。游戏中的功能有:聊天,得分板,移动,射击,拾取物体。

如果你想用这个例子作为基础来写你自己的游戏,你可能用到的新功能大概有:

权威性服务器控制移动:预防作弊
角色动画:远程同步动画,或者让客户端计算何时播放正确的动画
武器切换

和多人游戏不太相关的项目有:
准星
改进GUI
观看模式
游戏回合时间
Tips!

同时打开多个unity(方便网络功能的查错)
你无法打开相同的unity项目两次,所以你要用一个脚本来开第二个unity。你可以拷贝你的项目,一个跑服务器另一个跑客户端,但是你要保存你的改动2次。


Windows上:

修改快捷方式的属性。 在后面加上个-projectPath,例如:   "D:\ProgramFiles\Unity\Editor\Unity.exe" -projectPath
这样的话运行的时候窗口底部会报一个找不到路径的错误,无所谓,clear一次就行。

Mac上:
把Unity.app复制一份。分开运行。

OnSerializeNetworkView bug in 2.6 (and earlier)

我一直都不想用OnSerializeNetworkView,因为我喜欢RPC~,不过当我用OnSerializeNetworkView的时候发现它有个缺陷。这个缺陷只发生在你要分配NetworkView ID给你自己的时候。

具体如下:
当你用OnSerializeNetworkView和NetworkView监视功能的时候,服务器端的玩家没有问题,但是当客户端玩家连接的时候,它会报错说不知道第一个玩家(服务器端玩家)的networkviewID

"Received state update for view ID ******random info here aboutyour specific number*** but no
initial state has ever been sent. Ignoring message."

这个问题是因为新的客户端从来没有初始化过他们自己的networkview,所以新的客户端连接的时候就会出问题。

Also see: http://forum.unity3d.com/viewtopic.php?p=77193

Group limit
你最多只能建32个组,你可以通过代码指定48个组,但是assigning 48 works like assigning482=16.(完全不明白)

Scopes
unity2.1有好多新功能,不过好像都不是支持RPC的,只支持OnSerializeNetworkWiew。

RPC bug?
当你使用权威性服务器,而且服务器本身也在跑一个玩家的时候,会用到下面的代码:

networkView.RPC("SendUserInput", RPCMode.Server, horizontalInput,verticalInput);

但是上面的代码其实不能用,用下面的:

if(Network.isServer)
{
SendUserInput(horizontalInput, verticalInput);
}
else
{
networkView.RPC("SendUserInput", RPCMode.Server,horizontalInput,verticalInput);
}


Run dedicated servers(专用服务器)
现在untiy的专业服务器还不太完善,不过也凑合能用。在Mac上面跑专用服务器的话,要执行的时候添加一个批处理参数。

Win从unity2.6开始也加上了相同的功能。参见以下链接
http://unity3d.com/support/documentation/Manual/CommandLine Arguments.html
当你用专用服务器的时候,应该用“Application.targetFrameRate”来控制帧率,否则unity可能会把帧率搞的太高,拖慢性能。

Connection issues: How to connect over the internet
(连接问题:如果通过互联网连接)

连接方面来说,本地局域网和互联网差不多,只是局域网速度肯定快一点。当你让你的游戏在局域网上跑起来之后,你会发现在互联网上跑设置起来还是有点麻烦。下面这个表可以帮助你检查哪儿出的问题:

互联网连接不工作:

确定是连接的互联网还是局域网
两台电脑都连接了互联网没?
确定两台电脑的防火墙没有关闭你用到的端口,或者暂时关闭防火墙
尝试一下直接连接,开一个服务器,然后用另一台电脑做客户端直接连接服务器端的互联网网络地址
如果还是连不上,你的路由器可能屏蔽了连接功能,作为安全措施。你有两个选择
1.用NAT穿透(见masterserver例子)然后祈祷你的路由器支持穿透功能。
2.你可以手动在路由器中打开你用到的端口,然后从这个端口转发所有的连接到你的内部局域网IP地址。这个绝对能用,但是你不能保证所有的玩家都会设置路由器端口。
Other networking options
这里是一些unity网络的资料,有一些第三方的网络支持,可以自行查看(2009.8月)
Create your own custom RakNet backend
• Smartfox
• Photon & Neutron from ExitGames
• Project DarkStar
• Netdog
• Lidgren

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值