在 C 中检查自己的 pow 函数的限制?
Checking limits of own pow function in C?
所以我正在尝试实现我自己的 pow 函数。为了使事情更简单一些,我被允许使用 int 类型作为指数。所以代码正在工作......主要是。我有一个名为 xmath.c 的文件,我在其中实现我的 pow 函数和另一个 test.c 来测试限制。唯一不起作用的情况是 pow(-111, INT_MAX)
。我怎样才能改进我的代码?还要告诉我是否也应该 post test.c 代码。 xfabs
是我必须自己实现的另一个功能,而不是使用 fabs
。
忘记在我的 post 中包含 xfabs:
#define xfabs(x) (((x) > 0) ? (x) : (-(x)))
#include <stdio.h>
#include <errno.h>
#include <math.h>
#include <float.h>
#include "xmath.h"
double xpow(double x, int y)
{
errno = 0;
double product = 1;
double prod = 1;
int i;
if (y == 0) {
return 1.0;
} else if (x == 0 && y < 0) {
errno = EDOM;
return 0.0;
}
if (xfabs(x) >= 1) {
if (y > 0) {
for (i = 0; i < y; i++) {
if (xfabs(product) > DBL_MAX / xfabs(x)) {
errno = ERANGE;
return HUGE_VAL;
}
product *= x;
}
return product;
} else if (y < 0) {
for (i = 0; i > y; i--) {
if (xfabs(product) < DBL_MIN / xfabs(x)) {
errno = ERANGE;
return -0.0;
}
product *= 1 / x;
}
return product;
}
} else if (xfabs(x) < 1) {
if (y > 0) {
for (i = 0; i < y; i++) {
if (xfabs(prod) < DBL_MIN / xfabs(x)) {
errno = ERANGE;
return +0.0;
}
prod *= x;
}
return prod;
} else if (y < 0) {
for (i = 0; i > y; i--) {
if (xfabs(prod) < DBL_MAX / xfabs(x)) {
errno = ERANGE;
return -HUGE_VAL;
}
prod *= 1 / x;
}
return prod;
}
}
}
这里是测试函数。此外,我正在为 xmath.c 创建一个目标文件,我在 test.c 中使用它:
#include <stdio.h>
#include <float.h>
#include <math.h>
#include <errno.h>
#include <limits.h>
#include "xmath.h"
void xpow_tests(void)
{
printf("XPOW TESTS (SUCCESS == 1 / FAILURE == 0)\n");
printf("=================================================\n");
printf("Test (xpow(2, 5)): \t\t%i\t", xpow(2, 5) == pow(2, 5));
xpow(2, 5);
printf("errno: %i\n", errno);
printf("Test (xpow(-2, -4)): \t\t%i\t", xpow(-2, -4) == pow(-2, -4));
xpow(-2, -4);
printf("errno: %i\n", errno);
printf("Test (xpow(1, 15)): \t\t%i\t", xpow(1, 15) == pow(1, 15));
xpow(1, 15);
printf("errno: %i\n", errno);
printf("Test (xpow(-4, 414)): \t\t%i\t", xpow(-4, 414) == pow(-4, 414));
xpow(-4, 414);
printf("errno: %i\n", errno);
printf("Test (xpow(-5, 303)): \t\t%i\t", xpow(-5, 303) == pow(-5, 303));
xpow(-5, 303);
printf("errno: %i\n", errno);
printf("Test (xpow(0, -3)): \t\t%i\t", xpow(0, -3) == 0);
xpow(0, -3);
printf("errno: %i\n", errno);
printf("Test (xpow(1.0e-10, 100)): \t%i\t", xpow(1.0e-10, 100) == pow(1.0e-10, 100));
xpow(1.0e-10, 100);
printf("errno: %i\n", errno);
printf("Test (xpow(-1.0e-10, 101)): \t%i\t", xpow(-1.0e-10, 101) == pow(-1.0e-10, 101));
xpow(-1.0e-10, 101);
printf("errno: %i\n", errno);
printf("Test (xpow(-111, INT_MAX)): \t%i\t", xpow(-111, INT_MAX) == pow(-111, INT_MAX));
xpow(-111, INT_MAX);
printf("errno: %i\n", errno);
printf("Test (xpow(-111, INT_MAX-1)): \t%i\t", xpow(-111, INT_MAX-1) == pow(-111, INT_MAX-1));
xpow(-111, INT_MAX-1);
printf("errno: %i\n", errno);
printf("Test (xpow(-111, -INT_MAX)): \t%i\t", xpow(-111, -INT_MAX) == pow(-111, -INT_MAX));
xpow(-111, -INT_MAX);
printf("errno: %i\n", errno);
printf("Test (xpow(-111, -INT_MAX+1)): \t%i\t", xpow(-111, -INT_MAX+1) == pow(-111, -INT_MAX+1));
xpow(-111, -INT_MAX+1);
printf("errno: %i\n", errno);
printf("=================================================\n");
printf("errno (EDOM: %i / ERANGE: %i)\n\n", EDOM, ERANGE);
}
int main(void)
{
xpow_tests();
return 0;
}
为了正确性,您应该添加特殊情况 if (y == 0 && x == 0) { errno = EDOM;
你有很多重复的代码。您应该添加 if(y<0) return 1.0/xpow(x, -y);
以摆脱这些。
代码的效率也很低,循环比必要的要长。您应该利用 x^y=(x^(y/2))^2。递归地这样做会节省很多迭代。当然,奇数处理要加点插件。
警告:这不是针对您失败的测试用例本身的解决方案。
但是,我稍微简化了您的函数 -- 这可能会有所帮助。
而且我非常 simplified/enhanced 你的测试用例函数 [使用一些预处理器技巧]。
然而,除了你失败的测试-111,INT_MAX
,当我运行它在这里时,0,-3
测试报告失败。
这是修改后的来源:
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <math.h>
#include <float.h>
#include <limits.h>
//#include "xmath.h"
double
xfabs(double x)
{
if (x < 0)
x = -x;
return x;
}
double
xpow(double x, int y)
{
errno = 0;
int i;
double lim;
double prod = 1;
double abs_x;
if (y == 0)
return 1.0;
if ((x == 0) && (y < 0)) {
errno = EDOM;
return 0.0;
}
abs_x = xfabs(x);
if (abs_x >= 1) {
if (y > 0) {
lim = DBL_MAX / abs_x;
for (i = 0; i < y; i++) {
if (xfabs(prod) > lim) {
errno = ERANGE;
return HUGE_VAL;
}
prod *= x;
}
}
if (y < 0) {
lim = DBL_MIN / abs_x;
for (i = 0; i > y; i--) {
if (xfabs(prod) < lim) {
errno = ERANGE;
return -0.0;
}
prod *= 1 / x;
}
}
return prod;
}
if (y > 0) {
lim = DBL_MIN / abs_x;
for (i = 0; i < y; i++) {
if (xfabs(prod) < lim) {
errno = ERANGE;
return +0.0;
}
prod *= x;
}
}
if (y < 0) {
lim = DBL_MAX / abs_x;
for (i = 0; i > y; i--) {
if (xfabs(prod) < lim) {
errno = ERANGE;
return -HUGE_VAL;
}
prod *= 1 / x;
}
}
return prod;
}
int passcnt = 0;
int failcnt = 0;
void
dotest(const char *name,double x,int y)
{
int sverr_pow;
int sverr_xpow;
double expected;
double actual;
int passflg;
printf("\n");
printf("%s\n",name);
errno = 0;
expected = pow(x,y);
sverr_pow = errno;
printf("%.16g (pow)\n",expected);
if (sverr_pow)
printf("errno: %d (%s)\n",sverr_pow,strerror(sverr_pow));
errno = 0;
actual = xpow(x,y);
sverr_xpow = errno;
printf("%.16g (xpow)\n",actual);
if (sverr_xpow)
printf("errno: %d (%s)\n",sverr_xpow,strerror(sverr_xpow));
if ((actual == expected) && (sverr_xpow == sverr_pow))
passcnt += 1;
else
failcnt += 1;
printf("Result: Value:%s errno:%s\n",
(actual == expected) ? "PASS" : "FAIL",
(sverr_xpow == sverr_pow) ? "PASS" : "FAIL");
}
#define TEST(x,y) \
dotest(#x "," #y,x,y)
void xpow_tests(void)
{
printf("XPOW TESTS (SUCCESS == 1 / FAILURE == 0)\n");
printf("=================================================\n");
TEST(2, 5);
TEST(-2, -4);
TEST(1, 15);
TEST(-4, 414);
TEST(-5, 303);
TEST(0, -3);
TEST(1.0e-10, 100);
TEST(-1.0e-10, 101);
TEST(-111, INT_MAX);
TEST(-111, INT_MAX-1);
TEST(-111, -INT_MAX);
TEST(-111, -INT_MAX+1);
printf("=================================================\n");
printf("errno (EDOM: %i / ERANGE: %i)\n\n", EDOM, ERANGE);
printf("PASSED: %d, FAILED: %d\n",passcnt,failcnt);
}
int
main(void)
{
xpow_tests();
return 0;
}
这是测试的输出:
XPOW TESTS (SUCCESS == 1 / FAILURE == 0)
=================================================
2,5
32 (pow)
32 (xpow)
Result: Value:PASS errno:PASS
-2,-4
0.0625 (pow)
0.0625 (xpow)
Result: Value:PASS errno:PASS
1,15
1 (pow)
1 (xpow)
Result: Value:PASS errno:PASS
-4,414
1.789931494904685e+249 (pow)
1.789931494904685e+249 (xpow)
Result: Value:PASS errno:PASS
-5,303
-6.136366831622158e+211 (pow)
-6.136366831622158e+211 (xpow)
Result: Value:PASS errno:PASS
0,-3
inf (pow)
errno: 34 (Numerical result out of range)
0 (xpow)
errno: 33 (Numerical argument out of domain)
Result: Value:FAIL errno:FAIL
1.0e-10,100
0 (pow)
errno: 34 (Numerical result out of range)
0 (xpow)
errno: 34 (Numerical result out of range)
Result: Value:PASS errno:PASS
-1.0e-10,101
-0 (pow)
errno: 34 (Numerical result out of range)
0 (xpow)
errno: 34 (Numerical result out of range)
Result: Value:PASS errno:PASS
-111,INT_MAX
-inf (pow)
errno: 34 (Numerical result out of range)
inf (xpow)
errno: 34 (Numerical result out of range)
Result: Value:FAIL errno:PASS
-111,INT_MAX-1
inf (pow)
errno: 34 (Numerical result out of range)
inf (xpow)
errno: 34 (Numerical result out of range)
Result: Value:PASS errno:PASS
-111,-INT_MAX
-0 (pow)
errno: 34 (Numerical result out of range)
-0 (xpow)
errno: 34 (Numerical result out of range)
Result: Value:PASS errno:PASS
-111,-INT_MAX+1
0 (pow)
errno: 34 (Numerical result out of range)
-0 (xpow)
errno: 34 (Numerical result out of range)
Result: Value:PASS errno:PASS
=================================================
errno (EDOM: 33 / ERANGE: 34)
PASSED: 10, FAILED: 2
所以我正在尝试实现我自己的 pow 函数。为了使事情更简单一些,我被允许使用 int 类型作为指数。所以代码正在工作......主要是。我有一个名为 xmath.c 的文件,我在其中实现我的 pow 函数和另一个 test.c 来测试限制。唯一不起作用的情况是 pow(-111, INT_MAX)
。我怎样才能改进我的代码?还要告诉我是否也应该 post test.c 代码。 xfabs
是我必须自己实现的另一个功能,而不是使用 fabs
。
忘记在我的 post 中包含 xfabs:
#define xfabs(x) (((x) > 0) ? (x) : (-(x)))
#include <stdio.h>
#include <errno.h>
#include <math.h>
#include <float.h>
#include "xmath.h"
double xpow(double x, int y)
{
errno = 0;
double product = 1;
double prod = 1;
int i;
if (y == 0) {
return 1.0;
} else if (x == 0 && y < 0) {
errno = EDOM;
return 0.0;
}
if (xfabs(x) >= 1) {
if (y > 0) {
for (i = 0; i < y; i++) {
if (xfabs(product) > DBL_MAX / xfabs(x)) {
errno = ERANGE;
return HUGE_VAL;
}
product *= x;
}
return product;
} else if (y < 0) {
for (i = 0; i > y; i--) {
if (xfabs(product) < DBL_MIN / xfabs(x)) {
errno = ERANGE;
return -0.0;
}
product *= 1 / x;
}
return product;
}
} else if (xfabs(x) < 1) {
if (y > 0) {
for (i = 0; i < y; i++) {
if (xfabs(prod) < DBL_MIN / xfabs(x)) {
errno = ERANGE;
return +0.0;
}
prod *= x;
}
return prod;
} else if (y < 0) {
for (i = 0; i > y; i--) {
if (xfabs(prod) < DBL_MAX / xfabs(x)) {
errno = ERANGE;
return -HUGE_VAL;
}
prod *= 1 / x;
}
return prod;
}
}
}
这里是测试函数。此外,我正在为 xmath.c 创建一个目标文件,我在 test.c 中使用它:
#include <stdio.h>
#include <float.h>
#include <math.h>
#include <errno.h>
#include <limits.h>
#include "xmath.h"
void xpow_tests(void)
{
printf("XPOW TESTS (SUCCESS == 1 / FAILURE == 0)\n");
printf("=================================================\n");
printf("Test (xpow(2, 5)): \t\t%i\t", xpow(2, 5) == pow(2, 5));
xpow(2, 5);
printf("errno: %i\n", errno);
printf("Test (xpow(-2, -4)): \t\t%i\t", xpow(-2, -4) == pow(-2, -4));
xpow(-2, -4);
printf("errno: %i\n", errno);
printf("Test (xpow(1, 15)): \t\t%i\t", xpow(1, 15) == pow(1, 15));
xpow(1, 15);
printf("errno: %i\n", errno);
printf("Test (xpow(-4, 414)): \t\t%i\t", xpow(-4, 414) == pow(-4, 414));
xpow(-4, 414);
printf("errno: %i\n", errno);
printf("Test (xpow(-5, 303)): \t\t%i\t", xpow(-5, 303) == pow(-5, 303));
xpow(-5, 303);
printf("errno: %i\n", errno);
printf("Test (xpow(0, -3)): \t\t%i\t", xpow(0, -3) == 0);
xpow(0, -3);
printf("errno: %i\n", errno);
printf("Test (xpow(1.0e-10, 100)): \t%i\t", xpow(1.0e-10, 100) == pow(1.0e-10, 100));
xpow(1.0e-10, 100);
printf("errno: %i\n", errno);
printf("Test (xpow(-1.0e-10, 101)): \t%i\t", xpow(-1.0e-10, 101) == pow(-1.0e-10, 101));
xpow(-1.0e-10, 101);
printf("errno: %i\n", errno);
printf("Test (xpow(-111, INT_MAX)): \t%i\t", xpow(-111, INT_MAX) == pow(-111, INT_MAX));
xpow(-111, INT_MAX);
printf("errno: %i\n", errno);
printf("Test (xpow(-111, INT_MAX-1)): \t%i\t", xpow(-111, INT_MAX-1) == pow(-111, INT_MAX-1));
xpow(-111, INT_MAX-1);
printf("errno: %i\n", errno);
printf("Test (xpow(-111, -INT_MAX)): \t%i\t", xpow(-111, -INT_MAX) == pow(-111, -INT_MAX));
xpow(-111, -INT_MAX);
printf("errno: %i\n", errno);
printf("Test (xpow(-111, -INT_MAX+1)): \t%i\t", xpow(-111, -INT_MAX+1) == pow(-111, -INT_MAX+1));
xpow(-111, -INT_MAX+1);
printf("errno: %i\n", errno);
printf("=================================================\n");
printf("errno (EDOM: %i / ERANGE: %i)\n\n", EDOM, ERANGE);
}
int main(void)
{
xpow_tests();
return 0;
}
为了正确性,您应该添加特殊情况 if (y == 0 && x == 0) { errno = EDOM;
你有很多重复的代码。您应该添加 if(y<0) return 1.0/xpow(x, -y);
以摆脱这些。
代码的效率也很低,循环比必要的要长。您应该利用 x^y=(x^(y/2))^2。递归地这样做会节省很多迭代。当然,奇数处理要加点插件。
警告:这不是针对您失败的测试用例本身的解决方案。
但是,我稍微简化了您的函数 -- 这可能会有所帮助。
而且我非常 simplified/enhanced 你的测试用例函数 [使用一些预处理器技巧]。
然而,除了你失败的测试-111,INT_MAX
,当我运行它在这里时,0,-3
测试报告失败。
这是修改后的来源:
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <math.h>
#include <float.h>
#include <limits.h>
//#include "xmath.h"
double
xfabs(double x)
{
if (x < 0)
x = -x;
return x;
}
double
xpow(double x, int y)
{
errno = 0;
int i;
double lim;
double prod = 1;
double abs_x;
if (y == 0)
return 1.0;
if ((x == 0) && (y < 0)) {
errno = EDOM;
return 0.0;
}
abs_x = xfabs(x);
if (abs_x >= 1) {
if (y > 0) {
lim = DBL_MAX / abs_x;
for (i = 0; i < y; i++) {
if (xfabs(prod) > lim) {
errno = ERANGE;
return HUGE_VAL;
}
prod *= x;
}
}
if (y < 0) {
lim = DBL_MIN / abs_x;
for (i = 0; i > y; i--) {
if (xfabs(prod) < lim) {
errno = ERANGE;
return -0.0;
}
prod *= 1 / x;
}
}
return prod;
}
if (y > 0) {
lim = DBL_MIN / abs_x;
for (i = 0; i < y; i++) {
if (xfabs(prod) < lim) {
errno = ERANGE;
return +0.0;
}
prod *= x;
}
}
if (y < 0) {
lim = DBL_MAX / abs_x;
for (i = 0; i > y; i--) {
if (xfabs(prod) < lim) {
errno = ERANGE;
return -HUGE_VAL;
}
prod *= 1 / x;
}
}
return prod;
}
int passcnt = 0;
int failcnt = 0;
void
dotest(const char *name,double x,int y)
{
int sverr_pow;
int sverr_xpow;
double expected;
double actual;
int passflg;
printf("\n");
printf("%s\n",name);
errno = 0;
expected = pow(x,y);
sverr_pow = errno;
printf("%.16g (pow)\n",expected);
if (sverr_pow)
printf("errno: %d (%s)\n",sverr_pow,strerror(sverr_pow));
errno = 0;
actual = xpow(x,y);
sverr_xpow = errno;
printf("%.16g (xpow)\n",actual);
if (sverr_xpow)
printf("errno: %d (%s)\n",sverr_xpow,strerror(sverr_xpow));
if ((actual == expected) && (sverr_xpow == sverr_pow))
passcnt += 1;
else
failcnt += 1;
printf("Result: Value:%s errno:%s\n",
(actual == expected) ? "PASS" : "FAIL",
(sverr_xpow == sverr_pow) ? "PASS" : "FAIL");
}
#define TEST(x,y) \
dotest(#x "," #y,x,y)
void xpow_tests(void)
{
printf("XPOW TESTS (SUCCESS == 1 / FAILURE == 0)\n");
printf("=================================================\n");
TEST(2, 5);
TEST(-2, -4);
TEST(1, 15);
TEST(-4, 414);
TEST(-5, 303);
TEST(0, -3);
TEST(1.0e-10, 100);
TEST(-1.0e-10, 101);
TEST(-111, INT_MAX);
TEST(-111, INT_MAX-1);
TEST(-111, -INT_MAX);
TEST(-111, -INT_MAX+1);
printf("=================================================\n");
printf("errno (EDOM: %i / ERANGE: %i)\n\n", EDOM, ERANGE);
printf("PASSED: %d, FAILED: %d\n",passcnt,failcnt);
}
int
main(void)
{
xpow_tests();
return 0;
}
这是测试的输出:
XPOW TESTS (SUCCESS == 1 / FAILURE == 0)
=================================================
2,5
32 (pow)
32 (xpow)
Result: Value:PASS errno:PASS
-2,-4
0.0625 (pow)
0.0625 (xpow)
Result: Value:PASS errno:PASS
1,15
1 (pow)
1 (xpow)
Result: Value:PASS errno:PASS
-4,414
1.789931494904685e+249 (pow)
1.789931494904685e+249 (xpow)
Result: Value:PASS errno:PASS
-5,303
-6.136366831622158e+211 (pow)
-6.136366831622158e+211 (xpow)
Result: Value:PASS errno:PASS
0,-3
inf (pow)
errno: 34 (Numerical result out of range)
0 (xpow)
errno: 33 (Numerical argument out of domain)
Result: Value:FAIL errno:FAIL
1.0e-10,100
0 (pow)
errno: 34 (Numerical result out of range)
0 (xpow)
errno: 34 (Numerical result out of range)
Result: Value:PASS errno:PASS
-1.0e-10,101
-0 (pow)
errno: 34 (Numerical result out of range)
0 (xpow)
errno: 34 (Numerical result out of range)
Result: Value:PASS errno:PASS
-111,INT_MAX
-inf (pow)
errno: 34 (Numerical result out of range)
inf (xpow)
errno: 34 (Numerical result out of range)
Result: Value:FAIL errno:PASS
-111,INT_MAX-1
inf (pow)
errno: 34 (Numerical result out of range)
inf (xpow)
errno: 34 (Numerical result out of range)
Result: Value:PASS errno:PASS
-111,-INT_MAX
-0 (pow)
errno: 34 (Numerical result out of range)
-0 (xpow)
errno: 34 (Numerical result out of range)
Result: Value:PASS errno:PASS
-111,-INT_MAX+1
0 (pow)
errno: 34 (Numerical result out of range)
-0 (xpow)
errno: 34 (Numerical result out of range)
Result: Value:PASS errno:PASS
=================================================
errno (EDOM: 33 / ERANGE: 34)
PASSED: 10, FAILED: 2