Unity中进程间通信——使用异步Socket
开发Unity项目过程中,即时通信功能来完成服务器与客户端自定义的数据结构封装。
如果要序列化和数据封装参考:Unity3D之C#用Socket传数据包
蓝鸥3G封装的类
客户端脚本ClientScript
/// <summary>
/// Client Script.
/// Created By 蓝鸥3G 2014.08.23
/// </summary> using UnityEngine;
using System.Collections; public class ClientScript: MonoBehaviour {
string msg = "";
// Use this for initialization LOSocket client;
void Start () {
client = LOSocket.GetSocket(LOSocket.LOSocketType.CLIENT);
client.InitClient ("127.0.0.1", 2222, ((string content) => {
//收到服务器的回馈信息
}));
} void OnGUI() {
msg = GUI.TextField(new Rect(0, 0, 500, 40), msg);
if(GUI.Button(new Rect(0, 50, 100, 30), "Send"))
{ client.SendMessage (msg);
}
}
}
服务器端脚本
using UnityEngine;
using System.Collections; public class ServerScript : MonoBehaviour { private string receive_str;
LOSocket server;
// Use this for initialization
void Start ()
{
server = LOSocket.GetSocket(LOSocket.LOSocketType.SERVER);
//初始化服务器
server.InitServer((string content) => {
receive_str = content;
});
} void OnGUI()
{
if (receive_str != null)
{
GUILayout.Label (receive_str);
}
}
}
LOSocket框架
using UnityEngine;
using System.Collections;
using System.Net;
using System.Net.Sockets;
using System.Text; //收到消息后的委托回调
public delegate void ReceiveCallBack(string content); public class LOSocket{ //可以创建的Socket端口类型
public enum LOSocketType
{
CLIENT = 0,
SERVER = 1,
}
#region --------取消构造器
private LOSocket()
{
} #endregion #region --------公共代码
//通过静态方法获取不同的端口类型
public static LOSocket GetSocket(LOSocket.LOSocketType type)
{
LOSocket socket = null; switch (type) {
case LOSocketType.CLIENT:
{
//创建一个新的客户端
socket = new LOSocket ();
break;
}
case LOSocketType.SERVER:
{
//获取服务端
socket = GetServer ();
break;
}
} return socket;
} #endregion
#region --------客户端部分代码
private Socket clientSocket; //声明客户端收到服务端返回消息后的回调委托函数
private ReceiveCallBack clientReceiveCallBack;
//用来存储服务端返回的消息数据
byte[] Buffer = new byte[1024]; //初始化客户端Socket信息
public void InitClient(string ip,int port,ReceiveCallBack ccb)
{
this.clientReceiveCallBack = ccb;
this.clientSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); IPAddress address = IPAddress.Parse (ip);
IPEndPoint ep = new IPEndPoint (address, port); this.clientSocket.Connect(ep);
//开始异步等待接收服务端消息
this.clientSocket.BeginReceive (Buffer, 0, Buffer.Length, SocketFlags.None, new System.AsyncCallback(ReceiveFromServer), this.clientSocket);
} //收到服务端返回消息后的回调函数
void ReceiveFromServer(System.IAsyncResult ar)
{
//获取当前正在工作的Socket对象
Socket worker = ar.AsyncState as Socket;
int ByteRead=0;
try
{
//接收完毕消息后的字节数
ByteRead = worker.EndReceive(ar);
}
catch (System.Exception ex)
{
this.clientReceiveCallBack (ex.ToString ());
}
if (ByteRead > 0)
{
string Content = Encoding.Default.GetString (Buffer);
//通过回调函数将消息返回给调用者
this.clientReceiveCallBack (Content);
}
//继续异步等待接受服务器的返回消息
worker.BeginReceive(Buffer, 0, Buffer.Length, SocketFlags.None, new System.AsyncCallback(ReceiveFromServer), worker);
} //客户端主动发送消息
public void SendMessage(string message)
{
if (message == null)
return; message += "\r\n";
byte[] sendData = Encoding.UTF8.GetBytes (message); //异步发送消息请求
clientSocket.BeginSend (sendData, 0, sendData.Length, SocketFlags.None, new System.AsyncCallback (SendToServer), clientSocket);
}
//发送消息结束的回调函数
void SendToServer(System.IAsyncResult ar)
{
Socket worker = ar.AsyncState as Socket;
worker.EndSend (ar);
} #endregion #region -------服务器部分代码
//服务器端收到消息的存储空间
byte[] ReceiveBuffer = new byte[1024];
//服务器收到消息后的回调委托
private ReceiveCallBack callback; //单例模式
private static LOSocket serverSocket;
private static LOSocket GetServer() {
if (serverSocket == null) {
serverSocket = new LOSocket();
}
return serverSocket;
} //初始化服务器信息
public void InitServer(ReceiveCallBack cb) {
this.callback = cb;
// 1.
Socket server_socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
// 2.
IPEndPoint endPoint = new IPEndPoint(IPAddress.Any, 2222);
// 3.
server_socket.Bind(endPoint);
// 4.
server_socket.Listen(10);
// 5.开始异步等待客户端的请求链接
server_socket.BeginAccept (new System.AsyncCallback (Accept), server_socket); this.callback ("开启服务器" + endPoint.ToString());
} //接受到客户端的链接请求后的回调函数
void Accept(System.IAsyncResult ar){
//获取正在工作的Socket对象
Socket socket = ar.AsyncState as Socket;
//存储异步操作的信息,以及用户自定义的数据
Socket worker = socket.EndAccept(ar); SocketError error; //开始异步接收客户端发送消息内容
worker.BeginReceive (ReceiveBuffer, 0, ReceiveBuffer.Length, SocketFlags.None, new System.AsyncCallback (Receive), worker);
//继续异步等待新的客户端链接请求
socket.BeginAccept(new System.AsyncCallback(Accept), socket); }
//服务端收到客户端的消息后的回调函数
void Receive(System.IAsyncResult ar)
{
//获取正在工作的Socket对象
Socket worker = ar.AsyncState as Socket;
int ByteRead=0;
try
{
ByteRead = worker.EndReceive(ar);
}
catch (System.Exception ex)
{
this.callback (ex.ToString ());
}
if (ByteRead > 0)
{
string Content = Encoding.Default.GetString (ReceiveBuffer);
this.callback (Content);
}
//继续异步等待客户端的发送消息请求
worker.BeginReceive(ReceiveBuffer, 0, ReceiveBuffer.Length, SocketFlags.None, new System.AsyncCallback(Receive), worker);
}
#endregion
}
#python客户端测试代码
#! /usr/env/bin python
#codinf=utf-8
import socket
import time def Client():
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.connect(('', 8000))#链接服务器
while 1:
print 'enter something:',
ent = raw_input()
if ent == '':
break
sock.send(ent)#发送数据给服务器
time.sleep(1)
data = sock.recv(1024)#接收服务器发过来到数据
print 'echo=>',data
sock.close() if __name__ == '__main__':
Client()
python服务器端测试代码
#! /usr/env/bin python
#coding=utf-8
import socket #Server
def Server():
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.bind(('localhost', 8000))
sock.listen(5)#监听,最大链接数
while 1:
connection, address = sock.accept()#开始接受请求,进入等待阻塞状态,直到有链接到达
while 1:
data = connection.recv(1024)#接收客户端发过来的数据
if not data:
break
print data,address
connection.send(data)#发送数据到客户端,即上面到connection
connection.close() if __name__ == '__main__':
Server()
注意,这里没有对服务器的连接关闭,自己可以加上:sock.close()
Unity中进程间通信——使用异步Socket的更多相关文章
- [Unity Socket]在Unity中如何实现异步Socket通信技术
在刚刚开发Unity项目的过程中,需要用到即时通信功能来完成服务器与客户端自定义的数据结构封装. 现在将部分主要功能的实现代码抽取出来实现了可以异步Socket请求的技术Demo. 客户端脚本Clie ...
- Unity中进程间通信——使用Protobuf-net序列化与反序列化
基于ProtoBuf协议实现网络传输(上) Protobuf 全称Protocol Buffers 是一种轻便高效的结构化数据存储格式,可以用于结构化数据串行化,很适合做数据存储或 RPC 数据交换格 ...
- C#使用ProtocolBuffer(ProtoBuf)进行Unity中的Socket通信
首先来说一下本文中例子所要实现的功能: 基于ProtoBuf序列化对象 使用Socket实现时时通信 数据包的编码和解码 下面来看具体的步骤: 一.Unity中使用ProtoBuf 导入DLL到Uni ...
- 《Unity 3D游戏客户端基础框架》多线程异步 Socket 框架构建
引言: 之前写过一个 demo 案例大致讲解了 Socket 通信的过程,并和自建的服务器完成连接和简单的数据通信,详细的内容可以查看 Unity3D -- Socket通信(C#).但是在实际项目应 ...
- Unity3D中简单的C#异步Socket实现
Unity3D中简单的C#异步Socket实现 简单的异步Socket实现..net框架自身提供了很完善的Socket底层.笔者在做Unity3D小东西的时候需要使用到Socket网络通信.于是决定自 ...
- GJM :异步Socket [转载]
原帖地址:http://blog.csdn.net/awinye/article/details/537264 原文作者:Awinye 目录(?)[-] 转载请原作者联系 Overview of So ...
- Play!中使用HTTP异步编程
本章译者:@Sam Liu (译者未留下自己的主页,请Sam Liu见此文,加入群168013302联系‘大黄蜂@翻译play’) 这一章主要讲解如何运用异步模式实现典型的长连接(long-polli ...
- Python简易聊天工具-基于异步Socket通信
继续学习Python中,最近看书<Python基础教程>中的虚拟茶话会项目,觉得很有意思,自己敲了一遍,受益匪浅,同时记录一下. 主要用到异步socket服务客户端和服务器模块asynco ...
- 项目笔记---C#异步Socket示例
概要 在C#领域或者说.net通信领域中有着众多的解决方案,WCF,HttpRequest,WebAPI,Remoting,socket等技术.这些技术都有着自己擅长的领域,或者被合并或者仍然应用于某 ...
随机推荐
- Hasura GraphQL 内部表结构
Hasura 使用pg 数据库存储引擎的元数据信息,在hdb_catalog schema 下面,是在初始化的时候生成的 对于表的管理.权限的信息存储都在这个schema下 hdb_table 这个表 ...
- Go语言编程 (许式伟 等 著)
第1章 初识Go语言 1.1 语言简史 1.2 语言特性 1.2.1 自动垃圾回收 1.2.2 更丰富的内置类型 1.2.3 函数多返回值 1.2.4 错误处理 1.2.5 匿名函数和闭包 1.2.6 ...
- HTTP请求属性说明
1)URL:页面地址. 2)Method :页面的提交方式,POST或GET. 3)EncType:编码类型.此参数给出一个内容类型(Content-Type),指定其做为回放脚本时“Content- ...
- taro 事件处理
https://nervjs.github.io/taro/docs/event.html Taro 元素的事件处理和 DOM 元素的很相似.但是有一点语法上的不同: Taro 事件绑定属性的命名采用 ...
- 关于TP5的一对一、一对多同时存在的关联查询
主表SQL(tp_member) CREATE TABLE `tp_member` ( `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键id', `us ...
- 五、springboot(二)配置数据源oracle
1.添加依赖 <!-- jpa --> <dependency> <groupId>org.springframework.boot</groupId> ...
- AXI_LITE源码学习笔记
AXI_LITE源码学习笔记 1. axi_awready信号的产生 准备接收写地址信号 // Implement axi_awready generation // axi_awready is a ...
- MySQL的瑞士军刀
这里主要讲mysql运维中的一些主要工具,这些工具可能大家都用过,特别是系统管理员或者做linux服务器维护的同学可能都知道这些小工具,这里讲得会比较多一些,除了系统监控的小工具,还包括一些mysql ...
- Azure ARM (15) 根据现有VHD文件,创建ARM VM
<Windows Azure Platform 系列文章目录> 在很多时候,我们需要根据现有VHD文件,创建ARM VM.在这里笔者简单介绍一下相关的Azure PowerShell 这里 ...
- js跨域调用mvc ActionResult扩展
背景 最近2个项目中都用到了js跨域访问的知识,2个项目都需要主站与各个分站之间进行数据交互.状态同步等相关操作.浏览器本身是不允许进行跨域访问,在MVC中我们可以扩展一个方法来实现这个功能.在此大家 ...