android与PC,C#与Java 利用protobuf 进行无障碍通讯【Socket】

protobuf 是什么?

Protocol buffers是一种编码方法构造的一种有效而可扩展的格式的数据。 谷歌使用其内部几乎RPC协议和文件格式的所有协议缓冲区。

参考文档

http://code.google.com/intl/zh-CN/apis/protocolbuffers/docs/overview.html

API的参考文档

protobuf 适用的语言

正宗(Google 自己内部用的)的protobuf支持三种语言:Java 、c++和Pyton,很遗憾的是并不支持.Net 或者 Lua 等语言,但社区的力量是不容忽视的,由于protobuf确实比Json、XML有速度上的优势和使用的方便,并且可以做到向前兼容、向后兼容等众多特点,所以protobuf社区又弄了个protobuf.net的组件并且还支持众多语言,详细可以看这个链接:http://code.google.com/p/protobuf/wiki/ThirdPartyAddOns,具体某种语言的使用请各自对号入座,本篇只是讲使用android 与c++服务器通讯(测试过)或者与PC 通讯,使用java与C#之间互相通讯方面的DEMO,方面读者做参考。

使用protobuf协议

定义protobuf协议

定义protobuf协议必须创建一个以.proto为后缀的文件,以本篇为例,本篇创建了一个叫msg.proto的消息文件,内容如下:

packagemsginfo;

messageCMsg
{
required
string msghead = 1 ;
required
string msgbody = 2 ;
}

messageCMsgHead
{
requiredint32msglen
= 1 ;
requiredint32msgtype
= 2 ;
requiredint32msgseq
= 3 ;
requiredint32termversion
= 4 ;
requiredint32msgres
= 5 ;
required
string termid = 6 ;
}

messageCMsgReg
{
optionalint32area
= 1 ;
optionalint32region
= 2 ;
optionalint32shop
= 3 ;
optionalint32ret
= 4 ;
optional
string termid = 5 [defalut = " 12345 " ];
}

messageCMsgLogin
{
optionalint32ret
= 1 ;
}

messageCMsgLogout
{
optionalint32ret
= 1 ;

}

package在Java里面代表这个文件所在的包名,在c#里面代表该文件的命名空间,message代表一个类,

required 代表该字段必填, optional 代表该字段可选,并可以为其设置默认值,默认值格式 :[ defalut =字符串就是"123" ,整型就是 123]。

如何编译该proto文件

java或android 使用的编译方法

正宗的proto可以在Linux下编译也有提供win版编译,由于Linux下编译要配置什么g++呀,之类的有点麻烦,之前做的步骤都忘得差不多,那还是回到win版编译吧,而net 版则是需要在win版下编译。

正宗google 的protobuf 下载列表请参照:http://code.google.com/p/protobuf/downloads/list ,选择其中的win版本下载。解压后会得到一个protoc.exe 文件,此时就可以开始编译了,先以java 为例,编译的步骤如下:

  • cmd 打开命令工具
  • 以我电脑为例,该exe 文件我放在F:\protoc目录下,先cd 到该目录 cd F:\protoc

  • 再次进入目录后会发现该目录多了一个文件夹,即以该proto的package命名的的目录,会产生一个Msg.java的文件,这时这个文件就可以使用到我们的java或者 android 工程了。
  • 最后一步下载一个protobuf-java-2.3.0.jar的jar 包引用到你的java和android工程 里面,OK。可以使用你的protobuf了。如下图:
c#或者以后的Windows Phone 7 使用的编译方法:

.net 版的protobuf来源于proto社区,有两个版本。一个版本叫protobuf-net,官方站点:http://code.google.com/p/protobuf-net/ 写法上比较符合c#一贯的写法。另一个版本叫protobuf-csharp-sport ,

官方站点:http://code.google.com/p/protobuf-csharp-port/写法上跟java上的使用极其相似,比较遵循Google 的原生态写法,所以做跨平台还是选择第二版本吧。因为你会发现几乎和java的写法没啥两样,本篇也是使用这个版本。

