有没有办法有效地将 rapidjson 与 std::string 一起使用?
Is there way to use rapidjson with std::string efficiently?
我正在尝试使用 rapidjson。
我想生成字符串并将其添加到一些 rapidjson::Value
对象中。
我在使用 qjson 时使用 std::string,但在 rapidjson 的情况下似乎不合适。我不想生成字符串然后复制它,字符串对象生命周期在对象 (rapidjson::Value
) 生命周期之前结束(因此 generated_string.c_str()
不是这种情况)。
json中可能有\0,因此,char*
以null结尾的字符串也不是解决方案。
所以,我必须写我自己的字符串类型?或者使用类似
auto create_string_object() -> rapidjson::GenericStringRef<char>
{
size_t len;
char* result;
// generate char* result with length len
const char* return_value = result;
return rapidjson::StringRef(return_value,len);
}
auto create_object_object(rapidjson::Document::AllocatorType &allocator) -> rapidjson::Value
{
// ...
rapidjson::Value result(rapidjson::kObjectType);
rapidjson::Value tmp; // tmp = create_string_object() will not compile
tmp = create_string_object();
result.AddMember("key", tmp, allocator);
// ...
}
或者还有其他一些处理字符串的方法?这对我来说似乎很难。我们不能将字符串移动到 rapidjson::Value
因为 Value
内部的结构不同,我们不能将 Value
内部的指针设置为 c_str()
因为字符串将在 [= 之前被销毁18=]。
即使使用 GenericStringRef<char>
我也必须重写几乎所有使用字符串的工作。
对了,为什么RAPIDJSON_HAS_STDSTRING
默认是0?工作有问题?我看到如果我知道字符串生命周期将在值生命周期之前结束,我可以将字符串复制到 rapidjson::Value
并复制指针。
UPD: 现在我看到 rapidjson 只释放被复制的字符串:
//! Destructor.
/*! Need to destruct elements of array, members of object, or copy-string.
*/
~GenericValue() {
if (Allocator::kNeedFree) { // Shortcut by Allocator's trait
switch(flags_) {
case kArrayFlag:
for (GenericValue* v = data_.a.elements; v != data_.a.elements + data_.a.size; ++v)
v->~GenericValue();
Allocator::Free(data_.a.elements);
break;
case kObjectFlag:
for (MemberIterator m = MemberBegin(); m != MemberEnd(); ++m)
m->~Member();
Allocator::Free(data_.o.members);
break;
case kCopyStringFlag:
Allocator::Free(const_cast<Ch*>(data_.s.str));
break;
default:
break; // Do nothing for other types.
}
}
}
因此,正如在回答中所说,以我的代码中的方式使用 GenericStringRef
是一个坏主意,因为在这种情况下我必须自己管理内存。
我没有完全理解这个问题。但我想在这里澄清一些事情。
GenericStringRef
用于防止字符串复制。仅当已知字符串的生命周期足够时才应使用它。对于动态创建的字符串,通常不应该使用 GenericStringRef
.
- 设置
RAPIDJSON_HAS_STDSTRING=1
即可。默认情况下不打开,因为它的支持是在早期发布后添加的。如果用户不需要,我不希望 RapidJSON header 包含 <string>
。既然你用的是std::string
,你就可以开启。它应该让你更容易处理 std::string
.
我正在尝试使用 rapidjson。
我想生成字符串并将其添加到一些 rapidjson::Value
对象中。
我在使用 qjson 时使用 std::string,但在 rapidjson 的情况下似乎不合适。我不想生成字符串然后复制它,字符串对象生命周期在对象 (rapidjson::Value
) 生命周期之前结束(因此 generated_string.c_str()
不是这种情况)。
json中可能有\0,因此,char*
以null结尾的字符串也不是解决方案。
所以,我必须写我自己的字符串类型?或者使用类似
auto create_string_object() -> rapidjson::GenericStringRef<char>
{
size_t len;
char* result;
// generate char* result with length len
const char* return_value = result;
return rapidjson::StringRef(return_value,len);
}
auto create_object_object(rapidjson::Document::AllocatorType &allocator) -> rapidjson::Value
{
// ...
rapidjson::Value result(rapidjson::kObjectType);
rapidjson::Value tmp; // tmp = create_string_object() will not compile
tmp = create_string_object();
result.AddMember("key", tmp, allocator);
// ...
}
或者还有其他一些处理字符串的方法?这对我来说似乎很难。我们不能将字符串移动到 rapidjson::Value
因为 Value
内部的结构不同,我们不能将 Value
内部的指针设置为 c_str()
因为字符串将在 [= 之前被销毁18=]。
即使使用 GenericStringRef<char>
我也必须重写几乎所有使用字符串的工作。
对了,为什么RAPIDJSON_HAS_STDSTRING
默认是0?工作有问题?我看到如果我知道字符串生命周期将在值生命周期之前结束,我可以将字符串复制到 rapidjson::Value
并复制指针。
UPD: 现在我看到 rapidjson 只释放被复制的字符串:
//! Destructor.
/*! Need to destruct elements of array, members of object, or copy-string.
*/
~GenericValue() {
if (Allocator::kNeedFree) { // Shortcut by Allocator's trait
switch(flags_) {
case kArrayFlag:
for (GenericValue* v = data_.a.elements; v != data_.a.elements + data_.a.size; ++v)
v->~GenericValue();
Allocator::Free(data_.a.elements);
break;
case kObjectFlag:
for (MemberIterator m = MemberBegin(); m != MemberEnd(); ++m)
m->~Member();
Allocator::Free(data_.o.members);
break;
case kCopyStringFlag:
Allocator::Free(const_cast<Ch*>(data_.s.str));
break;
default:
break; // Do nothing for other types.
}
}
}
因此,正如在回答中所说,以我的代码中的方式使用 GenericStringRef
是一个坏主意,因为在这种情况下我必须自己管理内存。
我没有完全理解这个问题。但我想在这里澄清一些事情。
GenericStringRef
用于防止字符串复制。仅当已知字符串的生命周期足够时才应使用它。对于动态创建的字符串,通常不应该使用GenericStringRef
.- 设置
RAPIDJSON_HAS_STDSTRING=1
即可。默认情况下不打开,因为它的支持是在早期发布后添加的。如果用户不需要,我不希望 RapidJSON header 包含<string>
。既然你用的是std::string
,你就可以开启。它应该让你更容易处理std::string
.