【protobuf】ProtoBuf——proto3语法详解、enum类型、enum类型的使用和注意事项、Any类型、通讯录录入号码类型和地址的功能实现

ProtoBuf

在这里插入图片描述

  

5. proto3语法详解

5.3 enum类型

  定义规则:

  proto3支持我们定义枚举类型并使用:

  枚举类型的名称采用驼峰命名法且首字母大写,如 MyEnum ,这样的命名方式符合常见的编程规范,能够提高代码的可读性。例如,ColorEnum 、StatusEnum 等都是符合要求的枚举类型名称。

  对于枚举常量值的名称,使用全大写字母并用 _ 连接,如 ENUM_CONST = 0; 。这种命名方式可以明确区分常量值和普通变量,增强代码的自解释性。例如,STATUS_SUCCESS = 0; 、COLOR_RED = 1;

  
  我们可以定义⼀个名为 PhoneType 的枚举类型:

enum PhoneType {
   
	MP = 0; // 移动电话
	TEL = 1; // 固定电话
}

  

  要注意枚举类型的定义有以下几种规则:

  规则 1:0 值常量必须存在且作为第一个元素:在枚举类型中,将 0 值常量置于首位并作为默认值,这是为了保持与 proto2 语义的兼容性。例如:

enum Phone {
   
	DEFAULT_VALUE = 0;
	MP = 1;
	TEL = 2;
}

  这样,当没有明确指定枚举值时,默认就会使用 DEFAULT_VALUE 。

  

  规则 2:枚举类型可以在消息外或消息体内定义(嵌套)在消息外定义的枚举类型可以被多个消息使用,而在消息体内定义的枚举类型则仅能在该消息中使用。消息外定义的示例:

enum Phone {
   
  MP = 0;
  TEL = 1;
}

message Message {
   
  Phone phone = 1;
}

  消息体内定义(嵌套)的示例:

message Message {
   
  enum Phone {
   
    MP = 0;
    TEL = 1;
  }

  Phone phone = 1;
}

  

  规则 3:枚举的常量值在 32 位整数范围内,但因负值无效因而不建议使用。 由于编码规则的原因,负值在枚举常量值中不太有效且不被推荐使用。例如,有效的常量值可以是 0 到 2147483647 之间的整数。

  

  定义时注意:

  将两个 ‘具有相同枚举值名称’ 的枚举类型放在单个 .proto 文件下测试时,编译后会报错:某某某常量已经被定义!所以这里要注意:

  (1)当在单个 .proto 文件中,同级(同层)的枚举类型中,各个枚举类型的常量不能重名。

enum PhoneType {
   
	MP = 0; // 移动电话
	TEL = 1; // 固定电话
}

enum PhoneTypeCopy {
   
	MP = 0; // 移动电话 // 编译后报错:MP 已经定义
}

  
  (2)但单个 .proto 文件下,最外层枚举类型和嵌套枚举类型不算同级。

enum PhoneTypeCopy {
   
	MP = 0; // 移动电话 // ⽤法正确
}

message Phone {
   
	string number = 1; // 电话号码
	
	enum PhoneType {
   
		MP = 0; // 移动电话
		TEL = 1; // 固定电话
	}
}

  
  (3)在多个 .proto 文件的情况下,如果一个文件引入了其他文件,且每个文件都未声明package,每个文件中的枚举类型都在最外层,此时算同级,不能有重名的常量。

// phone1.proto
import "phone1.proto"

enum PhoneType {
   
	MP = 0; // 移动电话 // 编译后报错:MP 已经定义
	TEL = 1; // 固定电话
}

// phone2.proto
enum PhoneTypeCopy {
   
	MP = 0; // 移动电话 
}

  
  (4)在多个 .proto 文件的情况下,如果一个文件引入了其他文件,而如果每个文件都声明了 package,则不算同级,可以有相同名称的枚举类型。

// phone1.proto
import "phone1.proto"
package phone1;

enum PhoneType {
   
	MP = 0; // 移动电话 // ⽤法正确
	TEL = 1; // 固定电话
}

// phone2.proto
package phone2;
enum PhoneTypeCopy {
   
	MP = 0; // 移动电话 
}

  
  使用枚举类型向我们的字段中添加电话类型:

syntax = "proto3";
package contacts;

// 联系⼈
message PeopleInfo {
   
    string name = 1; // 姓名
    int32 age = 2; // 年龄
    
    message Phone 
    {
   
        string number = 1; // 电话号码
        enum PhoneType 
        {
   
            MP = 0; // 移动电话
            TEL = 1; // 固定电话
        }
        
        PhoneType type = 2; // 类型
    }
 
    repeated Phone phone = 3; // 电话
}

// 通讯录
message Contacts {
   
    repeated PeopleInfo contacts = 1;
}

  
  protoc编译文件:

protoc --cpp_out=. contacts.proto

  
  查看contacts.pb.h文件,这就是自动生成的有关enum的函数:

在这里插入图片描述

在这里插入图片描述

  

  bool PeopleInfo_Phone_PhoneType_IsValid(int value) :这个函数检查一个整数是不是有效的 PeopleInfo_Phone_PhoneType 枚举值,返回是或否。
  

  inline bool PeopleInfo_Phone_PhoneType_Parse(::PROTOBUF_NAMESPACE_ID::ConstStringParam name, PeopleInfo_Phone_PhoneType* value) :它把一个字符串转成 PeopleInfo_Phone_PhoneType 枚举值存到指定位置,返回转成功还是没成功。
  

  template<typename T> inline const std::string& PeopleInfo_Phone_PhoneType_Name(T enum_t_value) :这是个模板函数,能得到给定的 PeopleInfo_Phone_PhoneType 枚举值对应的名字字符串。

  
  枚举类型中的成员也有自己的函数:

在这里插入图片描述
  
  更新 write.cc :

#include <iostream>
#include <fstream>
#include "contacts.pb.h"
using namespace std;
using namespace contacts;
/**
 * 新增联系⼈
 */
void AddPeopleInfo(PeopleInfo *people_info_ptr)
{
   
    cout << "-------------新增联系⼈-------------" << endl;
    cout << "请输⼊联系⼈姓名: ";
    string name;
    getline(cin, name);
    people_info_ptr->set_name(name);
    cout << "请输⼊联系⼈年龄: ";
    int age
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

鳄鱼麻薯球

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值