Unity3D客户端和Java服务端使用Protobuf
转自:http://blog.csdn.net/kakashi8841/article/details/17334493
前几天有位网友问我关于Unity3D里面使用Protobuf的方法,一时有事拖到现在才写这篇文章,不好意思哈。
本文测试环境:
系统:WINDOWS 7(第3、6步)、OS X 10.9(第4步)
软件:VS 2012(第3、6步)、Eclipse(第5、6步)
硬件:iPad 2(第4步)、Macbook Pro Mid 2012(第4步)
文章目录:
1、关于Protobuf的C#实现
2、为什么有些Protobuf发布到iOS就用不了,甚至有些在PC都用不了?
3、手动处理C#版本的Protobuf
3.1、创建一个C#工程,先手动创建每一个要通过Protobuf序列化或反序列化的数据模型类,然后导出dll
3.2、创建一个用于序列化的C#工程,然后运行生成dll
3.3、将上面两个工程生成的dll拖到unity中
4、在Unity中反序列化Protobuf
5、服务端Java也用Protobuf
6、太烦了?!客户端也要自动处理Protobuf
1、关于Protobuf的C#实现
首先,U3D里面Protobuf使用的是C#的实现,那么目前有几个可选的C#实现:
C#: http://code.google.com/p/protobuf-csharp-port
C#: http://code.google.com/p/protosharp/
C#: https://silentorbit.com/protobuf/
C#/.NET/WCF/VB: http://code.google.com/p/protobuf-net/
我这里选用的是http://code.google.com/p/protobuf-net/(你可以在https://code.google.com/p/protobuf-net/downloads/list 这里下载到他的代码和工具),它比较好的一点是,提供了各种平台的支持,解压后在“Full”目录中可以看到各个平台的支持。(现在google被各种封杀,如果你打不开上面的地址,可以下载我上传到CSDN的,里面的csharp文件夹就是各个平台的protobuf需要的dll,点击下载protobuf-net)。
看到里面的unity了吗,它里面的protobuf-net.dll将是我们准备用到的。
2、为什么有些Protobuf发布到iOS就用不了,甚至有些在PC都用不了?
a、Protobuf使用了JIT,即在运行时动态编译,而这个特性在Unity发布到iOS时候是不支持的。因此,会导致你在PC上可以正常运行,发布到iOS就有问题。
b、Protobuf是基于.net 2.0以上框架写的,而Unity仅支持.net 2.0,或者有些使用2.0中比较多的特性,而你在Unity中发布设置了.net 2.0的子集。后者你只需要在Player setting中修改设置就可以了。
上面两项也可适用于其它第三方类库,如果你自己下载了一个在PC上或C#里面能正常使用的类库,在U3D里面就不能用了,那么请检查是否是上面两条原因导致的。
3、手动处理C#版本的Protobuf
知道了上面问题,我们只要选一个.net2.0的Protobuf,然后它又不是JIT,那就可以正常使用了。
这里用的思路是:
3.1、创建一个C#工程,先手动创建每一个要通过Protobuf序列化或反序列化的数据模型类,然后导出dll
以VS为例,首先,创建一个类库工程:“文件”>"新建">"项目">"类库"(记得选择 .net framework 2.0)
将unity的protobuf的dll添加到项目引用
然后假设你有一个类WorkerInfo是需要通过Protobuf进行序列化和反序列化的,那么创建一个WorkerInfo类,内容如下:
- using System;
- using System.Collections.Generic;
- using System.Text;
- using ProtoBuf;
- namespace Com.YourCompany.Project.Proto.Module{
- [ProtoContract]
- public class WorkerInfo {
- [ProtoMember(1)]
- public int workerId;
- [ProtoMember(2)]
- public int leftClosingTimeSec;
- [ProtoMember(3)]
- public int buildingId;
- }
- }
按下Shift+F6生成dll,在项目的bin\Debug目录下就可以找到ProtoModelDLL.dll了
3.2、创建一个用于序列化的C#工程,然后运行生成dll
也是以VS为例,首先创建一个控制台应用程序:“文件”>"新建">"项目">"控制台应用程序"(记得选择 .net framework 2.0)
将Protobuf和3.1生成的dll添加到引用
在项目生成的Program.cs中写入:
- using System;
- using System.Collections.Generic;
- using System.Text;
- using ProtoBuf.Meta;
- using ProtoBuf;
- using ProtoBuf.Compiler;
- using Com.YourCompany.Project.Proto.Module;
- namespace ProtoModelSerializerCreator
- {
- class Program
- {
- static void Main(string[] args)
- {
- var model = TypeModel.Create();
- model.Add(typeof(object), true);
- model.Add(typeof(WorkerInfo), true);
- model.AllowParseableTypes = true;
- model.AutoAddMissingTypes = true;
- model.Compile("ProtoModelSerializer", "ProtoModelSerializer.dll");
- }
- }
- }
然后ctrl+F5运行,这时候你就可以在bin\Debug中看到ProtoModelSerializer.dll。
3.3、将上面两个工程生成的dll(ProtoModelDLL.dll和ProtoModelSerializer.dll)以及protobuf-net.dll拖到unity中
怎么用?看第4步
4、在Unity中反序列化Protobuf
由于一般游戏客户端请求的数据量比较简单,也比较少,因此我们前端请求是不直接二进制请求。而前端收到后端返回才采用了Protobuf,因此。这里只讨论Protobuf的反序列化。
代码很简单,下面写个测试代码:
- using UnityEngine;
- using System.Collections;
- using ProtoBuf.Meta;
- using Com.YourCompany.Project.Proto.Module;
- using System.IO;
- using Com.Duoyu001.Proto.Building;
- using Com.Duoyu001.Proto.Worker;
- public class TestProto : MonoBehaviour
- {
- // init
- void Start ()
- {
- byte[] dataFromServ = new byte[]{8, 233, 7, 16, 100, 24, 1}; //these bytes are generated by server
- RuntimeTypeModel serializer = ProtoModelSerializer.Create ();
- System.IO.MemoryStream memStream = new System.IO.MemoryStream ();
- WorkerInfo w = new WorkerInfo ();
- serializer.Deserialize (memStream, w, w.GetType ()); //asign value to proto model
- Debug.Log (w.workerId + ", " + w.buildingId + ", " + w.leftClosingTimeSec);
- }
- }
运行后Unity控制台输出了worker的信息。代码中的dataFromServ字节数组的内容实际应该是通信时候后端返回的。这里测试就不涉及Socket通信的知识了。
5、服务端Java也用Protobuf
看到一个客户端用Protobuf这么麻烦,那后端会怎样呢?其实后端是比较简单的,有Google官方的支持。
下载:https://code.google.com/p/protobuf/downloads/list
下完解压是这样的(2.5.0):
进入“protobuf-2.5.0\java”文件夹,里面是一个maven项目,你直接用maven clean install在target目录就会生成一个protobuf-java-2.5.0.jar的jar包了,没maven的在这里下载吧,我用maven生成的http://download.csdn.net/detail/kakashi8841/6723689。这个要时候导入你的Java项目。就可以了。
然后你写一个proto文件,调用“protobuf-2.5.0\src”里面的protoc.exe进行生成,它会帮你生成一个java文件。(详细看https://developers.google.com/protocol-buffers/docs/javatutorial),我这里有提供一个bat,用于调用protoc来生成java文件的,手动输入的话太麻烦了。在windows下把它保存到.bat然后双击运行就可以了。
- @echo off
- echo ** setting runtime variable
- REM _protoSrc 是你的proto文件目录的位置
- set _protoSrc=F:\project_proto_src\trunk\xgame-controllers\protos
- REM protoExe 是用于从proto生成java的protoc.exe程序的位置
- set protoExe=C:\Users\john\Desktop\protobuf-2.5.0\src\protoc.exe
- REM java_out_file 存放生成的Java文件目录的位置
- set java_out_file=F:\project_proto_src\trunk\xgame-controllers\src\main\java\
- for /R "%_protoSrc%" %%i in (*) do (
- set filename=%%~nxi
- if "%%~xi" == ".proto" (
- %protoExe% --proto_path=%_protoSrc% --java_out=%java_out_file% %%i
- )
- )
OK啦,这样你只要把生成的Java复制到或直接生成到你的Java项目源码目录中,然后就可以使用了。比如:
以前面说的WorkerInfo为例
- package com.duoyu001.xgame;
- import java.util.Arrays;
- import com.duoyu001.xgame.worker.proto.WorkerInfoBuilder.WorkerInfo;
- public class TestProto {
- public static void main(String[] args) {
- WorkerInfo w = WorkerInfo.newBuilder().setBuildingId(1)
- .setLeftClosingTimeSec(100).setWorkerId(1001).build();
- byte[] byteArray = w.toByteArray();
- System.out.println(Arrays.toString(byteArray));
- }
- }
控制台就会输出:
细心的同学会发现这里的字节和上面的“8, 233, 7, 16, 100, 24, 1”有点不太一样。第二个数字分别为-23和233.
其实,这个只是byte的有无符号的表示差异而已。
他们的二进制表示都是:11101001
6、太烦了?!客户端也要自动处理Protobuf
我们看到客户端没加一个类,都需要在两个VS项目中增加代码,而后端是直接根据proto文件生成代码的。这样,好像有点不公平,而且这样前后端还是可能会写出不一样的结构。其实用protobuf我个人觉得最大的好处:
a、数据量小
b、通过proto模板生成代码,减少前后端联调
但是现在只是后端减少了工作,前端并没有减少,因此第二个好处不是很明显。
那好吧。我没有就这样满足。因此,我决定,前端也要根据proto来生成cs文件。
因此,我使用Java编写了一个解析Proto文件的工具,并且根据proto生成cs文件,然后用bat调用vs的命令行执行vs项目的构建,最后生成两个dll。
我把上面的命令都整合到一个bat中,因此这个bat的任务是:
执行java程序,把proto文件生成cs文件。
调用vs接口构建两个vs项目,生成两个dll。
通过svn把这两个dll提交到客户端的主干上。
调用上面根据proto生成java的bat片段,生成java代码。
通过svn把生成的java代码提交到服务端主干上。
因此,这样一来,现在只要编写好proto文件,然后双击bat,前后端就都可以通过更新svn来获取最新的Protobuf数据对象。
bat运行
根据proto生成cs文件并编译执行两个vs项目,然后把生成的dll提交到svn上
生成java代码并提交svn
这一步相关操作大家有兴趣可以自行试试,有问题欢迎在本博客讨论。
Unity3D客户端和Java服务端使用Protobuf的更多相关文章
- C#使用Thrift简介,C#客户端和Java服务端相互交互
C#使用Thrift简介,C#客户端和Java服务端相互交互 本文主要介绍两部分内容: C#中使用Thrift简介 用Java创建一个服务端,用C#创建一个客户端通过thrift与其交互. 用纯C#实 ...
- RPC学习--C#使用Thrift简介,C#客户端和Java服务端相互交互
本文主要介绍两部分内容: C#中使用Thrift简介 用Java创建一个服务端,用C#创建一个客户端通过thrift与其交互. 用纯C#实现Client和Server C#服务端,Java客户端 其中 ...
- Socket通讯-C#客户端与Java服务端通讯(发送消息和文件)
设计思路 使用websocket通信,客户端采用C#开发界面,服务端使用Java开发,最终实现Java服务端向C#客户端发送消息和文件,C#客户端实现语音广播的功能. Java服务端设计 packag ...
- C++客户端访问Java服务端发布的SOAP模式的WebService接口
gSOAP是一个绑定SOAP/XML到C/C++语言的工具,使用它可以 简单快速地开发出SOAP/XML的服务器端和客户端 Step1 使用gsoap-2.8\gsoap\bin\win32\wsdl ...
- android客户端向java服务端post发送json
android 端: private void HttpPostData() { try { HttpClient httpclient = new DefaultHttpClient( ...
- “快的打车”创始人陈伟星的新项目招人啦,高薪急招Java服务端/Android/Ios 客户端研发工程师/ mysql DBA/ app市场推广专家,欢迎大家加入我们的团队! - V2EX
"快的打车"创始人陈伟星的新项目招人啦,高薪急招Java服务端/Android/Ios 客户端研发工程师/ mysql DBA/ app市场推广专家,欢迎大家加入我们的团队! - ...
- thrift例子:python客户端/java服务端
java服务端的代码请看上文. 1.说明: 这两篇文章其实解决的问题是,当使用python去访问大数据线上集群的时候,遇到两个问题: 1)python-hadoop和python-hive相关包链接不 ...
- java版gRPC实战之六:客户端动态获取服务端地址
欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...
- NIO【同步非阻塞io模型】关于 NIO socket 的详细总结【Java客户端+Java服务端 + 业务层】【可以客户端间发消息】
1.前言 以前使用 websocket来实现双向通信,如今深入了解了 NIO 同步非阻塞io模型 , 优势是 处理效率很高,吞吐量巨大,能很快处理大文件,不仅可以 做 文件io操作, 还可以做sock ...
随机推荐
- 利用WSCF进行契约先行的Web Services开发
http://www.cnblogs.com/goody9807/archive/2007/06/05/772107.html 什么是契约先行(Contract-First)? 如果说一个新的软件开发 ...
- edmx代码分析
http://www.cnblogs.com/FoundationSoft/archive/2011/01/08/1930479.html 本文分析Entity Framework从数据库自动生成的模 ...
- 【原】ajaxupload.js上传报错处理方法
相信大家在工作中经常用到文件上传的操作,因为我是搞前端的,所以这里主要是介绍ajax在前端中的操作.代码我省略的比较多,直接拿js那里的 $.ajaxFileUpload({ url:'www.cod ...
- 申请https证书需要注意的4大问题
HTTPS证书是什么 https证书是数字证书中的一种,由受信任的数字证书颁发机构CA如[沃通CA]在验证服务器身份后颁发,具有服务器身份验证和数据传输加密 功能,因其要配置在服务器上,所以也称SSL ...
- CentOS系统rsync文件同步 安装配置
rsync是类unix系统下的数据镜像备份工具,从软件的命名上就可以看出来了——remote sync 它的特性如下: 可以镜像保存整个目录树和文件系统. 可以很容易做到保持原来文件的权限.时间.软硬 ...
- 《深入理解bootstrap》读书笔记:第4章 CSS组件(上)
bootstrap三大核心之二. 包括glyphicon图标,下拉菜单(dropdown),按钮组(button group)....等等. 一. 图标字体 bootstrap3提供了200多个免费图 ...
- yourphp内置编辑器
{: $Form->editor(array(')),$vo[desc])} form.php 如: <tr> <td>机器故障:</td> <td&g ...
- 路径 dirname(__FILE__)
require(dirname(__FILE__).'/include/init.php');
- Redis学习——链表源码分析
0. 前言 Redis 中的链表是以通用链表的形式实现的,而对于链表的用途来说,主要的功能就是增删改查,所以对于查找来说,redis其提供了一个match函数指针,用户负责实现其具体的匹配操作,从而实 ...
- Python开发【第四篇】:Python基础之函数
三元运算 三元运算(三目运算),是对简单的条件语句的缩写. ? 1 2 3 4 5 # 书写格式 result = 值1 if 条件 else 值2 # 如果条件成立,那么将 “值1” 赋值给 ...