之前我们讨论过“如何把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
void
Start()
{
// setup the callback method
_p2PSessionRequestCallback = Callback
}
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/