进入该站点,下载你要的win版。编译步骤如下:

  • 将刚才你的proto文件放在你解压出来的目录与protoc.exe、ProtoGen.exe、ProtoGen.exe.config放于一起。其他文件可以删除或者 备份。
  • 还是打开命令行,定位于对应的目录里面,你放proto文件的目录里面。
  • 输入:protoc --descriptor_set_out=msg.protobin --include_imports msg.proto
  • msg.protobin是要生成的prtobobin文件,可以使用这个bin文件生成cs文件
  • 再输入protogen msg.protobin使用该bin文件生成cs文件,这样你就可以得到该 msg.cs 的CSharp版文件了,同时在VS里面使用要引入Google.ProtocolBuffers.dll。为了方便你可以将其做成一个批处理文件代码如下: echoon
    protoc
    -- descriptor_set_out = msg.protobin -- include_importsmsg.proto
    protogenmsg.protobin

    将其另存为.bat文件即可

使用protobuf编译后的文件来进行socket连接

android 与PC

android 做为客户端向PC的Java服务端发送数据,服务端得到数据进行解析,并打印出来。

客户端代码:

packagenet.testSocket;

importjava.io.IOException;
importjava.io.InputStream;
importjava.net.Socket;
importjava.net.UnknownHostException;

importsocket.exception.SmsClientException;
importsocket.exception.SmsObjException;

importmsginfo.Msg.CMsg;
importmsginfo.Msg.CMsgHead;
importmsginfo.Msg.CMsgReg;
importandroid.app.Activity;
importandroid.os.Bundle;
importandroid.view.View;
importandroid.widget.Button;
importandroid.widget.TextView;

importcom.google.protobuf.InvalidProtocolBufferException;

// 客户端的实现
public class TestSocketextendsActivity{
private TextViewtext1;
private Buttonbut1;
Socketsocket
= null ;

public void onCreate(BundlesavedInstanceState){
super.onCreate(savedInstanceState);

// ThreaddesktopServerThread=newThread(newAndroidServer());
// desktopServerThread.start();

setContentView(R.layout.main);

text1
= (TextView)findViewById(R.id.text1);
but1
= (Button)findViewById(R.id.but1);

but1.setOnClickListener(
new Button.OnClickListener(){
@Override
public void onClick(Viewv){

// edit1.setText("");
// Log.e("dddd","sentid");
// newThread(){
// publicvoidrun(){
try {
// socket=newSocket("192.168.1.102",54321);
// socket=newSocket("192.168.1.110",10527);
socket = new Socket( " 192.168.1.116 " , 12345 );
// 得到发送消息的对象
// SmsObjsmsobj=newSmsObj(socket);

// 设置消息头和消息体并存入消息里面
// head
CMsgHeadhead = CMsgHead.newBuilder().setMsglen( 5 )
.setMsgtype(
1 ).setMsgseq( 3 ).setTermversion( 41 )
.setMsgres(
5 ).setTermid( " 11111111 " ).build();

// body
CMsgRegbody = CMsgReg.newBuilder().setArea( 22 )
.setRegion(
33 ).setShop( 44 ).build();

// Msg
CMsgmsg = CMsg.newBuilder()
.setMsghead(head.toByteString().toStringUtf8())
.setMsgbody(body.toByteString().toStringUtf8())
.build();

// PrintWriterout=newPrintWriter(newBufferedWriter(
// newOutputStreamWriter(socket.getOutputStream())),
// true);
// out.println(m.toString());
// out.println(m.toByteString().toStringUtf8());

// 向服务器发送信息
msg.writeTo(socket.getOutputStream());
// byte[]b=msg.toByteArray();
// smsobj.sendMsg(b);

// System.out.println("====msg==="
// +m.toByteString().toStringUtf8());

// byte[]backBytes=smsobj.recvMsg();
//
// 接受服务器的信息
InputStreaminput = socket.getInputStream();

// DataInputStreamdataInput=newDataInputStream();
// byte[]by=smsobj.recvMsg(input);
byte []by = recvMsg(input);
setText(CMsg.parseFrom(by));

// BufferedReaderbr=newBufferedReader(
// newInputStreamReader(socket.getInputStream()));
// Stringmstr=br.readLine();
// if(!str.equals("")){
// text1.setText(str);
// }else{
// text1.setText("数据错误");
// }
// out.close();
// br.close();

input.close();
// smsobj.close();
socket.close();
}
catch (UnknownHostExceptione){
e.printStackTrace();
}
catch (IOExceptione){
e.printStackTrace();
}
catch (Exceptione){
System.
out .println(e.toString());
}
// };
// }.start();

}
});

}

/* *
*接收server的信息
*
*@return
*@throwsSmsClientException
*@authorfisher
*/
public byte []recvMsg(InputStreaminpustream)throwsSmsObjException{
try {

byte len[] = new byte [ 1024 ];
int count = inpustream.read(len);

byte []temp = new byte [count];
for ( int i = 0 ;i < count;i ++ ){
temp[i]
= len[i];
}
return temp;
}
catch (ExceptionlocalException){
throw new SmsObjException( " SmapObj.recvMsg()occurexception! "
+ localException.toString());
}
}

/* *
*得到返回值添加到文本里面
*
*@paramg
*@throwsInvalidProtocolBufferException
*/
public void setText(CMsgg)throwsInvalidProtocolBufferException{
CMsgHeadh
= CMsgHead.parseFrom(g.getMsghead().getBytes());
StringBuffersb
= new StringBuffer();
if (h.hasMsglen())
sb.append(
" ==len=== " + h.getMsglen() + " \n " );
if (h.hasMsgres())
sb.append(
" ==res=== " + h.getMsgres() + " \n " );
if (h.hasMsgseq())
sb.append(
" ==seq=== " + h.getMsgseq() + " \n " );
if (h.hasMsgtype())
sb.append(
" ==type=== " + h.getMsgtype() + " \n " );
if (h.hasTermid())
sb.append(
" ==Termid=== " + h.getTermid() + " \n " );
if (h.hasTermversion())
sb.append(
" ==Termversion=== " + h.getTermversion() + " \n " );

CMsgRegbo
= CMsgReg.parseFrom(g.getMsgbody().getBytes());
if (bo.hasArea())
sb.append(
" ==area== " + bo.getArea() + " \n " );
if (bo.hasRegion())
sb.append(
" ==Region== " + bo.getRegion() + " \n " );
if (bo.hasShop())
sb.append(
" ==shop== " + bo.getShop() + " \n " );
if (bo.hasRet())
sb.append(
" ==Ret== " + bo.getRet() + " \n " );
if (bo.hasTermid())
sb.append(
" ==Termid== " + bo.getTermid() + " \n " );

text1.setText(sb.toString());
}

}

