- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在测试一段代码,我不明白为什么它会打印“较高的值为 0”。这是主要功能:
int main() {
double a,b,c;
a=576;
b=955;
c=higher(a,b);
printf("The higher value is %g\n", c);
return 0;
}
在另一个 .c 中我有这个函数:
double higher(double a, double b){
if (a>b)
return a;
return b;
}
注意:如果我将 higher() 放在 main.c 中,它可以正常工作,但那样它会告诉我较高的值为 0。如果我像这样将 return 转换到 higer() 中,它也可以工作:
return (int)b;
返回 double 的函数与 main() 在同一个 .c 中还是在不同的 .c 中有区别吗?
最佳答案
使用 C99 或 C11 编译器编译并阅读警告。您正在使用没有原型(prototype)的函数。
在没有原型(prototype)的情况下,C99 之前的版本假设一个函数默认返回 int
。C99 及更高版本,需要原型(prototype)。
即使没有启用额外的警告:
$ cat test.c
int main()
{
int i = f();
return 0;
}
int f(void)
{
return 1;
}
$ gcc -std=c11 test.c
test.c: In function ‘main’:
test.c:13:2: warning: implicit declaration of function ‘f’ [-Wimplicit-function-declaration]
int i = f();
请注意,如果编译 -std=c90
,gcc 不会发出警告,但如果启用警告 -Wall
,则会发出警告。
因此,由于 higher()
预期返回一个 int
,该值通过赋值(的类型)转换为 double
c
没有改变)。
现在是有趣的部分:undefined behaviour (UB,记住这个短语!)由于函数调用和实现的签名不同。
可能发生的事情是根据过程调用标准 (PCS) 和应用程序二进制接口(interface) (ABI) - 检查维基百科。简而言之:higher
本身返回一个 double
。这很可能在浮点 CPU 寄存器中传递给调用者。调用者 OTOH 期望返回值(由于缺少原型(prototype))在一个整数 CPU 寄存器中(碰巧保存 0
)。
因此,由于他们显然沟通不畅,您得到了错误的结果。请注意,这有点推测性并且取决于 PCS/ABI。要记住的是这是 UB,所以任何事情都有可能发生,即使是 demons flying out of your nose .
为什么要使用原型(prototype):
好吧,你已经注意到了,编译器不知道你是否正确地调用了一个函数。更糟糕的是,它不知道使用了哪些参数类型以及返回了哪种结果类型。这是一个特别的问题,因为 C 会自动转换某些类型(您在这里确实遇到过)。
由于经典的 K&R(标准前)C 没有原型(prototype),所有未知函数的参数都假定为 int
/double
用于调用中的标量参数。结果默认为 int
。 (很久以前,我可能遗漏了一些部分;我开始使用 K&R 进行一些编码,弄乱了类型(这正是您的问题,但没有一个干净的解决方案),等等,把它扔在角落里并愉快地在 Modula-2 中编程直到几年后我尝试了 ANSI-C)。
如果现在编译代码,您应该至少符合(并针对)C99 进行编译,最好使用当前标准 (C11)。
关于c - 为什么这个函数返回 0 而不是 double?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31196236/
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!