之前我们讨论过“如何把Steamworks.Net和Unity整合起来”,这是一个很好的开始,现在我们研究深一点,谈一谈Steam中的多人游戏。这不是教程,但是可以指导你在你的游戏中如何使用Steamworks。我们将使用Steamworks .Net的库, steam_api.dll。
 
注意,你的游戏运行,需要有一个Steam的App Id。你可以在这里申请一个( greenlit),或者由Valve公司直接通过的游戏也有一个Id。这是一个指南,如果你只是想知道Steam是否是一个方便的平台,读下去……


 
P2P 多人游戏

 
Steamworks最伟大的特性就是它的匹配机制和P2P网络通信,你不用担心服务器的设置,所有的事情都已经准备就绪。
 
你可能不熟悉用P2P的连接来建一个多人游戏,现在流行的方法是为客户端和服务器商搭建独立的实体。这种情况下,客户端是游戏本身,服务端是一个包括了服务端逻辑,连接了所有的玩家,保护玩家数据正常不受欺骗的应用程序。客户端、服务端这种模式可能对大型游戏更好,对于一些小的,非竞技类游戏,考虑放弃服务器部分,两个客户端直接对话可能会更好一些。
 
真的很简单吧?
 
Steamworks尽可能的让它简单。你不需对链接担心,你只需要一个SteamID(SteamId是用户唯一的标识,它是封装在CSteamID对象中的一个很大的数,你可以用得到跟你互动的用户的CSteamID,例如过通大厅(lobby)得到)。当你有了SteamID之后,你需要执行下面这个方面:
 

[C#] 纯文本查看 复制代码

?

 
1

// class SteamNetworking
public static bool SendP2PPacket(CSteamID steamIDRemote, byte[] pubData, uint cubData, EP2PSend eP2PSendType, int nChannel = 0)

pubData是我们想要发送的数据,cubData是发送数据的字节大小,eP2PSendType是传送的方式,nChannal默认值为空,现在还没有用,不用讨论。
 
这里是如何发送“Hello!”字串的例子:
 

[C#] 纯文本查看 复制代码

?

 
1
2
3
4
5
6
7

CSteamID receiver = ...;
string hello = "Hello!";
 
// allocate new bytes array and copy string characters as bytes
byte[] bytes = new byte[hello.Length * sizeof(char)];
System.Buffer.BlockCopy(hello.ToCharArray(), 0, bytes, 0, bytes.Length);
 
SteamNetworking.SendP2PPacket(receiver, bytes, (uint) bytes.Length, EP2PSend.k_EP2PSendReliable);


这里有四种送传的方式:
 

k_EP2PSendUnreliable – 小包,可以丢失,不需要依次发送,但要快
k_EP2PSendUnreliableNoDelay –     跟上面一样,但是不做链接检查,因为这样,它可能被丢失,但是这种方式是最快的传送方式。
k_EP2PSendReliable – 可靠的信息,大包,依次收发。
k_EP2PSendReliableWithBuffering –     跟上面一样,但是在发送前会缓冲数据,如果你发送大量的小包,它不会那么及时。(可能会延迟200ms)


 
另一边做些什么呢?

如果一个人发送了数据,另一个会以某种方式收到数据。当然,他们都有保密的安全措施。你不会发送数据到你范围之外的其他Steamworks的客户端上。一个Client能接收你的数据之前,他已经和你接受了你的请求,建立起一个P2P会话。

 

P2P会话请求发生在你第一次向SteamWork客户端发送数据时。当你还有发送任何数据时,这个过程会自动重复(通常是几分钟一次),你应该只接受你希望的连接,例如,你所在大厅(lobby)的其他玩家。

 

如何接受一个会话请求?非常简单!你可以像下面这样写代码:

[C#] 纯文本查看 复制代码

?

 
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19

// create a callback field. Having a field will make sure that the callback
// handle won't be eaten by garbage collector.
private Callback<P2PSessionRequest_t> _p2PSessionRequestCallback;
 
void Start()
{
    // setup the callback method
    _p2PSessionRequestCallback = Callback<P2PSessionRequest_t>.Create(OnP2PSessionRequest);
}
 
void OnP2PSessionRequest(P2PSessionRequest_t request)
{
    CSteamID clientId = request.m_steamIDRemote;
    if (ExpectingClient(clientId))
    {
        SteamNetworking.AcceptP2PSessionWithUser(clientId);
    } else {
        Debug.LogWarning("Unexpected session request from " + clientId);
    }
}



这样一个P2P会话就会被接受,你就可以开始……
 
读消息

 

所有消息都存在Steamwork消息对队列中。你准备取它时就可以读它。一般在Update()函数中要处理这些,你的应用可以尽快的检查到是否有新消息。

 

[C#] 纯文本查看 复制代码

?

 
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24

void Update()
{
    uint size;
 
    // repeat while there's a P2P message available
    // will write its size to size variable
    while (SteamNetworking.IsP2PPacketAvailable(out size))
    {
        // allocate buffer and needed variables
        var buffer = new byte[size];
        uint bytesRead;
        CSteamID remoteId;
 
        // read the message into the buffer
        if (SteamNetworking.ReadP2PPacket(buffer, size, out bytesRead, out remoteId))
        {
            // convert to string
            char[] chars = new char[bytesRead / sizeof(char)];
            Buffer.BlockCopy(buffer, 0, chars, 0, length);
 
            string message = new string(chars, 0, chars.Length);
            Debug.Log("Received a message: " + message);
        }
    }
}



 
就是这些!


 
总结

这个指南没有覆盖清理部分(这不是必须的,因为没有用到的会话会被自动清理)和异常处理。你可以在官方文档读到它们 official Steamworks documentation,。记住你需要成为Steam的伙伴才可能获得,如果你还不是,我希望你读完本文之后,可以考虑成为其中一员。

 

原文链接:http://blog.theknightsofunity.com/steamworks-and-unity-p2p-multiplayer/