服务端代码:

packageserver;


importjava.io.DataInputStream;
importjava.io.DataOutputStream;
importjava.io.IOException;
importjava.io.InputStream;
importjava.net.ServerSocket;
importjava.net.Socket;

importmsginfo.Msg.CMsg;
importmsginfo.Msg.CMsgHead;
importmsginfo.Msg.CMsgReg;

public class AndroidServerimplementsRunnable{

public void run(){
try {
System.
out .println( " beign: " );
ServerSocketserverSocket
= new ServerSocket( 12345 );
while ( true ){
System.
out .println( " 等待接收用户连接: " );
// 接受客户端请求
Socketclient = serverSocket.accept();

DataOutputStreamdataOutputStream;
DataInputStreamdataInputStream;

try {
// 接受客户端信息
// BufferedReaderin=newBufferedReader(
// newInputStreamReader(client.getInputStream()));
// Stringstr=in.readLine();
// System.out.println("readlength:"+str.length());
// System.out.println("read:"+str);

// InputStreaminputstream=client.getInputStream();
// byte[]buffer=newbyte[1024*4];
// inttemp=0;
// while((temp=inputstream.read(buffer))!=-1){
// str=newString(buffer,0,temp);
// System.out.println("===str==="+str);

// Filefile=newFile("user\\log\\login.log");
// appendLog(file,str);

InputStreaminputstream
= client.getInputStream();

dataOutputStream
= new DataOutputStream(
client.getOutputStream());
// dataInputStream=newDataInputStream(inputstream);

// byte[]d=newBufferedReader(newInputStreamReader(
// dataInputStream)).readLine().getBytes();
// byte[]bufHeader=newbyte[4];
// dataInputStream.readFully(bufHeader);
// intlen=BytesUtil.Bytes4ToInt(bufHeader);
// System.out.println(d.length);
// System.out.println(dataInputStream.readLine().toString());
byte len[] = new byte [ 1024 ];
int count = inputstream.read(len);

byte []temp = new byte [count];

for ( int i = 0 ;i < count;i ++ ){

temp[i]
= len[i];
}

// 协议正文
// byte[]sendByte=newbyte[30];
//
// dataInputStream.readFully(sendByte);
// for(byteb:sendByte){
// System.out.println(""+b);
// }
CMsgmsg = CMsg.parseFrom(temp);
//
//
CMsgHeadhead = CMsgHead.parseFrom(msg.getMsghead()
.getBytes());
System.
out .println( " ==len=== " + head.getMsglen());
System.
out .println( " ==res=== " + head.getMsgres());
System.
out .println( " ==seq=== " + head.getMsgseq());
System.
out .println( " ==type=== " + head.getMsgtype());
System.
out .println( " ==Termid=== " + head.getTermid());
System.
out .println( " ==Termversion=== "
+ head.getTermversion());

CMsgRegbody
= CMsgReg.parseFrom(msg.getMsgbody()
.getBytes());
System.
out .println( " ==area== " + body.getArea());
System.
out .println( " ==Region== " + body.getRegion());
System.
out .println( " ==shop== " + body.getShop());

// PrintWriterout=newPrintWriter(newBufferedWriter(
// newOutputStreamWriter(client.getOutputStream())),
// true);
// out.println("return"+msg.toString());

// in.close();
// out.close();

sendProtoBufBack(dataOutputStream);

inputstream.close();
// dataInputStream.close();
} catch (Exceptionex){
System.
out .println(ex.getMessage());
ex.printStackTrace();
}
finally {
client.close();
System.
out .println( " close " );
}
}
}
catch (IOExceptione){
System.
out .println(e.getMessage());
}
}

public static void main(String[]args){
ThreaddesktopServerThread
= new Thread( new AndroidServer());
desktopServerThread.start();
}

private byte []getProtoBufBack(){

// head
CMsgHeadhead = CMsgHead.newBuilder().setMsglen( 5 )
.setMsgtype(
1 ).setMsgseq( 3 ).setTermversion( 41 )
.setMsgres(
5 ).setTermid( " 11111111 " ).build();

// body
CMsgRegbody = CMsgReg.newBuilder().setArea( 22 )
.setRegion(
33 ).setShop( 44 ).build();

// Msg
CMsgmsg = CMsg.newBuilder()
.setMsghead(head.toByteString().toStringUtf8())
.setMsgbody(body.toByteString().toStringUtf8())
.build();

return msg.toByteArray();
}

private void sendProtoBufBack(DataOutputStreamdataOutputStream){

byte []backBytes = getProtoBufBack();
// 协议头部
// Integerlen2=backBytes.length;
// 前四个字节,标示协议正文长度
// byte[]cmdHead2=BytesUtil.IntToBytes4(len2);

try {
// dataOutputStream.write(cmdHead2,0,cmdHead2.length);
dataOutputStream.write(backBytes, 0 ,backBytes.length);
dataOutputStream.flush();
}
catch (IOExceptione){
e.printStackTrace();
}
}

}

