zoukankan      html  css  js  c++  java
  • nanopb教程

    nanopb的安装和使用

    nanopbprotobuf协议的纯C实现,没有依赖其他库,只需要几个C文件就可以了.非常适合用来做嵌入式设备的通信协议.

    1. 第一步安装protobuf
      去github上下载一个protobuf的release版本,下载all版本,在本地解压缩之后,通过make install来安装.并安装python语言支持

    bogon:protobuf-3.5-1.1 see$ ./configure
    .....(等待执行完成)
    bogon:protobuf-3.5-1.1 see$ make install
    ....(等待执行完成,protobuf就安装好了)
    bogon:protobuf-3.5-1.1 see$ protoc --version
    libprotoc 3.5.1
    bogon:protobuf-3.5-1.1 see$ cd python/
    bogon:python see$ python setup.py build
    bogon:python see$ python setup.py install
    .....(等待执行完成)

    1. 下载nanopb
      去github上下载一个nanopb的release版本,解压之后,能够在目录下看到下面7文件,这7个文件我们需要添加到c工程里面的

    bogon:nanopb-0.3.9 see$ ls
    pb_common.c pb_common.h pb_encode.c pb_encode.h
    pb_decode.c pb_decode.h pb.h

    1. 编译.proto文件
      先用protoc命令编译.proto文件,生成中间文件,然后再执行nano的python脚本(./generator/nanopb_generator.py),将中间文件生成所需要的c文件

    bogon:lock see$ protoc lock.proto -o lock.pb
    bogon:lock see$ ls
    bogon:lock see$ lock.proto lock.pb
    bogon:lock see$ python ../nanopb-0.3.9/generator/nanopb_generator.py lock.pb
    bogon:lock see$ ls
    lock.pb.c lock.pb.h

    1. 完成
      至此,我们就完成了nanopb的安装和.proto文件的生成,为了方便使用,我把执行python那段命令写成了一个shell脚本,每次生成的时候,顺便把所需要的7个文件也拷贝到同一个文件夹下面,将脚本修改成可执行文件(chmod +x nanopb),放到/usr/local/bin 目录下,以后使用就可以使用命令nanopb来用了

    bogon:lock see$ nanopb lock.pb
    bogon:lock see$ ls
    lock.proto lock.pb ccode
    bogon:lock see$ cd ccode
    bogon:ccode see$ ls
    pb_common.c pb_common.h pb_encode.c pb_encode.h
    pb_decode.c pb_decode.h pb.h lock.pb.c lock.pb.h

    #!/bin/sh
    nanodir="nanopb的根目录"
    nanopy=$nanodir"/generator/nanopb_generator.py"
    file1=$nanodir"/pb_common.c"
    file2=$nanodir"/pb_common.h"
    file3=$nanodir"/pb_decode.c"
    file4=$nanodir"/pb_decode.h"
    file5=$nanodir"/pb_encode.c"
    file6=$nanodir"/pb_encode.h"
    file7=$nanodir"/pb.h"
    srcdir=`pwd`
    mkdir $srcdir"/ccode"
    dir=$srcdir"/ccode"
    for i in "$@"; do
        python $nanopy $i
        mv $srcdir"/"$i".c" $dir
        mv $srcdir"/"$i".h" $dir
    done
    cp $file1 $dir
    cp $file2 $dir
    cp $file3 $dir
    cp $file4 $dir
    cp $file5 $dir
    cp $file6 $dir
    cp $file7 $dir
    

    nanopb关于string类型的处理

    在nanopb中,string类型在生成c语言文件的时候,会有两种结构,一种是指定了最大长度的,一种是没有指定最大长度.指定了最大长度的string,会生成char[] 数组类型,没有指定最大长度的,会生成pb_callback_t类型.具体的可以参照nanopb文档

    pb_callback_t 是一个结构体,有两个成员变量,一个是回调函数指针,这个回调函数是一个union,在编码的时候,需要赋值encode函数,解码的时候赋值decode,如果不赋值,则该属性值会忽略.
    另外一个变量是arg,这是一个指针,会在回调函数中作为最后一个参数传递给回调函数.

    typedef struct pb_callback_s pb_callback_t;
    struct pb_callback_s {
    #ifdef PB_OLD_CALLBACK_STYLE
        /* Deprecated since nanopb-0.2.1 */
        union {
            bool (*decode)(pb_istream_t *stream, const pb_field_t *field, void *arg);
            bool (*encode)(pb_ostream_t *stream, const pb_field_t *field, const void *arg);
        } funcs;
    #else
        /* New function signature, which allows modifying arg contents in callback. */
        union {
            bool (*decode)(pb_istream_t *stream, const pb_field_t *field, void **arg);
            bool (*encode)(pb_ostream_t *stream, const pb_field_t *field, void * const *arg);
        } funcs;
    #endif    
        
        /* Free arg for use by callback */
        void *arg;
    };
    

    调用方法如下例:

    //
    //  main.c
    //  Test
    //
    //  Created by Wangchun on 2018/2/11.
    //  Copyright © 2018年 Benzhuo. All rights reserved.
    //
    
    #include <stdio.h>
    #include "pb.h"
    #include "message.pb.h"
    #include "pb_common.h"
    #include "pb_encode.h"
    #include "pb_decode.h"
    
    /**
     编码的回调函数
     **/
    bool write_string(pb_ostream_t *stream, const pb_field_t *field, void * const *arg)
    {
        char *str = *arg;
        if (!pb_encode_tag_for_field(stream, field))
            return false;
        
        return pb_encode_string(stream, (uint8_t*)str, strlen(str));
    }
    
    /**
     解码回调函数
     **/
    bool read_ints(pb_istream_t *stream, const pb_field_t *field, void **arg)
    {
        int i=0;
        char* tmp = *arg;
        while (stream->bytes_left)
        {
            uint64_t value;
            if (!pb_decode_varint(stream, &value))
                return false;
            *(tmp+i)=value;
            i++;
        }
        return true;
    }
    
    int main(int argc, const char * argv[]) {
        // insert code here...
        printf("Hello, World!
    ");
        printf("hello world!!!!!
    ");
        uint8_t buffer[128];
        size_t message_length;
        bool status;
        {
            SigninReq sign = SigninReq_init_zero;
            sign.lockId.funcs.encode = write_string;
            sign.lockId.arg = &"hahaha";
            sign.sid=12345;
            pb_ostream_t stream  = pb_ostream_from_buffer(buffer, sizeof(buffer));
            status = pb_encode(&stream, SigninReq_fields, &sign);
            message_length = stream.bytes_written;
            /* Then just check for any errors.. */
            if (!status)   {
                printf("Encoding failed: %s
    ", PB_GET_ERROR(&stream));
                return 1;
            }
        }
        
        {
            /* Allocate space for the decoded message. */
            SigninReq message = SigninReq_init_zero;
            message.lockId.funcs.decode = read_ints;
            char tmp[128]={0};
            message.lockId.arg = &tmp;
            /* Create a stream that reads from the buffer. */
            pb_istream_t stream = pb_istream_from_buffer(buffer, message_length);
            
            /* Now we are ready to decode the message. */
            status = pb_decode(&stream, SigninReq_fields, &message);
            
            /* Check for errors... */
            if (!status)
            {
                // printf("Decoding failed: %s
    ", PB_GET_ERROR(&stream));
                return 1;
            }
            
            /* Print the data contained in the message. */
             printf("Your sid = %d!, lockId=%s
    ", message.sid,tmp);
            
        }
        
        return 0;
    }
    

    这种方式使用起来比较复杂,一般情况下,我们的长度都不会太长,这样我们可以设定string的最大长度,就可以用char[]数组的方式来处理了.这样使用起来比较简单.设定最大长度的方式有几种,可以参照网站上来做,我用的是options配置文件的方式来生成.

    首先创建.proto文件

    syntax = "proto3";
    
    message SigninReq {
        int32 sid = 1;// session id ,int型随机数
        string uid = 2; //userId
    }
    

    再创建一个.options文件

    # lock.options
    SigninReq.uid   max_size:16
    

    protoc生成中间文件的方式不变,在执行nanopb_generator.py的时候,添加上参数 -f lock.options, 这样就可以生成char[]数组类型了.为了方便,将上篇文章中的shell脚本进行了修改.去掉了编译多个pb文件生成c文件功能,一次只编译一个pb文件.

    #!/bin/sh
    nanodir="nanopb的home目录"
    nanopy=$nanodir"/generator/nanopb_generator.py"
    file1=$nanodir"/pb_common.c"
    file2=$nanodir"/pb_common.h"
    file3=$nanodir"/pb_decode.c"
    file4=$nanodir"/pb_decode.h"
    file5=$nanodir"/pb_encode.c"
    file6=$nanodir"/pb_encode.h"
    file7=$nanodir"/pb.h"
    srcdir=`pwd`
    mkdir $srcdir"/ccode"
    dir=$srcdir"/ccode"
    params=""
    isP=0
    pbfile=""
    for i in "$@"; do
        if [[ $i == "-"* ]];then 
            params=$params" "$i
            isP=1
        elif [ $isP -eq 1 ];then
           params=$params" "$i
           isP=0
        else
           pbfile=$i
        fi
    done
    echo "pbfile:"$pbfile
    echo "params:"$params
    python $nanopy $pbfile $params
    mv $srcdir"/"$pbfile".c" $dir
    mv $srcdir"/"$pbfile".h" $dir
    cp $file1 $dir
    cp $file2 $dir
    cp $file3 $dir
    cp $file4 $dir
    cp $file5 $dir
    cp $file6 $dir
    cp $file7 $dir
    

    生成源码命令

    $ protoc -o message.pb message.proto
    $ python nanopb/generator/nanopb_generator.py message.pb

    参考链接:

    https://www.jianshu.com/p/27d5877b7289

    https://www.jianshu.com/p/cad578f48e0a

    https://github.com/nanopb/nanopb

  • 相关阅读:
    mysql 初始密码 设置
    jsp基础知识(基本的语法及原理)
    hdu 2473 Junk-Mail Filter (并查集之点的删除)
    java版本的学生管理系统
    java操作数据库出现(][SQLServer 2000 Driver for JDBC]Error establishing socket.)的问题所在即解决办法
    Java学习之约瑟夫环的两中处理方法
    hdu 3367(Pseudoforest ) (最大生成树)
    hdu 1561 The more, The Better (树上背包)
    Nginx + Lua 搭建网站WAF防火墙
    长连接和短连接
  • 原文地址:https://www.cnblogs.com/hzcya1995/p/13312605.html
Copyright © 2011-2022 走看看