- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我不确定我是否选择了正确的标题,但今天我发现(作为 C 初学者),对我来说,strlen 并不总是我需要时做出的正确决定。
所以我尝试了以下方法:
#include<stdio.h>
#include<string.h>
int foo(char *s){
int len = strlen(s);
/* code here */
return len;
}
int main(void){
char *name = "Michi";
int len = foo(name);
int a = 20, b = 10, c = a - b;
if(c < len){
printf("True: C(%d) < Len(%d)\n",c,len);
}else{
printf("False: C(%d) > Len(%d)\n",c,len);
}
return 0;
}
输出:
False: C(10) > Len(5)
但是当我用“-Wconversion”编译时,我得到:
program.c:5:19: warning: conversion to ‘int’ from ‘size_t’ may alter its value [-Wconversion]
int len = strlen(s);
^
一个快速修复方法是强制转换 strlen:
int len = (int)strlen(s);
但我不同意,所以我决定我真的需要别的东西,也许另一种方法?我尝试了以下方法:
#include<stdio.h>
#include<string.h>
unsigned int size(char *s){
unsigned int len;
/* code here */
len = (unsigned int)strlen(s);
return len;
}
int main(void){
char *name = "Michi";
unsigned int len = size(name);
int a = 20, b = 10, c = a - b;
if(c < (signed int)len){
printf("True: C(%d) < Len(%d)\n",c,len);
}else{
printf("False: C(%d) > Len(%d)\n",c,len);
}
return 0;
}
但我仍然需要强制转换 strlen,因为它的返回类型(size_t,我知道它是无符号类型(typedef long unsigned int size_t;))
最后我决定采用另一种方法,创建我自己的函数,这使事情变得更容易并且 future 可能出现的问题更少,我得到了:
#include<stdio.h>
long int stringLEN(char *s){
int i = 0;
long int len = 0;
while (s[i] != '\0'){
len++;
i++;
}
return len;
}
long int foo(char *s){
long int len = stringLEN(s);
/* code here */
return len;
}
int main(void){
char *name = "Michi";
long int len = foo(name);
int a = 20, b = 10, c = a - b;
if(c < len){
printf("True: C(%d) < Len(%ld)\n",c,len);
}else{
printf("False: C(%d) > Len(%ld)\n",c,len);
}
return 0;
}
不再需要 Actor 。
所以我的问题是:这是(对于我的情况)更好的方法吗?如果不是,我需要一些解释,我的书(我有 3 本书)并没有以我能理解这些事情的方式解释我。
我只知道在某种程度上 Actor 可能会成为一个大问题。
编辑:此代码也无法使用 -Wconversion 进行编译:
#include<stdio.h>
#include<string.h>
size_t foo(char *s){
size_t len = strlen(s);
/* code here */
return len;
}
int main(void){
char *name = "Michi";
size_t len = foo(name);
int a = 20, b = 10, c = a - b;
if(c < len){
printf("True: C(%d) < Len(%zu)\n",c,len);
}else{
printf("False: C(%d) > Len(%zu)\n",c,len);
}
return 0;
}
输出:
error: comparison between signed and unsigned integer expressions [-Werror=sign-compare]|
但是如果我投 len 作品。我意识到,如果尺寸大于 int 它永远不会适合。
最佳答案
深入研究所有其他答案,您的真正问题似乎是如何处理这样的情况:
#include <string.h>
#include <libfoo.h>
extern void foo(void);
extern void bar(void);
void pick_foo_or_bar(const char *s)
{
size_t slen = strlen(s);
int value = libfoo_api_returning_an_int();
if (slen > value) // -Wconversion warning on this line
foo();
else
bar();
}
...您无法更改 slen
的类型或value
,因为对于它们接收结果的 API 来说,两者都是正确的。
-Wconversion
warning 试图告诉你一些有意义的事情。 C 中有符号和无符号整数类型的比较会产生一些非常奇怪的结果,不是您从 ℤ 中的算术定律中所期望的结果;像我上面写的那样的天真的比较可能并且已经导致了灾难性的错误。但治愈方法不是类型转换或发明自己的strlen
;解决方法是修复比较,使其达到您对算术定律的期望。其原则是:
在本例中,size_t
几乎肯定大于或等于 int
,所以你会写
#include <assert.h>
#include <limits.h>
#include <string.h>
#include <libfoo.h>
extern void foo(void);
extern void bar(void);
// Code below is correct only if size_t is at least as large as int.
static_assert(SIZE_MAX >= INT_MAX);
void pick_foo_or_bar(const char *s)
{
size_t slen = strlen(s);
int value = libfoo_api_returning_an_int();
if (value < 0 || (size_t)value < slen)
foo();
else
bar();
}
static_assert
存在是因为,如果我没记错的话,C 标准不保证 size_t
至少与unsigned int
一样大。例如,我可以想象 80286 的 ABI,其中 int
是四个字节宽,但是 size_t
只有两个。在这种情况下,您需要以相反的方式进行转换:
void pick_foo_or_bar(unsigned short a, long b)
{
if (b < 0 || b < (long)a)
foo();
else
bar();
}
如果您不知道这两种类型中哪一个更大,或者您不知道其中哪一个有符号,则标准 C 中您唯一的资源是 (u)intmax_t
:
void pick_foo_or_bar(uid_t a, gid_t b)
{
if (a < 0 && b < 0) {
if ((intmax_t)a < (intmax_t)b)
bar();
else
foo();
} else if (a < 0) {
bar();
} else if (b < 0) {
foo();
} else {
if ((uintmax_t)a < (uintmax_t)b)
bar();
else
foo();
}
}
...而且,考虑到 C99 wrt long
设定的极其不幸的先例,可能会有一天(u)intmax_t
不是编译器支持的最大整数类型,那么你就完蛋了。
关于c - 使用 strlen 并不总是一个好主意,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32417194/
有一个 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
我是一名优秀的程序员,十分优秀!