- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
The perror() function shall not change the orientation of the standard error stream.
This是 GNU libc 中 perror()
的实现。
以下是在调用 perror()
之前,当 stderr 是面向宽、面向多字节和不面向时的测试。测试 1) 和 2) 正常。问题在测试 3 中)。
1) stderr
是面向宽的:
#include <stdio.h>
#include <wchar.h>
#include <errno.h>
int main(void)
{
fwide(stderr, 1);
errno = EINVAL;
perror("");
int x = fwide(stderr, 0);
printf("fwide: %d\n",x);
return 0;
}
$ ./a.out
Invalid argument
fwide: 1
$ ./a.out 2>/dev/null
fwide: 1
2) stderr
是面向多字节的:
#include <stdio.h>
#include <wchar.h>
#include <errno.h>
int main(void)
{
fwide(stderr, -1);
errno = EINVAL;
perror("");
int x = fwide(stderr, 0);
printf("fwide: %d\n",x);
return 0;
}
$ ./a.out
Invalid argument
fwide: -1
$ ./a.out 2>/dev/null
fwide: -1
3) stderr
没有定向:
#include <stdio.h>
#include <wchar.h>
#include <errno.h>
int main(void)
{
printf("initial fwide: %d\n", fwide(stderr, 0));
errno = EINVAL;
perror("");
int x = fwide(stderr, 0);
printf("fwide: %d\n", x);
return 0;
}
$ ./a.out
initial fwide: 0
Invalid argument
fwide: 0
$ ./a.out 2>/dev/null
initial fwide: 0
fwide: -1
如果重定向,为什么 perror()
会改变流的方向?这是正确的行为吗?
如何this代码工作?这个 __dup
技巧到底是什么?
最佳答案
TL;DR:是的,这是 glibc 中的错误。如果你关心它,你应该报告它。
perror
不改变流方向的引用要求在 Posix 中,但 C 标准本身似乎没有要求。然而,Posix 似乎非常坚持 stderr
的方向不会被 perror
改变,即使 stderr
还没有定向。 XSH 2.5 Standard I/O Streams:
The perror(), psiginfo(), and psignal() functions shall behave as described above for the byte output functions if the stream is already byte-oriented, and shall behave as described above for the wide-character output functions if the stream is already wide-oriented. If the stream has no orientation, they shall behave as described for the byte output functions except that they shall not change the orientation of the stream.
并且 glibc 尝试实现 Posix 语义。不幸的是,它并没有完全正确。
当然,如果不设置方向就不可能写入流。因此,为了满足这个奇怪的要求,glibc 尝试使用 OP 末尾指向的代码,基于与 stderr
相同的 fd 创建一个新流:
58 if (__builtin_expect (_IO_fwide (stderr, 0) != 0, 1)
59 || (fd = __fileno (stderr)) == -1
60 || (fd = __dup (fd)) == -1
61 || (fp = fdopen (fd, "w+")) == NULL)
62 { ...
去掉内部符号,本质上等同于:
if (fwide (stderr, 0) != 0
|| (fd = fileno (stderr)) == -1
|| (fd = dup (fd)) == -1
|| (fp = fdopen (fd, "w+")) == NULL)
{
/* Either stderr has an orientation or the duplication failed,
* so just write to stderr
*/
if (fd != -1) close(fd);
perror_internal(stderr, s, errnum);
}
else
{
/* Write the message to fp instead of stderr */
perror_internal(fp, s, errnum);
fclose(fp);
}
fileno
从标准 C 库流中提取 fd。 dup
接受一个 fd,复制它,并返回副本的编号。 fdopen
从一个 fd 创建一个标准的 C 库流。简而言之,这不会重新打开 stderr
;相反,它创建(或尝试创建)stderr
的副本,可以写入该副本而不影响 stderr
的方向。
不幸的是,由于模式的原因,它不能可靠地工作:
fp = fdopen(fd, "w+");
尝试打开一个允许读写的流。它将与原始的 stderr
一起工作,它只是控制台 fd 的一个副本,最初是为读写而打开的。但是,当您使用重定向将 stderr 绑定(bind)到其他设备时:
$ ./a.out 2>/dev/null
您正在传递一个只为输出而打开的 fd 的可执行文件。和 fdopen
不会让你逃脱的:
The application shall ensure that the mode of the stream as expressed by the mode argument is allowed by the file access mode of the open file description to which fildes refers.
fdopen
的 glibc 实现实际上会检查,如果您指定的模式需要访问权限而不是可供 fd 使用。
因此,如果您重定向 stderr 以进行读写,您就可以通过测试:
$ ./a.out 2<>/dev/null
但是您首先可能想要的是在追加模式下重定向 stderr
:
$ ./a.out 2>>/dev/null
据我所知,bash 不提供读取/附加重定向的方法。
我不知道为什么 glibc 代码使用 "w+"
作为模式参数,因为它无意从 stderr
读取。 "w"
应该可以正常工作,尽管它可能不会保留追加模式,这可能会产生不幸的后果。
关于file-io - 为什么 perror() 在重定向时改变流的方向?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39950678/
我正在尝试使用谷歌浏览器的 Trace Event Profiling Tool分析我正在运行的 Node.js 应用程序。选择点样本后,我可以在三种 View 之间进行选择: 自上而下(树) 自上而
对于一个可能是菜鸟的问题,我们深表歉意,但尽管在 SO 上研究了大量教程和其他问题,但仍找不到答案。 我想做的很简单:显示一个包含大量数据库存储字符串的 Android ListView。我所说的“很
我已经开始了一个新元素的工作,并决定给 Foundation 5 一个 bash,看看它是什么样的。在创建带有水平字段的表单时,我在文档中注意到的第一件事是它们使用大量 div 来设置样式。所以我在下
我有一个 Windows 窗体用户控件,其中包含一个使用 BeginInvoke 委托(delegate)调用从单独线程更新的第 3 方图像显示控件。 在繁重的 CPU 负载下,UI 会锁定。当我附加
我有一堆严重依赖dom元素的JS代码。我目前使用的测试解决方案依赖于 Selenium ,但 AFAIK 无法正确评估 js 错误(addScript 错误不会导致您的测试失败,而 getEval 会
我正在制作一款基于滚动 2D map /图 block 的游戏。每个图 block (存储为图 block [21][11] - 每个 map 总共 231 个图 block )最多可以包含 21 个
考虑到以下情况,我是前端初学者: 某个 HTML 页面应该包含一个沉重的图像(例如 - 动画 gif),但我不想强制客户缓慢地等待它完全下载才能享受一个漂亮的页面,而是我更愿意给他看一个轻量级图像(例
我正在设计一个小软件,其中包括: 在互联网上获取资源, 一些用户交互(资源的快速编辑), 一些处理。 我想使用许多资源(它们都列在列表中)来这样做。每个都独立于其他。由于编辑部分很累,我想让用户(可能
我想比较两个理论场景。为了问题的目的,我简化了案例。但基本上它是您典型的生产者消费者场景。 (我关注的是消费者)。 我有一个很大的Queue dataQueue我必须将其传输给多个客户端。 那么让我们
我有一个二元分类问题,标签 0 和 1(少数)存在巨大不平衡。由于测试集带有标签 1 的行太少,因此我将训练测试设置为至少 70-30 或 60-40,因此仍然有重要的观察结果。由于我没有过多地衡量准
我是一名优秀的程序员,十分优秀!