c++ - scanf 没有接受 long double

标签 c++ mingw

我遇到问题,scanf 无法读取下面代码中的 long double:

(请原谅我糟糕的英语)

#include <iostream>
#include <cstdlib>
#include <math.h>
using namespace std;
int main()
    {
    int n;
    scanf("%d",&n);
    long double a,b,c,ha,hb,hc,ma,cosa,r,l,res,area;
    for (int i=0;i<n;i++)
    {
        scanf("%Lf %Lf %Lf %Lf",&a,&ha,&hb,&hc);//this is where the problem lies,
  //i need to read 4 long double a,ha,hb,hc
        printf("%Lf %Lf %Lf %Lf\n",a,ha,hb,hc);//but it returned wrong answer so
  //i used printf to check, ps: the code works with float but not with double or
  //long double
        ha*=3;hb*=3;hc*=3;
        c=(a*ha)/hc; b=(a*ha)/hb;
        ma=sqrt(0.5*b*b+0.5*c*c-0.25*a*a);
        cosa=ha/ma;
        r=(2*ma)/3;
        l=b*(b-sqrt(a*a-hb*hb))/ha;
        res=sqrt(l*l+r*r-2*cosa*l*r);
        area=a*ha/2;
        printf("%.3Lf %.3Lf\n",area,res);
    }
    system("PAUSE");
    return 0;}
}

这是输入:

2
3.0 0.8660254038 0.8660254038 0.8660254038
657.8256599140 151.6154399062 213.5392629932 139.4878846649

这里是命令行中显示的内容:

2
3.0 0.8660254038 0.8660254038 0.8660254038
3.000000 -4824911833695204400000000000000000000000000000000000000000000000000000
00000000000000000000000000000000000000000000000000000000000000000000000000000000
000000000000000000000000000000000000000000000.000000 284622047019579100000000000
00000000000000000000000000000000000000000000000000000000000000000000000000000000
00000000000000000000000000000000000000000000000000000000000000000000000000000000
000000000000000000000000000000000000000000000000000000000000000000000000000000.0
00000 0.866025
-2.000 0.000
657.8256599140 151.6154399062 213.5392629932 139.4878846649
657.825660 -0.000000 28969688850499604000000000000000000000000000000000000000000
00000000000000000000000000000000000000000000000000000000000000000000000000000000
00000000000000000000000000000000000000000000000000000000000000000000000000000000
0000000000000000000000000000000000000000000000.000000 213.539263
-2.000 0.000
Press any key to continue . . .

我想知道为什么 scanf 不会在代码中接受 long double 以及如何修复它。

提前致谢!

最佳答案

Dev-c++ 使用 MinGW,它使用 gcc 编译器和 Microsoft 运行时库。不幸的是,这些组件不同意用于 long double 的基础类型。 (我认为是 64 位与 80 位或 96 位)。 Windows 假定 long doubledouble 大小相同; gcc 使 long double更大。

任一选择均有效,但组合会导致 C 和 C++ 实现中断。

如果您不需要额外的范围和精度,您可以读入 double并存储到 long double .

否则,您可以编写或借用自定义字符串到long double转换器,或者只是使用不同的实现。

编辑

更多详情:

微软自己的编译器和运行时库在处理long double上是一致的为 64 位,大小与 double 相同.语言标准允许这样做(它要求 long double 至少double 一样宽,但对两者提出相同的要求),但它没有利用它似乎很奇怪x86 的 80 位浮点硬件。

x86 上的 gcc 对待 long double作为 96 位 ( sizeof (long double) == 12 )。我认为其中只有 80 位是重要的;额外的 16 位用于对齐目的。

MinGW 使用 gcc 作为其编译器,但使用 Microsoft 的运行时库。对于大多数语言功能,这工作正常,但不匹配 long double意味着您可以使用 long double 进行计算,但不能将 long double 值(或指向它们的指针)传递给运行时库。这是 MinGW 中的错误。

MinGW 中有变通办法。您可以定义宏 __USE_MINGW_ANSI_STDIO , 通过传递 -D__USE_MINGW_ANSI_STDIO在 gcc 命令行上或通过添加一行

#define __USE_MINGW_ANSI_STDIO

到你的源文件。 (必须在 #include <stdio.h> 之前定义它。)评论者 paulsm4 说 -ansi-posix选项导致 MinGW 使用它自己的符合标准的库(我没有理由怀疑这一点,但我目前无法确认)。或者您可以调用__mingw_printf()直接。

假设您使用的是 Windows,Cygwin 可能是一个不错的选择(它使用 gcc,但不使用 Microsoft 的运行时库)。或者你可以使用 long double在内部,但是 double用于 I/O。

关于c++ - scanf 没有接受 long double,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7822928/

相关文章:

c++ - gcc 找不到-lgcc, g++.exe : error: CreateProcess: No such file or directory

c++ - 没有主体但参数名称不同的类函数宏

c++ - Qt5 无法在 Windows 8 上编译

c - 如果源代码在 win32 中是 UTF-8 编码的,如何显示 UTF-16 const 字符串?

c++ - UDP 客户端未正确形成数据包

C++ 编译错误 : "cast from ' WCHAR *' to ' WORD' loses precision"

c++ - MinGW 不将从模板类继承为模板

c++ - 为 QTableWidget 中的单元格设置默认对齐方式

c++ - 简化一段代码—— vector 旋转一圈

c++ - 安卓 NDK + Eclipse : No such file or directory