29

根据Matlab 文档,当 Java 方法返回 a 时long,它会double在 Matlab 中分配之前转换为 a。

精度已经丢失。我对longJava 方法返回的低位数字感兴趣。Adouble不能代表它们,但 Matlabint64可以。(考虑到这两种类型都有 64 位,并且 adouble使用其中的一些来表示指数,这一点就很清楚了。)

如果我可以控制 Java 代码,我可以返回一个包含一个元素的数组,其中包含long- 在这种情况下,Matlab 将它们保留为int64s,但在我的情况下,我正在调用一个库函数。

目前,我能看到的最好方法是用 Java 编写一个包装器,它将调用该方法并在数组中返回答案。但是这种方法存在可移植性问题。有没有更好的办法?

4

2 回答 2

3

如果long您的 Java 方法返回的结果在 [-9,007,199,254,740,992 到 9,007,199,254,740,992] 范围内,那么您根本不需要在代码中执行任何操作。您可以将结果转换回 Matlab 的int64,而不会损失任何精度。该范围内的所有整数都可以表示为double数字而不会损失精度,因为该值可以存储在 64 位值表示提供的 53 位有效二进制数字中double

您可以在我的书Code Quality: The Open Source Perspective的第 8 章(浮点运算)或任何其他涵盖浮点运算的教科书中找到更多详细信息。

下面的程序演示了double精确表示范围末尾的 1 亿个整数表示的准确性。

#include <stdio.h>

int
main(int argc, char *argv[])
{
    int i;
    volatile long long n1, n2;
    volatile long long p1, p2;
    volatile double d;

    /* Include one number outside the range, to show that the test works. */
    n1 = -9007199254740993ll;
    p1 =  9007199254740993ll;
    for (i = 0; i < 100000000; i++) {
        d = p1;
        p2 = d;
        if (p1 != p2)
            printf("%lld != %lld\n", p1, p2);

        d = n1;
        n2 = d;
        if (n1 != n2)
            printf("%lld != %lld\n", n1, n2);
        p1--;
        n1++;
    }
    return 0;
}
于 2013-05-26T06:41:37.840 回答
1

它基本上在评论中得到了回答,但为了完整起见,将 Java 接收long到 Matlab 中的最佳方法int64似乎是编写一个小的 Java 包装器,它调用您拥有的任何方法并以long[].

(如果要分发代码,请考虑使用比最新版本稍早的目标 JVM 版本编译该 Java 类 - 否则一些用户将被迫升级他们的 JVM 以运行您的软件,而其中一些则没有管理员权限执行此操作。)

于 2013-09-28T07:20:04.543 回答