C++ Iterating Over va_list
C++ Iterating Over va_list
我正在尝试编写一个类似于 C# 的 String::Format 的函数,其中不是采用以"%"开头的参数("%d %s %i"(,而是采用像"{0} {1} {2}"这样的参数,我已经让它工作了(大部分(。
它正确查找并替换所有匹配项,但在到达 args 末尾时中断。在中断之前,调试器显示"result"设置为"\f;@",其中最后一个字符是随机的"非标准"字符。
注意**:"字符串"是 std::string,字符串::Format1 工作正常并使用vsnprintf_s,字符串::替换查找并将所有出现的查找替换为替换。
string String::Format2(const string format, ...)
{
string output = format;
va_list args;
va_start(args, format);
{
uint i = 0;
while (args[i] != NULL)
{
string find = String::Format1("{%i}", i);
// Breaks here
string replace = va_arg(args, const char*);
output = String::Replace(output, find, replace);
i++;
}
}
va_end(args);
return output;
}
感谢@Justin的评论,让我朝着正确的方向前进。
我使用了一个可变参数模板,它似乎可以:)
在这里找到我的答案
template<typename T, typename ... Args>
static string Format3(const string fmt, const T& first, const Args&... args)
{
std::stringstream stream;
stream << first << std::endl;
int sink[] = { 0, ((void)(stream << args << std::endl), 0)... };
(void)sink;
string out = fmt;
uint pos = 0;
while (stream.good())
{
string find = String::Format1("{%i}", pos);
string replace;
stream >> replace;
out = String::Replace(out, find, replace);
pos++;
}
return out;
}
我建议不要将varargs用于格式化API,因为它们本质上是不安全的。请改用可变参数模板。
fmt 库使用大括号分隔的占位符({0}
、{1}
、...(实现文本格式设置,它使用可变参数模板和数组初始化来构建参数数组。以下是其工作原理的草图:
template <typename... Args>
std::string format(std::string_view format_str, const Args&... args) {
Arg array[] = {args...};
// Format arguments and return a string.
}
其中Arg
是一个类似变体的类,可以从格式化参数类型隐式转换:
class Arg {
public:
Arg(int value);
Arg(double value);
// ...
};
这种方法的优点是,与递归模板 (https://github.com/fmtlib/fmt/pull/243( 相比,它具有更好的编译时间。此外,您还可以在array
中按索引轻松访问参数。
免责声明:我是 fmt 库的作者。
相关文章:
- 使用std::multimap迭代器创建std::list
- 来自 std::list 的迭代器 .end() 按预期返回"0xcdcdcdcdcdcdcdcd"但 .begin()
- 如何在 C 中正确使用 libiconv 使其不会报告"Arg list too long"?
- C++中带有List类的迭代器Segfault
- 使用"std::unordereded_map"映射到"std::list"对象
- GCC对可能有效的代码抛出init list生存期警告
- 使用std::list创建循环链表
- 重载Singly Linked List中的赋值运算符
- '[](std::list& list)<int>{return std::move(list)}(list)' 是否保证将 'list' 留空?
- 如果 KEY 是 std::list 或 std::vector 而不是值,那么 std::map 的默认行为是什么?
- 为什么这个 std::queue/指向结构的指针列表直到 List.Size() == 0 才释放内存?
- "std::list::splice(std::const_iterator pos, std::list&& other)"是否保证将"其他"留空?
- 从嵌套循环中的 std::list 中删除将返回访问冲突
- QStringList vs list<shared_ptr<QString>> 性能比较C++
- 包含 std::list 的结构体的 C++ 初始化
- 在C++中使用 Catch 测试框架编译错误"error: expected ';' at end of declaration list"
- C++ assigment std::list:<typename>:itrator 在 main 中工作,但在方法中它不起作用
- 在基于范围的 for 循环期间插入 std::list 的后面
- 循环挂起迭代的 std::擦除 on std::list
- 如何增加以前由新运算符分配的 C++ std::list 数组的大小?