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

android与PC,C#与Java 利用protobuf 进行无障碍通讯【Socket】 2011-04-27 17:00:11
标签:休闲 Java Socket 移动开发 android
原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 、作者信息和本声明。否则将追究法律责任。http://terryblog.blog.51cto.com/1764499/554930

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的消息文件,内容如下:

package msginfo;

message CMsg
{
    required 
string msghead = 1;
    required 
string msgbody = 2;
}

message CMsgHead
{
    required int32 msglen 
= 1;
    required int32 msgtype 
= 2;
    required int32 msgseq 
= 3;
    required int32 termversion 
= 4;
    required int32 msgres 
= 5;
    required 
string termid = 6;
}

message CMsgReg
{
    optional int32 area 
= 1;
    optional int32 region 
= 2;
    optional int32 shop 
= 3;
    optional int32 ret 
= 4;
    optional 
string termid = 5[defalut="12345"];
}

message CMsgLogin
{
    optional int32 ret 
= 1;
}

message CMsgLogout
{
    optional int32 ret 
= 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。为了方便你可以将其做成一个批处理文件代码如下:
    echo on
    protoc 
    --descriptor_set_out=msg.protobin --include_imports msg.proto 
    protogen msg.protobin   

     将其另存为.bat文件即可

     

 

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

android 与PC

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

客户端代码:

package net.testSocket;

import java.io.IOException;
import java.io.InputStream;
import java.net.Socket;
import java.net.UnknownHostException;

import socket.exception.SmsClientException;
import socket.exception.SmsObjException;

import msginfo.Msg.CMsg;
import msginfo.Msg.CMsgHead;
import msginfo.Msg.CMsgReg;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;

import com.google.protobuf.InvalidProtocolBufferException;

//客户端的实现
public class TestSocket extends Activity {
    
private TextView text1;
    
private Button but1; 
    Socket socket 
= null;

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

        
// Thread desktopServerThread=new Thread(new AndroidServer());
        
// 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(View v) {
 
                try {
                  
                     socket = new Socket("192.168.1.116"12345);
                    
//得到发送消息的对象 
                 
                    CMsgHead head = CMsgHead.newBuilder().setMsglen(5)
                            .setMsgtype(
1).setMsgseq(3).setTermversion(41)
                            .setMsgres(
5).setTermid("11111111").build();

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

                    
// Msg
                    CMsg msg = CMsg.newBuilder()
                            .setMsghead(head.toByteString().toStringUtf8())
                            .setMsgbody(body.toByteString().toStringUtf8())
                            .build();
 
                    // 向服务器发送信息
                    msg.writeTo(socket.getOutputStream());
                 
                    // 接受服务器的信息
                    InputStream input = socket.getInputStream();

                 
                    byte[] by=recvMsg(input);
                    setText(CMsg.parseFrom(by));

                 
                    input.close();
                 
                    socket.close();
                } 
catch (UnknownHostException e) {
                    e.printStackTrace();
                } 
catch (IOException e) {
                    e.printStackTrace();
                } 
catch (Exception e) {
                    System.
out.println(e.toString());
                }
                
// };
                
// }.start();

            }
        });

    }
    
    
/**
     * 接收server的信息
     * 
     * @return
     * @throws SmsClientException
     * @author fisher
     
*/
    
public byte[] recvMsg(InputStream inpustream) throws SmsObjException {
        
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 (Exception localException) {
            
throw new SmsObjException("SmapObj.recvMsg() occur exception!"
                    
+ localException.toString());
        }
    }

    
/**
     * 得到返回值添加到文本里面
     * 
     * @param g
     * @throws InvalidProtocolBufferException
     
*/
    
public void setText(CMsg g) throws InvalidProtocolBufferException {
        CMsgHead h 
= CMsgHead.parseFrom(g.getMsghead().getBytes());
        StringBuffer sb 
= 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");

        CMsgReg bo 
= 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());
    }

}

 

服务端代码:

 package server;


import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;

import msginfo.Msg.CMsg;
import msginfo.Msg.CMsgHead;
import msginfo.Msg.CMsgReg;

public class AndroidServer implements Runnable {

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

                DataOutputStream dataOutputStream;
                DataInputStream dataInputStream;

                
try {
                
                    InputStream inputstream = client.getInputStream();

                    dataOutputStream 
= new DataOutputStream(
                            client.getOutputStream());
                  
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];                              
                    } 

                    
// 协议正文
//                                      CMsg msg = CMsg.parseFrom(temp);
                    
//
                    
//
                    CMsgHead head = 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());

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

                    inputstream.close();
                 
                } catch (Exception ex) {
                    System.
out.println(ex.getMessage());
                    ex.printStackTrace();
                } 
finally {
                    client.close();
                    System.
out.println("close");
                }
            }
        } 