最后得到的效果:

客户端:

服务端:

protobuf .net版的实现代码如下:

using System;
using System.IO;
using System.Net;
using System.Net.Sockets;
using System.Threading;
using Google.ProtocolBuffers;
using msginfo;
using System.Text;
using System.Collections;
using System.Collections.Generic;

namespace protobuf_csharp_sport
{
class Program
{
private static ManualResetEventallDone = new ManualResetEvent( false );

static void Main( string []args)
{
beginProtocbuf();
}

private static void beginProtocbuf()
{
// 启动服务端
TcpListenerserver = new TcpListener(IPAddress.Parse( " 127.0.0.1 " ), 12345 );
server.Start();
server.BeginAcceptTcpClient(clientConnected,server);
Console.WriteLine(
" SERVER:等待数据--- " );

// 启动客户端
ThreadPool.QueueUserWorkItem(runClient);
allDone.WaitOne();

Console.WriteLine(
" SERVER:退出--- " );
// server.Stop();
}

// 服务端处理
private static void clientConnected(IAsyncResultresult)
{
try
{
TcpListenerserver
= (TcpListener)result.AsyncState;
using (TcpClientclient = server.EndAcceptTcpClient(result))
{
using (NetworkStreamstream = client.GetStream())
{
// 获取
Console.WriteLine( " SERVER:客户端已连接,数据读取中--- " );
byte []myRequestBuffer = new byte [ 1024 ];

int myRequestLength = 0 ;
do
{
myRequestLength
= stream.Read(myRequestBuffer, 0 ,myRequestBuffer.Length);
}
while (stream.DataAvailable);

CMsgmsg
= CMsg.ParseFrom(myRequestBuffer.RemoveEmptyByte(myRequestLength));

CMsgHeadhead
= CMsgHead.ParseFrom(Encoding.ASCII.GetBytes(msg.Msghead));
CMsgRegbody
= CMsgReg.ParseFrom(Encoding.ASCII.GetBytes(msg.Msgbody));

IDictionary
< Google.ProtocolBuffers.Descriptors.FieldDescriptor, object > d = head.AllFields;
foreach (varitem in d)
{
Console.WriteLine(item.Value.ToString());
}

d
= body.AllFields;
Console.WriteLine(
" =========================================== " );
foreach (varitem in d)
{
Console.WriteLine(item.Value.ToString());
}

Console.WriteLine(
" SERVER:响应成功--- " );

Console.WriteLine(
" SERVER:关闭连接--- " );
stream.Close();
}
client.Close();
}
}
finally
{
allDone.Set();
}
}

// 客户端请求
private static void runClient( object state)
{
try
{
CMsgHeadhead
= CMsgHead.CreateBuilder()
.SetMsglen(
5 )
.SetMsgtype(
1 )
.SetMsgseq(
3 )
.SetTermversion(
4 )
.SetMsgres(
5 )
.SetTermid(
" 11111111 " )
.Build();

CMsgRegbody
= CMsgReg.CreateBuilder().
SetArea(
22 )
.SetRegion(
33 )
.SetShop(
44 )
.Build();

CMsgmsg
= CMsg.CreateBuilder()
.SetMsghead(head.ToByteString().ToStringUtf8())
.SetMsgbody(body.ToByteString().ToStringUtf8())
.Build();


Console.WriteLine(
" CLIENT:对象构造完毕... " );

using (TcpClientclient = new TcpClient())
{
// client.Connect(newIPEndPoint(IPAddress.Parse("192.168.1.116"),12345));
client.Connect( new IPEndPoint(IPAddress.Parse( " 127.0.0.1 " ), 12345 ));
Console.WriteLine(
" CLIENT:socket连接成功... " );

using (NetworkStreamstream = client.GetStream())
{
// 发送
Console.WriteLine( " CLIENT:发送数据... " );

msg.WriteTo(stream);

// 关闭
stream.Close();
}
client.Close();
Console.WriteLine(
" CLIENT:关闭... " );
}
}
catch (Exceptionerror)
{
Console.WriteLine(
" CLIENTERROR:{0} " ,error.ToString());
}
}

}
// endclass



public static class ExtensionClass{
public static byte []RemoveEmptyByte( this byte []by, int length)
{
byte []returnByte = new byte [length];

for ( int i = 0 ;i < length;i ++ )
{
returnByte[i]
= by[i];
}
return returnByte;

}
}

}

