- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
在 C++ 中解析由流中的整数序列组成的文本非常简单:只需解码它们即可。当数据以某种方式接收并且在程序中很容易获得时,例如,接收 base64 编码的文本(解码不是问题),情况有点不同。数据位于程序的缓冲区中,只需要解码,不需要读取。当然,可以使用 std::istringstream
:
std::vector<int> parse_text(char* begin, char* end) {
std::istringstream in(std::string(begin, end));
return std::vector<int>(std::istream_iterator<int>(in),
std::istream_iterator<int>());
}
由于收到了很多这样的缓冲区,而且它们可能相当大,因此最好不复制字符数组的实际内容,并且最好避免为每个缓冲区创建一个流.于是,问题就变成了:
给定一个 char
缓冲区,其中包含一系列(空格分隔;处理其他分隔符很容易完成,例如,使用合适的操纵器)整数如何在不复制序列的情况下解码它们,并且,如果可能,甚至不创建 std::istream
?
最佳答案
使用自定义流缓冲区可以轻松避免缓冲区的拷贝,该缓冲区只需设置获取区域即可使用缓冲区。流缓冲区实际上甚至不需要覆盖任何虚函数,只需设置内部缓冲区即可:
class imemstream
: private virtual std::streambuf
, public std::istream
{
public:
imemstream(char* begin, char* end)
: std::streambuf()
, std::istream(static_cast<std::streambuf*>(this))
{
this->setg(begin, begin, end);
}
};
std::vector<int> parse_data_via_istream(char* begin, char* end)
{
imemstream in(begin, end);
return std::vector<int>(std::istream_iterator<int>(in),
std::istream_iterator<int>());
}
这种方法避免了复制流并使用现成的 std::istream
功能。但是,它确实创建了一个流对象。通过适当的更新功能,可以扩展流流/流缓冲区以重置缓冲区并处理多个缓冲区。
为了避免创建流,底层功能来自 std::num_get<...>
可用于。实际解析由 std::locale
之一完成方面。 std::istream
的数值解析由 std::num_get<char, std::istreambuf_iterator<char>>
完成.这个方面没有多大帮助,因为它使用由 std::istreambuf_iterator<char>
指定的序列只是一个 std::num_get<char, char const*>
facet可以被实例化。它不会属于默认 std::locale
的一部分但很容易创建相应的 std::locale
并安装它,例如,作为全局 std::locale
main()
中的第一件事:
int main()
{
std::locale::global(std::locale(std::locale(),
new std::num_get<char, char const*>()));
...
请注意 std::locale
object 将清理添加的 facet,即不需要添加任何清理代码:facet 被引用计数并在最后一个 std::locale
时释放。持有特定方面消失。不幸的是,要实际使用它,需要一个 std::ios_base
。只能从某个流对象中真正获得的对象。但是,可以使用任何流(尽管在多线程系统中,每个流可能应该是一个单独的流对象,以避免意外的竞争条件):
char const* skipspace(char const* it, char const* end)
{
return std::find_if(it, end,
[](unsigned char c){ return !std::isspace(c); });
}
std::vector<int> parse_data_via_istream(std::ios_base& fmt,
char const* it, char const* end)
{
std::vector<int> rc;
std::num_get<char, char const*> const& ng
= std::use_facet<std::num_get<char, char const*>>(std::locale());
std::ios_base::iostate error;
for (long tmp;
(it = ng.get(skipspace(it, end), end, fmt, error, tmp))
, error == std::ios_base::goodbit; ) {
rc.push_back(tmp);
}
return rc;
}
其中大部分只是一些错误处理和跳过前导空格:主要是 std::istream
提供自动跳过格式化输入空白的工具,并处理必要的错误协议(protocol)。上面概述的方法在每个缓冲区仅获取一次分面并避免创建 std::istream::sentry
方面可能有一个小优势。对象以及避免创建流。当然,代码假设可以使用一些流将其作为 std::ios_base&
传递进来。子对象提供解析标志,例如要使用的基数。
好的,这是相当多的代码,用于 strtol()
的东西大多数情况下也可以。使用 std::num_get<char, char const*>
的方法具有 strtol()
不提供的一些灵 active :
std::locale
的 facet 可以被覆盖以解析任意表示格式,例如罗马数字,它在输入格式方面更加灵活。std::numpunct<char>
使用的 std::locale
中的 fmt
即可设置这些)。it
来解析由 8 个数字值组成的连续字符序列。和 it+8
作为调用 std::num_get<char, char const*>::get()
时的范围.然而,strtol()
对于大多数用途来说,这可能是一种好方法。另一方面,上述内容提供了一种可能在某些情况下有用的替代方法。
关于c++ - 如何解析存储在文本缓冲区中的整数序列?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23036783/
这个问题在这里已经有了答案: Possible to make an event handler wait until async / Promise-based code is done? (2
我经常有多个运行的进程(R,Python,eshell/shell),对于每个进程,我经常都有一个相关的脚本,可以从中发送摘要。为此,我通常将每个框架垂直地分成两个窗口,以便脚本文件(例如.py)位于
如何修改 emacs 在关闭缓冲区后选择要显示的缓冲区的方式? 当我有多个列显示相同的缓冲区,然后在其中一个缓冲区中打开另一个文件,然后关闭新打开的缓冲区时,它不会切换回前一个缓冲区,而是切换到另一个
如何将 ex 命令复制到剪贴板或粘贴到缓冲区? 在 Windows 上使用 gvim。 最佳答案 windows剪贴板可以通过the buffer + 访问.因此,可以使用 + 将剪贴板粘贴为前命令。
在 javascript 中如何以比以下更简单的方式获取 b 缓冲区? var num=6553599 var a = new Buffer(4); a.writeInt32LE(num)
每次我在 Google 上搜索有关 OpenGL 编程的文章时,我都会找到一些文章,但似乎所有文章都提到了着色器和缓冲区。那些是什么?你能解释其中的一些吗: 深度缓冲区 模板缓冲区 像素着色器 帧缓冲
我有java考试,当我学习时,我看到了这个练习,我尝试解决它,但我发现一些困难,所以请帮助我考虑实用程序中方法的以下注释、 header 和部分代码名为 Atbash 的加密类。 /**
每次我在 Google 上搜索有关 OpenGL 编程的文章时,我都会找到一些文章,但似乎所有文章都提到了着色器和缓冲区。那些是什么?你能解释其中的一些吗: 深度缓冲区 模板缓冲区 像素着色器 帧缓冲
对于每个属性使用跨步顶点缓冲区与紧密打包缓冲区有何优缺点?我的意思是例如: 步幅:xyzrgb xyzrgb xyzrgb 紧:xyzxyzxyz rgbrgbrgb 乍一看,使用步幅时您似乎可以轻松
我正在尝试将文本文件中每行的数字读取到 ArrayList 中。当我执行以下函数时,它总是跳过最后一个元素。有人可以帮我吗?因为我在这里没有遇到问题,因为它读取直到缓冲区为空,所以他应该在到达 Fil
#include #include int main () { time_t time_raw_format; struct tm * ptr_time; char *buff
基本上我有一个包含不同类型数据的自定义结构。例如: typedef struct example_structure{ uint8_t* example_1[4]; int example_2[4];
我之前的列表实现是一个简单的 LinearLayout,位于一个装满我的项目的 ScrollView 中。 我切换到 ListView 的 Android 实现以简单地使用 CursorAdapter
我想创建一个可变长度的输入事件窗口/缓冲区,当它接收到额外的事件时会变长。 这是为了实现“键入时搜索”功能。我想捕获点击,但为了不给服务器造成压力,我想明智地进行服务调用。 我想到的逻辑是缓冲击键,从
我想将 yuv420P 像素写入缓冲区而不是二进制文件。假设我在指针中存储了 luma 、 Cb 和 Cr。 luma = output_pixel.luma; cb = output_pixel.c
我想在 Go 中构建一个支持多个并发读取器和一个写入器的缓冲区。所有写入缓冲区的内容都应由所有读者读取。允许新读者随时加入,这意味着已经写入的数据必须能够为迟到的读者回放。 缓冲区应满足以下接口(in
本文转载自微信公众号「小明菜市场」,作者小明菜市场。转载本文请联系小明菜市场公众号。 前言 Java NIO 需要理解的主要有缓冲区,通道,选择器,这三个主要的部分。 基础
一 点睛 NIO,可以称为 New IO 或 Non Blocking IO,是在 JDK 1.4 后提供的新 API。传统的I/O 是阻塞式的 I/O、面向流的操作;而 NIO 是非阻塞 I/O 、
我正在寻找一种切换到包含搜索文本的缓冲区的方法。 例如。如果我打开了 100 个缓冲区,我想切换到一个包含 'fooBar = 1' 的缓冲区 最佳答案 我写了一个 Vim 插件来做到这一点:buff
我正在尝试将提取的视频帧(我使用 ffmpeg)推送到 FFMPEG 缓冲区中。我已经查看了 ffmpeg 的缓冲区源文件,例如 buffersrc.c 和 fifo.c,以确定我是否可以这样做,但我
我是一名优秀的程序员,十分优秀!