- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
一段时间以来,我一直在阅读有关严格别名规则的内容,但我开始感到非常困惑。首先,我已经阅读了这些问题和一些答案:
根据他们的说法(据我所知),使用指向另一种类型的指针访问 char
缓冲区违反了严格的别名规则。但是,strlen()
的 glibc 实现有这样的代码(删除了注释和 64 位实现):
size_t strlen(const char *str)
{
const char *char_ptr;
const unsigned long int *longword_ptr;
unsigned long int longword, magic_bits, himagic, lomagic;
for (char_ptr = str; ((unsigned long int) char_ptr
& (sizeof (longword) - 1)) != 0; ++char_ptr)
if (*char_ptr == '\0')
return char_ptr - str;
longword_ptr = (unsigned long int *) char_ptr;
himagic = 0x80808080L;
lomagic = 0x01010101L;
for (;;)
{
longword = *longword_ptr++;
if (((longword - lomagic) & himagic) != 0)
{
const char *cp = (const char *) (longword_ptr - 1);
if (cp[0] == 0)
return cp - str;
if (cp[1] == 0)
return cp - str + 1;
if (cp[2] == 0)
return cp - str + 2;
if (cp[3] == 0)
return cp - str + 3;
}
}
}
longword_ptr = (unsigned long int *) char_ptr;
行显然将 unsigned long int
别名为 char
。我不明白是什么让这成为可能。我看到代码处理对齐问题,所以没有问题,但我认为这与严格的别名规则无关。
第三个链接问题的公认答案是:
However, there is a very common compiler extension allowing you to cast properly aligned pointers from char to other types and access them, however this is non-standard.
我唯一想到的是 -fno-strict-aliasing
选项,是这样吗?我无法在 glibc 实现者所依赖的任何地方找到它的记录,并且评论以某种方式暗示这个转换是在没有任何顾虑的情况下完成的,就像很明显不会有任何问题一样。这让我觉得这确实很明显,我遗漏了一些愚蠢的东西,但我的搜索失败了。
最佳答案
在 ISO C 中,此代码将违反严格的别名规则。 (并且还违反了不能定义与标准库函数同名的函数的规则)。然而,这段代码不受 ISO C 规则的约束。标准库甚至不必用类 C 语言实现。该标准仅指定实现实现标准功能的行为。
在这种情况下,我们可以说实现是在类似 C 的 GNU 方言中,如果代码是使用作者预期的编译器和设置编译的,那么它将成功实现标准库函数。
关于c - glibc 的严格别名规则和 strlen 实现,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44446659/
有一个 strlen 和一个 wcslen 函数,但是有一个模板字符数组长度函数,所以你可以做一些类似 strlen 的事情或 strlen ? 如果没有,那我想我会自己写。 最佳答案 你有 std:
我目前正在编写一个需要频繁比较字符串长度的 C 程序,所以我编写了以下帮助函数: int strlonger(char *s1, char *s2) { return strlen(s1) -
我有一些代码获取一个文件,将每一行读入一个新的字符串数组(并向每个字符添加 128),然后将每个数组分配给一个指针数组,然后打印每个数组。尝试运行代码时,我收到一条错误消息,指出由于以下原因导致的段错
假设我有一个大小相同的字符串数组。 char strings[][MAX_LENGTH]; strlen(strings) 和 strlen(*strings) 之间有什么区别? 我知道 string
我不知道是什么原因导致这个问题...感谢任何帮助!我已经尝试了很多 strlen 代码,但这是唯一一个我可以实现且只有 1 个错误的代码。使用此代码,我尝试从文件中读取字符串,将其分解为由空格分隔的单
我有这个代码: int main() { char ch[15]; cout<
所以我正在学习嵌入式系统类(class),我们正在使用 C 语言。现在是第 2 周,所以我们只是刷新我们的 C 代码内存。 这段代码是如何打印出数字 6 的?幕后发生了什么? int main (vo
这个问题在这里已经有了答案: What do the parentheses around a function name mean? (3 个答案) 关闭 8 年前。 在 bstrlib.c(bs
编码 strlen($a); } “正确”的解决方案,在所有版本中都能正常工作(至少自从引入了飞船运算符(operator)以来)。 https://3v4l.org/6XRYW 关于php - P
我说strlen没用出于效率目的。因为如果你使用strlen那么你已经迭代了一个字符串,并且最好的算法总是迭代给定的容器不超过一次。所以请帮助我思考如何实现一个功能 bool contains ( c
这个问题已经有答案了: error: conflicting types for built-in function ‘tolower’ [-Werror] (2 个回答) 已关闭 4 年前。 我正在
我很困惑。有什么区别: char *someFunction(char *src) { char str[strlen(src) + 1]; ... return str; }
我有以下来自数据库的字符串:Let's Get Functional 如果我通过 strlen 运行它,它会返回 25 个字符而不是预期的 20 个字符。var 转储显示字符串看起来像上面的字符串(没
我正在使用 C 字符串库的 strlen 函数。我传递了一个 NULL字符串并找到神秘的结果。我知道我不应该传递 NULL 字符串,但我需要一个解释。代码看起来像这样 main() { int k
此代码返回 n=11,第 10 个和第 11 个字符为 ' ' 和 '@' 这是如何运作的? strlen函数怎么把它当成11个字符?在某些编译器中,它似乎将字符串长度设为 12 个字符。 #incl
以下代码能够确定 DWORD 的一个或多个字节是否设置为 0。 mov eax, value mov edx, 07EFEFEFFh add edx, eax xor eax, 0FFFFFFFFh
我在这里找到解决方案时遇到问题。我正在为使用 for() 的客户开发 WordPress 主题。循环遍历页面标题,以便将其包装在 中s 并垂直显示.. 循环使用 strlen()找到标题的长度,但由
我在理解 strlen 和/或 memcpy 时遇到问题。这是片段: char * restP; char * str; //this returns a pointer restP = strrst
好的,我正在检查一个字符串是否至少有 4 个字符长且至少有 25 个字符短 我试过这样使用strlen $userNameSignupLength = strlen($userNameSignup);
#include #include #include int main(void) { char qq[] = {'a' , 'b' , 'c' , 'd'}; char qqq
我是一名优秀的程序员,十分优秀!