zoukankan      html  css  js  c++  java
  • Linux下安装protobuf并实现简单的客户端服务器端通信

    http://code.google.com/p/protobuf/downloads/list上可以下载Protobuf的源代码。

    安装步骤如下所示:

     1>tar -xzf protobuf-2.5.0.tar.gz


      //进入目录

     cd protobuf-2.5.0 


    //将protobuf的函数库放在此目录文件下

     ./configure--prefix=/usr/local/protobuf        



    //下面的make是操作目录下的一个Makefile文件,如果之前用过此命令,又先重新编译过,可在此之前输 make clean

     make                                                         

     make check 

     make install       //安装protobuf

     

     2> sudo vim /etc/profile

     添加路径

    export PATH=$PATH:/usr/local/protobuf/bin/                                            

    export PKG_CONFIG_PATH=/usr/local/protobuf/lib/pkgconfig/

    保存并执行下面语句,让添加的路径生效

    source /etc/profile



    同时在~/.profile中添加上面两行代码,否则会出现登录用户找不到protoc命令



    3>配置动态链接库路径

    sudovim /etc/ld.so.conf

    插入:

    /usr/local/protobuf/lib



    4> su ldconfig   //root权限,让上一步的动态链接生效。



    5>写消息文件:msg.proto

    package Im;   

    message helloworld   

    {   

        required int32     id = 1;  // ID     

        required string    str = 2;  // str    

        optional int32     opt = 3;  //optional field   

    }  

    将消息文件msg.proto映射成cpp文件

    protoc-I=. --cpp_out=. msg.proto

    可以看到生成了

    msg.pb.hmsg.pb.cc


    下面写服务器端,server.cc

    #include "msg.pb.h"
    #include <iostream>
    #include <string>
    #include <stdio.h>
    #include <errno.h>
    #include <stdlib.h>
    
    #include <unistd.h>
    #include <arpa/inet.h>
    #include <sys/types.h>
    #include <sys/socket.h>
    
    #define EXIT_ERR(m)
            do
            {
                    perror(m);
                    exit(EXIT_FAILURE);
            }while(0)
    
    int main(void)
    {
            Im::helloworld msg;
            int listenfd;
            //监听套接字
            if((listenfd = socket(PF_INET, SOCK_STREAM, 0)) < 0)
            {
                    EXIT_ERR("socket");
            }
    
            struct sockaddr_in servaddr;
            memset(&servaddr, 0, sizeof(servaddr)); //初始化地址,必须要有
            servaddr.sin_family = AF_INET;
            servaddr.sin_port = htons(5188);
            servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");
    
            //允许套接口和一个正在使用的地址捆绑    
            int on = 1;
            if(setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) <0)
                    EXIT_ERR("setsockopt");
    
            //绑定端口和地址
            if(bind(listenfd, (struct sockaddr*)&servaddr, sizeof(servaddr)) < 0)
                    EXIT_ERR("bind");
    
            //建立连接套接字
            int conn;
            if(listen(listenfd, SOMAXCONN) < 0)
                    EXIT_ERR("listen");
    
            //对方的地址
            struct sockaddr_in peeraddr;
    //      memset(&peeraddr, 0, sizeof(peeraddr));
            socklen_t len = sizeof(peeraddr);
    
            //接受连接
            if((conn = accept(listenfd, (struct sockaddr*)&peeraddr, &len)) < 0)
                    EXIT_ERR("accept");
    
            close(listenfd);        //接受连接后可以关闭监听套接口
    
            char recvbuf[1025] = {0};
            while(1)
            {
                    memset(recvbuf, 0, sizeof(recvbuf));
                    int ret = recv(conn, recvbuf, 1024, 0);
                    if(ret == 0)
                    {
                            std::cout<<"client close"<<std::endl;
                            break;
                    }
                    std::string data = recvbuf;
                    msg.ParseFromString(data);
                    std::cout<<msg.id()<<std::endl;
                    std::cout<<msg.str()<<std::endl;
            }
    
            close(conn);
            return 0;
    }
    
    


    客户端,client.cc

    #include "msg.pb.h"
    #include <iostream>
    #include <string>
    
    #include <stdio.h>
    #include <errno.h>
    #include <stdlib.h>
    
    #include <arpa/inet.h>
    #include <unistd.h>
    #include <sys/types.h>
    #include <sys/socket.h>
    
    #define EXIT_ERR(m)
            do
            {
                    perror(m);
                    exit(EXIT_FAILURE);
            }while(0)
    
    int main(void)
    {
            Im::helloworld msg;
            int listenfd;
            if((listenfd = socket(PF_INET, SOCK_STREAM, 0)) < 0)
                    EXIT_ERR("socket");
    
            //要连接的对方的地址
            struct sockaddr_in servaddr;
            servaddr.sin_family = AF_INET;
            servaddr.sin_port = htons(5188);
            servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");
    
            //连接
            if(connect(listenfd, (struct sockaddr*)&servaddr, sizeof(servaddr)) < 0)
                    EXIT_ERR("connect");
    
            char sendbuf[1024] = {0};
            int num;
            std::string str;
            std::string data;
            while(1)
            {
                    memset(sendbuf, 0, sizeof(sendbuf));
                    std::cout<<"input number: ";
                    std::cin>>num;
                    std::cout<<"input string: ";
                    std::cin>>str;
                    fflush(stdout);
                    msg.set_id(num);
                    msg.set_str(str);
                    msg.SerializeToString(&data);
                    strcpy(sendbuf, data.c_str());
                    if(send(listenfd, sendbuf, strlen(sendbuf), 0) <= 0)
                    {
                            EXIT_ERR("send");
                            break;
                    }
            }
    
            close(listenfd);
            return 0;
    }
                                                                    57,1-8        81%
                                                                                    1,1          顶端
    


    分别编译两个文件

    g++ msg.pb.cc server.cc -o server `pkg-config --cflags --libs protobuf` -lpthread


    g++ msg.pb.cc client.cc -o client `pkg-config --cflags --libs protobuf` -lpthread


    执行即可。


    饮水思源:

    http://download.csdn.net/detail/fairy_tale__________/4692551

  • 相关阅读:
    PHP is_float()、 is_double()、is_real()函数
    自动驾驶关键技术分解和流程
    自动驾驶行业内时间表和技术解析
    自动驾驶架构与实现路径
    ADAS单目摄像头行驶区域环境光检测图片标注
    多目标检测整合算法
    道路场景语义分割算法
    TTC测距算法
    TSR交通标志检测与识别
    Mobileye_EyeQ4功能和性能
  • 原文地址:https://www.cnblogs.com/keanuyaoo/p/3370834.html
Copyright © 2011-2022 走看看