c - 如何在溢出时停止双重自动转换?

标签 c linux double long-integer

我有一个代码,其中我使用 getenv 从环境中提取字符串,然后使用 strtod 将它们解析为数字。 如果用户输入,213.123。然后将 213 和 123 分别送入 long 类型。

long a1 = 213;长 a2 = 123

我面临的问题是,如果用户输入一个非常长的数字,例如:123456789123.45678,它会自动四舍五入,这是我不想要的,而是抛出一个错误,但是 ERANGE 不起作用。

9 static  volatile int flag;                   /* flag variable to indicate when the measurement should start */
10 static  time_t       ef_errtrack_start_sec;  /* error track start time in seconds */
11 static  long         ef_errtrack_start_nsec; /* error track start time in nanoseconds */
12 static  time_t       ef_errtrack_end_sec;    /* error track end time in seconds */
13 static  long         ef_errtrack_end_nsec;   /* error track end time in nanoseconds */

21 int main(int argc, char **argv)
22 {
23     extractTime(1); /* Extracting start time */
24     extractTime(0); /* Extracting end time   */
25 
26     printf("start: %12d, %12d\n", ef_errtrack_start_sec, ef_errtrack_start_nsec);
27     printf("end:   %12d, %12d\n", ef_errtrack_end_sec,   ef_errtrack_end_nsec);
28 
29     return 0;
30 }


35 void extractTime(int extractStartTime)
36 {
37         char * charPtr, * numberFormatErr;
38         regex_t re;
39 
40         ( extractStartTime == 1 ) ? ( charPtr = getenv("EF_ERRTRACK_START") ) :
41                 ( charPtr = getenv("EF_ERRTRACK_END") );
42 
43         if ( charPtr == NULL )
44                 return;
45 
46         double envVal = strtod(charPtr, &numberFormatErr);
47 
48         if ( (numberFormatErr == charPtr) || (*numberFormatErr != '\0') ) {
49                 ( extractStartTime == 1 ) ? printf("eFence exited: EF_ERRTRACK_START is not a number\n") :
50                         printf("eFence exited: EF_ERRTRACK_END is not a number\n");
51                 exit(1);
52         }
53         if ( errno == ERANGE )
54         {
55                 ( extractStartTime == 1 ) ? EF_Print("eFence exited: EF_ERRTRACK_START is out of range\n") :
56                         EF_Print("eFence exited: EF_ERRTRACK_END is out of range\n");
57                 exit(1);
58         }
59         else if ( envVal < 0 ) {
60                 ( extractStartTime == 1 ) ? printf("eFence exited: EF_ERRTRACK_START a negative number\n") :
61                         printf("eFence exited: EF_ERRTRACK_END is a negative number\n");
62                 exit(1);
63         }
64 
65         if ( extractStartTime ) {
66                 ef_errtrack_start_sec = envVal;
67                 double nsec = (envVal) - (double)(ef_errtrack_start_sec);
68                 ef_errtrack_start_nsec = (long)(nsec * 1000000000);
69         }
70         else {
71                 ef_errtrack_end_sec = envVal;
72                 double nsec = (envVal) - (double)(ef_errtrack_end_sec);
73                 ef_errtrack_end_nsec = (long) (nsec * 1000000000);
74         }
75 }

这是输出:

Output:
/tmp # export EF_ERRTRACK_START=1234567891234.123456789123
/tmp # export EF_ERRTRACK_END=10e2

/tmp/time_related # ./a.out 

start:   2147483647,   2147483647
end:           1000,            0

最佳答案

“超出可表示值的范围”意味着大于 DBL_MAX。您的输入在范围内,只是不能完全表示为 double。

就此而言,0.1 也在范围内,但也不能完全表示。这也应该是一个错误吗?如果不是,有什么区别?

我不确定该建议您做什么,因为我不确定您为什么认为您的情况是错误的。一种选择是,一旦你有了 double 值,使用 snprintf 将其转换回字符串并与原始输入进行比较,看看它们是否至少在小数点上相等。不过,这忽略了科学记数法,因此可能需要做更多的工作来识别您不喜欢的数字。

编辑:啊,最初我并没有真正理解这个:“如果用户输入 213.123。然后 213 和 123 将被单独馈送到一个 long 类型。”

听起来您正在阅读的不是 double 值,它是由句点字符分隔的两个整数值。所以不要使用 strtod,找到 . 然后在它的每一边调用 strtol

关于c - 如何在溢出时停止双重自动转换?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7554276/

相关文章:

c# - 为什么 5/10 = 0.0?

php - PHP 扩展中的 SEG 错误

c++ - 我应该使用 C 类型 (uint8_t/.../uint64_t) 还是 (u_int8_t/.../u_int64_t)?

C 用已知的最大长度逐个字符构建字符串

linux - 在 Linux 上构建服务示例 AllJoyn 瘦客户端

linux - 使用 AWK 组合(合并)多行

java - 如何让 JButton 在每次单击时添加 double

c - 为什么 gets() 比 scanf() 更危险?

linux - 如何在 Linux 中创建历史甘特图?

c++ - 简单的 C++ 求和程序