- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章C#使用Protocol Buffer(ProtoBuf)进行Unity中的Socket通信由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
首先来说一下本文中例子所要实现的功能:
下面来看具体的步骤:
1、Unity中使用ProtoBuf 。
导入DLL到Unity中, 创建网络传输的模型类:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
using
System;
using
ProtoBuf;
//添加特性,表示可以被ProtoBuf工具序列化
[ProtoContract]
public
class
NetModel {
//添加特性,表示该字段可以被序列化,1可以理解为下标
[ProtoMember(1)]
public
int
ID;
[ProtoMember(2)]
public
string Commit;
[ProtoMember(3)]
public
string Message;
}
using
System;
using
ProtoBuf;
//添加特性,表示可以被ProtoBuf工具序列化
[ProtoContract]
public
class
NetModel {
//添加特性,表示该字段可以被序列化,1可以理解为下标
[ProtoMember(1)]
public
int
ID;
[ProtoMember(2)]
public
string Commit;
[ProtoMember(3)]
public
string Message;
}
|
在Unity中添加测试脚本,介绍ProtoBuf工具的使用.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
|
using
System;
using
System.IO;
public
class
Test : MonoBehaviour {
void
Start () {
//创建对象
NetModel item =
new
NetModel(){ID = 1, Commit =
"LanOu"
, Message =
"Unity"
};
//序列化对象
byte[] temp = Serialize(item);
//ProtoBuf的优势一:小
Debug.Log(temp.Length);
//反序列化为对象
NetModel result = DeSerialize(temp);
Debug.Log(result.Message);
}
// 将消息序列化为二进制的方法
// < param name="model">要序列化的对象< /param>
private
byte[] Serialize(NetModel model)
{
try
{
//涉及格式转换,需要用到流,将二进制序列化到流中
using
(MemoryStream ms =
new
MemoryStream()) {
//使用ProtoBuf工具的序列化方法
ProtoBuf.Serializer.Serialize<NetModel> (ms, model);
//定义二级制数组,保存序列化后的结果
byte[] result =
new
byte[ms.Length];
//将流的位置设为0,起始点
ms.Position = 0;
//将流中的内容读取到二进制数组中
ms.Read (result, 0, result.Length);
return
result;
}
}
catch
(Exception ex) {
Debug.Log (
"序列化失败: "
+ ex.ToString());
return
null;
}
}
// 将收到的消息反序列化成对象
// < returns>The serialize.< /returns>
// < param name="msg">收到的消息.</param>
private
NetModel DeSerialize(byte[] msg)
{
try
{
using
(MemoryStream ms =
new
MemoryStream()) {
//将消息写入流中
ms.Write (msg, 0, msg.Length);
//将流的位置归0
ms.Position = 0;
//使用工具反序列化对象
NetModel result = ProtoBuf.Serializer.Deserialize<NetModel> (ms);
return
result;
}
}
catch
(Exception ex) {
Debug.Log(
"反序列化失败: "
+ ex.ToString());
return
null;
}
}
}
using
System;
using
System.IO;
public
class
Test : MonoBehaviour {
void
Start () {
//创建对象
NetModel item =
new
NetModel(){ID = 1, Commit =
"LanOu"
, Message =
"Unity"
};
//序列化对象
byte[] temp = Serialize(item);
//ProtoBuf的优势一:小
Debug.Log(temp.Length);
//反序列化为对象
NetModel result = DeSerialize(temp);
Debug.Log(result.Message);
}
// 将消息序列化为二进制的方法
// < param name="model">要序列化的对象< /param>
private
byte[] Serialize(NetModel model)
{
try
{
//涉及格式转换,需要用到流,将二进制序列化到流中
using
(MemoryStream ms =
new
MemoryStream()) {
//使用ProtoBuf工具的序列化方法
ProtoBuf.Serializer.Serialize<NetModel> (ms, model);
//定义二级制数组,保存序列化后的结果
byte[] result =
new
byte[ms.Length];
//将流的位置设为0,起始点
ms.Position = 0;
//将流中的内容读取到二进制数组中
ms.Read (result, 0, result.Length);
return
result;
}
}
catch
(Exception ex) {
Debug.Log (
"序列化失败: "
+ ex.ToString());
return
null;
}
}
// 将收到的消息反序列化成对象
// < returns>The serialize.< /returns>
// < param name="msg">收到的消息.</param>
private
NetModel DeSerialize(byte[] msg)
{
try
{
using
(MemoryStream ms =
new
MemoryStream()) {
//将消息写入流中
ms.Write (msg, 0, msg.Length);
//将流的位置归0
ms.Position = 0;
//使用工具反序列化对象
NetModel result = ProtoBuf.Serializer.Deserialize<NetModel> (ms);
return
result;
}
}
catch
(Exception ex) {
Debug.Log(
"反序列化失败: "
+ ex.ToString());
return
null;
}
}
}
|
2、Unity中使用Socket实现时时通信 。
通信应该实现的功能:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
|
using
System;
using
System.Net.Sockets;
// 表示一个客户端
public
class
NetUserToken {
//连接客户端的Socket
public
Socket socket;
//用于存放接收数据
public
byte[] buffer;
public
NetUserToken()
{
buffer =
new
byte[1024];
}
// 接受消息
// < param name="data">Data.< /param>
public
void
Receive(byte[] data)
{
UnityEngine.Debug.Log(
"接收到消息!"
);
}
// 发送消息
//< param name="data">Data.< /param>
public
void
Send(byte[] data)
{
}
}
using
System;
using
System.Net.Sockets;
// 表示一个客户端
public
class
NetUserToken {
//连接客户端的Socket
public
Socket socket;
//用于存放接收数据
public
byte[] buffer;
public
NetUserToken()
{
buffer =
new
byte[1024];
}
// 接受消息
// < param name="data">Data.< /param>
public
void
Receive(byte[] data)
{
UnityEngine.Debug.Log(
"接收到消息!"
);
}
// 发送消息
//< param name="data">Data.< /param>
public
void
Send(byte[] data)
{
}
}
|
。
然后实现我们的服务器代码 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
|
using
System.Collections;
using
System.Collections.Generic;
using
System.Net;
using
System;
using
System.Net.Sockets;
public
class
NetServer{
//单例脚本
public
static
readonly NetServer Instance =
new
NetServer();
//定义tcp服务器
private
Socket server;
private
int
maxClient = 10;
//定义端口
private
int
port = 35353;
//用户池
private
Stack<NetUserToken> pools;
private
NetServer()
{
//初始化socket
server =
new
Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
server.Bind(
new
IPEndPoint(IPAddress.Any, port));
}
//开启服务器
public
void
Start()
{
server.Listen(maxClient);
UnityEngine.Debug.Log(
"Server OK!"
);
//实例化客户端的用户池
pools =
new
Stack<NetUserToken>(maxClient);
for
(
int
i = 0; i < maxClient; i++)
{
NetUserToken usertoken =
new
NetUserToken();
pools.Push(usertoken);
}
//可以异步接受客户端, BeginAccept函数的第一个参数是回调函数,当有客户端连接的时候自动调用
server.BeginAccept (AsyncAccept, null);
}
//回调函数, 有客户端连接的时候会自动调用此方法
private
void
AsyncAccept(IAsyncResult result)
{
try
{
//结束监听,同时获取到客户端
Socket client = server.EndAccept(result);
UnityEngine.Debug.Log(
"有客户端连接"
);
//来了一个客户端
NetUserToken userToken = pools.Pop();
userToken.socket = client;
//客户端连接之后,可以接受客户端消息
BeginReceive(userToken);
//尾递归,再次监听是否还有其他客户端连入
server.BeginAccept(AsyncAccept, null);
}
catch
(Exception ex) {
UnityEngine.Debug.Log(ex.ToString());
}
}
//异步监听消息
private
void
BeginReceive(NetUserToken userToken)
{
try
{
//异步方法
userToken.socket.BeginReceive(userToken.buffer, 0, userToken.buffer.Length, SocketFlags.None,
EndReceive, userToken);
}
catch
(Exception ex) {
UnityEngine.Debug.Log(ex.ToString());
}
}
//监听到消息之后调用的函数
private
void
EndReceive(IAsyncResult result)
{
try
{
//取出客户端
NetUserToken userToken = result.AsyncState as NetUserToken;
//获取消息的长度
int
len = userToken.socket.EndReceive(result);
if
(len > 0)
{
byte[] data =
new
byte[len];
Buffer.BlockCopy(userToken.buffer, 0, data, 0, len);
//用户接受消息
userToken.Receive(data);
//尾递归,再次监听客户端消息
BeginReceive(userToken);
}
}
catch
(Exception ex) {
UnityEngine.Debug.Log(ex.ToString());
}
}
}
using
System.Collections;
using
System.Collections.Generic;
using
System.Net;
using
System;
using
System.Net.Sockets;
public
class
NetServer{
//单例脚本
public
static
readonly NetServer Instance =
new
NetServer();
//定义tcp服务器
private
Socket server;
private
int
maxClient = 10;
//定义端口
private
int
port = 35353;
//用户池
private
Stack<NetUserToken> pools;
private
NetServer()
{
//初始化socket
server =
new
Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
server.Bind(
new
IPEndPoint(IPAddress.Any, port));
}
//开启服务器
public
void
Start()
{
server.Listen(maxClient);
UnityEngine.Debug.Log(
"Server OK!"
);
//实例化客户端的用户池
pools =
new
Stack<NetUserToken>(maxClient);
for
(
int
i = 0; i < maxClient; i++)
{
NetUserToken usertoken =
new
NetUserToken();
pools.Push(usertoken);
}
//可以异步接受客户端, BeginAccept函数的第一个参数是回调函数,当有客户端连接的时候自动调用
server.BeginAccept (AsyncAccept, null);
}
//回调函数, 有客户端连接的时候会自动调用此方法
private
void
AsyncAccept(IAsyncResult result)
{
try
{
//结束监听,同时获取到客户端
Socket client = server.EndAccept(result);
UnityEngine.Debug.Log(
"有客户端连接"
);
//来了一个客户端
NetUserToken userToken = pools.Pop();
userToken.socket = client;
//客户端连接之后,可以接受客户端消息
BeginReceive(userToken);
//尾递归,再次监听是否还有其他客户端连入
server.BeginAccept(AsyncAccept, null);
}
catch
(Exception ex) {
UnityEngine.Debug.Log(ex.ToString());
}
}
//异步监听消息
private
void
BeginReceive(NetUserToken userToken)
{
try
{
//异步方法
userToken.socket.BeginReceive(userToken.buffer, 0, userToken.buffer.Length, SocketFlags.None,
EndReceive, userToken);
}
catch
(Exception ex) {
UnityEngine.Debug.Log(ex.ToString());
}
}
//监听到消息之后调用的函数
private
void
EndReceive(IAsyncResult result)
{
try
{
//取出客户端
NetUserToken userToken = result.AsyncState as NetUserToken;
//获取消息的长度
int
len = userToken.socket.EndReceive(result);
if
(len > 0)
{
byte[] data =
new
byte[len];
Buffer.BlockCopy(userToken.buffer, 0, data, 0, len);
//用户接受消息
userToken.Receive(data);
//尾递归,再次监听客户端消息
BeginReceive(userToken);
}
}
catch
(Exception ex) {
UnityEngine.Debug.Log(ex.ToString());
}
}
}
|
。
在Unity中开启服务器,并使用C#控制台模拟客户端连接、发送消息操作。测试OK了,Unity中可以时时监听到消息.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
|
using
UnityEngine;
using
System.Collections;
public
class
CreateServer : MonoBehaviour {
void
StartServer () {
NetServer.Instance.Start();
}
}
//C#控制台工程
using
System;
using
System.Net;
using
System.Net.Sockets;
using
System.IO;
using
System.Text;
namespace
Temp
{
class
MainClass
{
public
static
void
Main (string[] args)
{
TcpClient tc =
new
TcpClient();
IPEndPoint ip =
new
IPEndPoint(IPAddress.Parse(
"127.0.0.1"
), 35353);
tc.Connect(ip);
if
(tc.Connected)
{
while
(
true
)
{
string msg = Console.ReadLine();
byte[] result = Encoding.UTF8.GetBytes(msg);
tc.GetStream().Write(result, 0, result.Length);
}
}
Console.ReadLine();
}
}
}
using
UnityEngine;
using
System.Collections;
public
class
CreateServer : MonoBehaviour {
void
StartServer () {
NetServer.Instance.Start();
}
}
//C#控制台工程
using
System;
using
System.Net;
using
System.Net.Sockets;
using
System.IO;
using
System.Text;
namespace
Temp
{
class
MainClass
{
public
static
void
Main (string[] args)
{
TcpClient tc =
new
TcpClient();
IPEndPoint ip =
new
IPEndPoint(IPAddress.Parse(
"127.0.0.1"
), 35353);
tc.Connect(ip);
if
(tc.Connected)
{
while
(
true
)
{
string msg = Console.ReadLine();
byte[] result = Encoding.UTF8.GetBytes(msg);
tc.GetStream().Write(result, 0, result.Length);
}
}
Console.ReadLine();
}
}
}
|
3、数据包的编码和解码 。
首先,举个例子,这个月信用卡被媳妇刷爆了,面对房贷车贷的压力,我只能选择分期付款。。.
那么OK了,现在我想问一下,当服务器向客户端发送的数据过大时怎么办呢?
当服务器需要向客户端发送一条很长的数据,也会“分期付款!”,服务器会把一条很长的数据分成若干条小数据,多次发送给客户端.
可是,这样就又有另外一个问题,客户端接受到多条数据之后如何解析?
这里其实就是客户端的解码。server发数据一般采用“长度+内容”的格式,Client接收到数据之后,先提取出长度来,然后根据长度判断内容是否发送完毕.
再次重申,用户在发送序列化好的消息的前,需要先编码后再发送消息;用户在接受消息后,需要解码之后再解析数据(反序列化).
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
|
using
UnityEngine;
using
System.Collections.Generic;
using
System.IO;
// 编码和解码
public
class
NetEncode {
// 将数据编码 长度+内容
/// < param name="data">内容< /param>
public
static
byte[] Encode(byte[] data)
{
//整形占四个字节,所以声明一个+4的数组
byte[] result =
new
byte[data.Length + 4];
//使用流将编码写二进制
MemoryStream ms =
new
MemoryStream();
BinaryWriter br =
new
BinaryWriter(ms);
br.Write(data.Length);
br.Write(data);
//将流中的内容复制到数组中
System.Buffer.BlockCopy(ms.ToArray(), 0, result, 0, (
int
)ms.Length);
br.Close();
ms.Close();
return
result;
}
// 将数据解码
// < param name="cache">消息队列< /param>
public
static
byte[] Decode(ref List<byte> cache)
{
//首先要获取长度,整形4个字节,如果字节数不足4个字节
if
(cache.Count < 4)
{
return
null;
}
//读取数据
MemoryStream ms =
new
MemoryStream(cache.ToArray());
BinaryReader br =
new
BinaryReader(ms);
int
len = br.ReadInt32();
//根据长度,判断内容是否传递完毕
if
(len > ms.Length - ms.Position)
{
return
null;
}
//获取数据
byte[] result = br.ReadBytes(len);
//清空消息池
cache.Clear();
//讲剩余没处理的消息存入消息池
cache.AddRange(br.ReadBytes((
int
)ms.Length - (
int
)ms.Position));
return
result;
}
}
using
UnityEngine;
using
System.Collections.Generic;
using
System.IO;
// 编码和解码
public
class
NetEncode {
// 将数据编码 长度+内容
/// < param name="data">内容< /param>
public
static
byte[] Encode(byte[] data)
{
//整形占四个字节,所以声明一个+4的数组
byte[] result =
new
byte[data.Length + 4];
//使用流将编码写二进制
MemoryStream ms =
new
MemoryStream();
BinaryWriter br =
new
BinaryWriter(ms);
br.Write(data.Length);
br.Write(data);
//将流中的内容复制到数组中
System.Buffer.BlockCopy(ms.ToArray(), 0, result, 0, (
int
)ms.Length);
br.Close();
ms.Close();
return
result;
}
// 将数据解码
// < param name="cache">消息队列< /param>
public
static
byte[] Decode(ref List<byte> cache)
{
//首先要获取长度,整形4个字节,如果字节数不足4个字节
if
(cache.Count < 4)
{
return
null;
}
//读取数据
MemoryStream ms =
new
MemoryStream(cache.ToArray());
BinaryReader br =
new
BinaryReader(ms);
int
len = br.ReadInt32();
//根据长度,判断内容是否传递完毕
if
(len > ms.Length - ms.Position)
{
return
null;
}
//获取数据
byte[] result = br.ReadBytes(len);
//清空消息池
cache.Clear();
//讲剩余没处理的消息存入消息池
cache.AddRange(br.ReadBytes((
int
)ms.Length - (
int
)ms.Position));
return
result;
}
}
|
用户接受数据代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
|
using
System;
using
System.Collections.Generic;
using
System.Net.Sockets;
// 表示一个客户端
public
class
NetUserToken {
//连接客户端的Socket
public
Socket socket;
//用于存放接收数据
public
byte[] buffer;
//每次接受和发送数据的大小
private
const
int
size = 1024;
//接收数据池
private
List<byte> receiveCache;
private
bool
isReceiving;
//发送数据池
private
Queue<byte[]> sendCache;
private
bool
isSending;
//接收到消息之后的回调
public
Action<NetModel> receiveCallBack;
public
NetUserToken()
{
buffer =
new
byte[size];
receiveCache =
new
List<byte>();
sendCache =
new
Queue<byte[]>();
}
// 服务器接受客户端发送的消息
// < param name="data">Data.< /param>
public
void
Receive(byte[] data)
{
UnityEngine.Debug.Log(
"接收到数据"
);
//将接收到的数据放入数据池中
receiveCache.AddRange(data);
//如果没在读数据
if
(!isReceiving)
{
isReceiving =
true
;
ReadData();
}
}
// 读取数据
private
void
ReadData()
{
byte[] data = NetEncode.Decode(ref receiveCache);
//说明数据保存成功
if
(data != null)
{
NetModel item = NetSerilizer.DeSerialize(data);
UnityEngine.Debug.Log(item.Message);
if
(receiveCallBack != null)
{
receiveCallBack(item);
}
//尾递归,继续读取数据
ReadData();
}
else
{
isReceiving =
false
;
}
}
// 服务器发送消息给客户端
public
void
Send()
{
try
{
if
(sendCache.Count == 0) {
isSending =
false
;
return
;
}
byte[] data = sendCache.Dequeue ();
int
count = data.Length / size;
int
len = size;
for
(
int
i = 0; i < count + 1; i++) {
if
(i == count) {
len = data.Length - i * size;
}
socket.Send (data, i * size, len, SocketFlags.None);
}
UnityEngine.Debug.Log(
"发送成功!"
);
Send ();
}
catch
(Exception ex) {
UnityEngine.Debug.Log(ex.ToString());
}
}
public
void
WriteSendDate(byte[] data){
sendCache.Enqueue(data);
if
(!isSending)
{
isSending =
true
;
Send();
}
}
}
using
System;
using
System.Collections.Generic;
using
System.Net.Sockets;
// 表示一个客户端
public
class
NetUserToken {
//连接客户端的Socket
public
Socket socket;
//用于存放接收数据
public
byte[] buffer;
//每次接受和发送数据的大小
private
const
int
size = 1024;
//接收数据池
private
List<byte> receiveCache;
private
bool
isReceiving;
//发送数据池
private
Queue<byte[]> sendCache;
private
bool
isSending;
//接收到消息之后的回调
public
Action<NetModel> receiveCallBack;
public
NetUserToken()
{
buffer =
new
byte[size];
receiveCache =
new
List<byte>();
sendCache =
new
Queue<byte[]>();
}
// 服务器接受客户端发送的消息
// < param name="data">Data.< /param>
public
void
Receive(byte[] data)
{
UnityEngine.Debug.Log(
"接收到数据"
);
//将接收到的数据放入数据池中
receiveCache.AddRange(data);
//如果没在读数据
if
(!isReceiving)
{
isReceiving =
true
;
ReadData();
}
}
// 读取数据
private
void
ReadData()
{
byte[] data = NetEncode.Decode(ref receiveCache);
//说明数据保存成功
if
(data != null)
{
NetModel item = NetSerilizer.DeSerialize(data);
UnityEngine.Debug.Log(item.Message);
if
(receiveCallBack != null)
{
receiveCallBack(item);
}
//尾递归,继续读取数据
ReadData();
}
else
{
isReceiving =
false
;
}
}
// 服务器发送消息给客户端
public
void
Send()
{
try
{
if
(sendCache.Count == 0) {
isSending =
false
;
return
;
}
byte[] data = sendCache.Dequeue ();
int
count = data.Length / size;
int
len = size;
for
(
int
i = 0; i < count + 1; i++) {
if
(i == count) {
len = data.Length - i * size;
}
socket.Send (data, i * size, len, SocketFlags.None);
}
UnityEngine.Debug.Log(
"发送成功!"
);
Send ();
}
catch
(Exception ex) {
UnityEngine.Debug.Log(ex.ToString());
}
}
public
void
WriteSendDate(byte[] data){
sendCache.Enqueue(data);
if
(!isSending)
{
isSending =
true
;
Send();
}
}
}
|
ProtoBuf网络传输到这里就全部完成了.
最后此篇关于C#使用Protocol Buffer(ProtoBuf)进行Unity中的Socket通信的文章就讲到这里了,如果你想了解更多关于C#使用Protocol Buffer(ProtoBuf)进行Unity中的Socket通信的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我期望 new Buffer(buffer.toString()) 始终是逐字节相等的。但是,我遇到的情况并非如此。 首先,这是一个真实的案例: var buf1 = new Buffer(32);
我有用于记录数据的 Protocol Buffer 。 message Message { required double val1 = 1; optional int val2 =
请注意以下简单程序(基于 protobuf-net 项目 v1 wiki 中的示例): using System.Collections.Generic; using System.Diagnosti
在 Protocol Buffer 中,有没有办法让消息包含嵌套消息的集合?例如,消息主管可能有一个员工集合以及主管的姓名和部门。 最佳答案 是的。您使用 repeated领域; message Em
我想知道 Protocol Buffer 在解析流时如何处理损坏的数据。有没有办法知道数据是否已损坏。 Protocol Buffer 是否提供任何内置的数据完整性检查机制? 谢谢, 最佳答案 没有任
Protocol Buffer 如何处理类型版本控制? 例如,当我需要随时间更改类型定义时?就像添加和删除字段一样。 最佳答案 Google 设计的 protobuf 对版本控制非常宽容: 意外数据要
我尝试阅读 Protobuf 文档,但无法想象它可以用于许多用例。我想知道一些实际的 Protocol Buffer 性能改进用例。 谢谢 最佳答案 Protocol buffers 是一个序列化库,
给定 Protocol Buffer 模式和一些数据, Protocol Buffer 序列化是否跨库和语言具有确定性? 基本上,无论使用什么库,我是否可以保证相同的数据总是以相同的方式(直到字节)序
我正在使用一个示例 UWP C++/CX 程序,该程序创建两个 UDP 网络通信线程,它们使用 Windows::Storage::Streams::DataWriter 相互发送数据。和 Windo
我正在使用以下代码 int lenSend = odl->ByteSize(); char* buf = (char *)malloc(lenSend); odl->SerializeToArray(
Protocol Buffer 文档警告说...... You should never add behaviour to the generated classes by inheriting fr
我有一个定义如下的原型(prototype)模式, message User { int64 id = 1; bool email_subscribed = 2; bool sms_
我试图了解 Protocol Buffer 编码方法,将消息转换为二进制(或十六进制)格式时,我无法理解嵌入消息的编码方式。 我猜可能和内存地址有关,但我找不到准确的关系。 这是我所做的。 第 1 步
我需要序列化和反序列化一系列与字节流之间的 Protocol Buffer 消息。有一些预先确定的消息类型。编码类型信息的推荐方法是什么,以便我的应用程序可以知道它应该读取哪种类型? 最佳答案 最常见
与GSON相比, Protocol Buffer (protobuf)的优缺点是什么? 在什么情况下,protobuf比GSON更合适? 对于一个非常笼统的问题,我感到抱歉。 最佳答案 json(通过
message Person { required Empid = 1 [default = 100]; required string name = 2 [default = "Raju"]
我正在研究一个小型设备,该设备具有相当大的一组配置参数(~100 KB),这些参数是从 PC 软件生成的。过去,我们将参数存储在二进制文件中并将它们加载到数据结构中。维护有点烦人(不同的语言,确保结构
来自Encoding - Protocol Buffers - Google Code上的“签名类型”: ZigZag encoding maps signed integers to unsigne
我正在使用 Protocol Buffer ,一切正常。除了我不明白的事实 - 为什么我需要 proto 中的编号标签文件 : message SearchRequest { required s
Protocol Buffer 的吸引人的功能之一是它允许您扩展消息定义,而不会破坏使用较旧定义的代码。对于枚举according to the documentation: a field with
我是一名优秀的程序员,十分优秀!