运行的效果:

这样就OK了,之后就可以把java 服务端的IP或端口改成C# IP和服务端的商品一样,或者反过来也是可以的。c++版本经过测试也是可以的。简直是一个爽字。

  • Ubutu 下编译
    下载protobuf-2.4.1.tar.gz:
    http://protobuf.googlecode.com/files/protobuf-2.4.1.tar.gz
    安装:
    tar zxvf protobuf-2.4.1.tar.gz
    cd protobuf-2.4.1
    ./configure
    make
    make check
    make install
    安装结束。
    验证:
    查看是否安装成功:protoc --version
    如果出现:libprotoc 2.4.1 则说明安装成功!

    需要G++

  • 如果出现错误:
    protoc: error while loading shared libraries: libprotobuf.so.0: cannot open
    shared object file: No such file or directory

    在命令窗口上键入:
    export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/usr/local/lib

  • 更多相关文章

    1. Android(安卓)SDK Manager无法显示可供下载的未安装SDK解决方案
    2. 在Android(安卓)Studio 中使用ADB命令模拟手机各种状态(记录中...
    3. 一起来学Android(安卓)Studio:(三)导入项目
    4. Android分包MultiDex源码分析
    5. android多线程断点下载——网络编
    6. 在Eclipse中导入新浪微博SDK
    7. android apk反编译(获取源码,资源文件等)
    8. Android应用开发之获取web服务器xml数据
    9. (三) 搭建cocos2d-x-android环境 Windows XP3 + Eclipse + NDKR7

    随机推荐

    1. Android(安卓)ListView(Selector 背景图片
    2. 在程序中设置android:gravity 和 android
    3. Android(安卓)线程池管理
    4. Android单元测试-javaeye
    5. Debug native code using addr2line on A
    6. 自定义Android(安卓)editText
    7. Android之日期及时间选择对话框
    8. Android(安卓)动态布局
    9. ch02 Android(安卓)TextView与EditView
    10. android实现调用系统音乐播放器