catch (IOException e) {
            System.
out.println(e.getMessage());
        }
    }

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

    
private byte[] getProtoBufBack() {

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

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

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

        
return msg.toByteArray();
    }

    
private void sendProtoBufBack(DataOutputStream dataOutputStream) {

        
byte[] backBytes = getProtoBufBack();
        
// 协议头部
 
        try {
 
            dataOutputStream.write(backBytes, 0, backBytes.length);
            dataOutputStream.flush();
        } 
catch (IOException e) {
            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 ManualResetEvent allDone = new ManualResetEvent(false);

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

        
private static void beginProtocbuf()
        {
            
//启动服务端
            TcpListener server = 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(IAsyncResult result)
        {
            
try
            {
                TcpListener server 
= (TcpListener)result.AsyncState;
                
using (TcpClient client = server.EndAcceptTcpClient(result))
                {
                    
using (NetworkStream stream = client.GetStream())
                    {
                        
//获取
Console.WriteLine("SERVER : 客户端已连接,数据读取中 --- ");
byte[] myRequestBuffer = new byte[1024];

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

                        CMsgHead head 
= CMsgHead.ParseFrom(Encoding.ASCII.GetBytes(msg.Msghead));
                        CMsgReg body 
= CMsgReg.ParseFrom(Encoding.ASCII.GetBytes(msg.Msgbody));

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

                        d 
= body.AllFields;
                        Console.WriteLine(
"===========================================");
                        
foreach (var item 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
            {
                CMsgHead head 
= CMsgHead.CreateBuilder()
                    .SetMsglen(
5)
                    .SetMsgtype(
1)
                    .SetMsgseq(
3)
                    .SetTermversion(
4)
                    .SetMsgres(
5)
                    .SetTermid(
"11111111")
                    .Build();

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

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


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

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

                    
using (NetworkStream stream = client.GetStream())
                    {
                        
//发送
                        Console.WriteLine("CLIENT : 发送数据 ...");
                      
                        msg.WriteTo(stream);

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

    }
//end class



    
public static class ExtensionClass {
        
public static 

} 

 

 运行的效果:

 

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

本文出自 “Terry_龙” 博客,请务必保留此出处http://terryblog.blog.51cto.com/1764499/554930

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/353855.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

jenkins api_接触Jenkins(Hudson)API,第1部分

jenkins api哪一个-哈德森还是詹金斯&#xff1f; 都。 几个月前&#xff0c;我开始使用Hudson v1.395来从事这个小项目&#xff0c;在出现巨大分歧之后又回到了这个项目。 我以此为契机&#xff0c;看我将来选择永久搬到詹金斯时是否会遇到任何重大问题。 出现了一些麻烦-最值…

Swing应用程序中的JavaFX 8 DatePicker

1.概述 本文显示了一个使用JavaFX 8 DatePicker控件的Java SE 8 Swing应用程序的示例。 DatePicker控件允许用户以文本形式输入日期或从日历弹出窗口中选择日期。 本示例使用其中带有FX控件的Swing JFrame 。 为了将FX内容嵌入Swing应用程序中&#xff0c; javafx.embed.swing…

Apache Bench安装与使用

转载&#xff1a;ApacheBench&#xff08;ab&#xff09;使用详解 一、Apache Bench简介 ApacheBench 是 Apache 服务器自带的一个web压力测试工具&#xff0c;简称ab。ab又是一个命令行工具&#xff0c;对发起负载的本机要求很低&#xff0c;根据ab命令可以创建很多的并发访问…

轻松搭建一个Windows SVN服务器

轻松搭建一个Windows SVN服务器 前文所述SVN客户端使用的时候&#xff0c;用的SVN服务器通常为外部&#xff0c;例如Google Code的服务器&#xff0c;不过&#xff0c;做为一个程序开发人员&#xff0c;就算自己一个人写程序&#xff0c;也应该有一个SVN版本控制系统&#xff0…

Spring框架中的内容协商

1.简介 使用BeanNameViewResolver完成的工作就是&#xff0c;我们刚刚在Spring上下文中创建了多个bean视图以生成预期的输出。 Spring很快引入了内容协商策略 &#xff0c;该策略可以使用传统的RESTful ResponseBody方法和HTTP消息转换器&#xff0c;以JSON或XML返回所需的输出…

logging配置

settings.py # Django的日志配置项BASE_LOG_DIR os.path.join(BASE_DIR, "log")LOGGING {version: 1, # 保留字disable_existing_loggers: False, # 禁用已经存在的logger实例# 日志文件的格式formatters: {# 详细的日志格式standard: {format: [%(asctime)s][%(…

Protobuf学习笔记

Protobuf学习笔记 Posted by iamxhuon 2012/05/22 Leave a comment (0)Go to commentsProtocol buffers是什么&#xff1f; 首先了解一下Protocol Buffers(简称ProtoBuf)是什么&#xff1f;官网对它的定义如下&#xff1a; Protocol buffers are Google’s language-neutral, …

Tortoise SVN 版本控制常用操作知识

Tortoise SVN 版本控制常用操作知识 Posted on 2010-11-26 23:07 szh114 阅读(5897) 评论(0) 编辑 收藏 今天老大跑过来问我如何把SVN服务器上的当前版本回退到某一个版本上去&#xff0c;我没回答上来&#xff0c;很失败&#xff0c;所以现在整理一下Tortoise SVN的操作知识&…

如何导入任何JBoss BRMS示例项目

在过去几周内&#xff0c;JBoss BRMS演示的用户反复询问我以下内容时&#xff0c;会给您这些提示和技巧&#xff1a; “如何将与各种JBoss BRMS演示项目相关的项目导入到我自己的现有安装中&#xff1f;” 这意味着用户希望在产品的个人安装中有一个示例项目&#xff0c;而无…

2110: 扫雷

http://acm.zcmu.edu.cn/JudgeOnline/problem.php?id2110 2110: 扫雷 Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 88 Solved: 36[Submit][Status][Web Board]Description 扫雷游戏你一定玩过吧&#xff01;现在给你若干个nm的地雷阵&#xff0c;请你计算出每个矩阵中每…

青椒苗

转载于:https://www.cnblogs.com/wainiwann/p/8793418.html

如何开始Java机器学习

什么是开始使用Java机器学习的最佳工具&#xff1f; 他们已经存在了一段时间&#xff0c;但如今看来&#xff0c;每个人都在谈论人工智能和机器学习。 对于科学家和研究人员而言&#xff0c;它已不再是秘密&#xff0c;几乎可以在任何新兴技术中实现。 在下面的文章中&#x…

ztree树

常规的ztree树 后台数据封装成list对象 public PageModel getTreeBuildingRegData(Map<String, String> params) {PageModel pageModelnew PageModel();String statusparams.get("status");String orgIdparams.get("org_id");List<OmsBuildingReg…

如何提高效率

如何提高效率 时间管理 April 28th, 2011 本文来自读者 桃雨 翻译投稿。 Aaron Swartz写过一篇很有名的文章&#xff0c;叫做《HOWTO: Be more productive》&#xff08;如何提高效率&#xff09;。这篇文章写的实在是太好了&#xff0c;我看了好多遍&#xff0c;很赞同作者的…

Andrew Ng - 深度学习工程师 - Part 2. 改善深层神经网络:超参数调试、正则化以及优化(Week 1. 机器学习的实用层面)...

第1周 机器学习的实用层面 1.1 训练/开发/测试 早期机器学习时代&#xff08;数据规模较小&#xff09;&#xff0c;如果不需要dev set&#xff0c;常见的划分有 70%/30% 的训练/测试 划分&#xff0c;如果需要验证集&#xff0c;常见的是 60%/20%/20%划分 在big data era&…

jms 如何测试_使用JMSTester对JMS层进行基准测试

jms 如何测试对于我去过的大多数客户端&#xff0c;使用ActiveMQ扩展JMS消息传递层是一个优先事项。 有多种方法可以实现这一目标&#xff0c;但毫无疑问&#xff0c;创建基准并分析实际硬件上的体系结构&#xff08;或者正如我的同事Gary Tully所说的“询问机器”&#xff09;…

为什么待办事项清单不管用

为什么待办事项清单不管用 时间管理 November 22nd, 2012 本文原文来自 Harvard Business Review&#xff0c;由 换装迷宫tayy 翻译。 停止制作你的待办事项清单吧。它们只会让你感觉失败和受挫。想想你正在管理的那些待办清单&#xff1a;有多少事项从年初起就已经在那儿了&…

charts 画饼图

统计某一天某类物体的百分比 新知识点&#xff1a;aggregate https://blog.csdn.net/congcong68/article/details/51619882 主要的 $group $match $sort $limit pipeline [{$group:{_id:$area,count:{$sum:1}}}, # count 是聚合之后新增的一个字段。{$sort:{count:1}}, # …

Apache Camel 2.18 –即将推出的功能的亮点

骆驼骑士正在忙于即将发布的Apache Camel 2.18版本。 当我们动态更新发行说明时 &#xff0c;这是一种快速查看即将发生的情况的方法。 我只是想在夏天在这里进行快速更新&#xff0c;到目前为止&#xff0c;我们已经完成了简短的重点介绍。 随着发行版的临近和文档的完善&…

实时多线程系统的日志实现

实时多线程系统的日志实现 2008-03-21 09:19 黄明/戴颖  软件世界 我要评论(0) 字号&#xff1a;T | T为了分析软件系统在测试和运行期产生的故障&#xff0c;目前大多数软件系统所广泛使用的一种方法就是日志记录。本文给出了利用循环缓冲区和单独的日志读写线程实现实时系统…