背景:由于某些原因在环境上无发使用protobuf动态库且使用静态库会出现protobuf的符号冲突,通过隐藏符号来解决该问题
概要:生成protobuf隐藏符号的静态库,并用该静态库生成so,再通过两者使用包含string字段的protobuf对象时,出现各种异常。
详细描述:
protobuf隐藏符号后生成静态库,使用该静态库再生成so(例如protoc生成的*.pb.cc文件)时,protobuf/sr/google/protobuf/fixmessage_lite.h
文件中的fixed_address_empty_string
对象会在该so中生成一个local对象。当同时使用生成的so和隐藏符号的静态库时(如main.cpp生成可执行文件),main中也存在一个fixed_address_empty_string
对象,当操作main中包含string字段的protobbuf对象时(string字段的生成和使用会依赖fixed_address_empty_string),会出现fixed_address_empty_string
初始化是使用so中的local对象,而在使用时是使用main中的fixed_address_empty_string
对象,从而导致冲突,造成各种奇怪现象。
解决方案:可以将protobuf/src/google/protobuf/generated_message_util.cc
文件中的fixed_address_empty_string
定义添加上__attribute__((visibility ("default")))
属性,从而生成静态库时不隐藏该符号,也就不会在so中产生该local对象
结论:使用protobuf包含string字段的对象时,要保证操作的fixed_address_empty_string
对象是同一个。
后续:由于此方法只能解决string字段的问题,可能还有其他隐藏问题,最后的解决方案为修改命名空间,google的开源项目都支持修改名字空间,编译时添加-Dgoogle=XXX
选项,生成的符号也就不会发生冲突。