8

我试图想出一个技巧来测试是否std::isnan在预处理器中没有特殊的套管编译器定义,并想出了以下内容,我希望它可以正常工作。

#include <cmath>
#include <type_traits>

namespace detail {
    using namespace std;

    struct dummy {};
    void isnan(dummy);

    //bool isnan(float); // Just adding this declaration makes it work!

    template <typename T>
    struct is_isnan_available {
        template <typename T1>
        static decltype(isnan(T1())) test(int);
        template <typename>
        static void test(...);

        enum { value = !std::is_void<decltype(test<T>(0))>::value };
    };
}

int main() {
    return detail::is_isnan_available<float>::value;
}

原来它没有检测到它。我知道肯定std::isnan是在ideone上定义的,因为我手动测试过。

当我取消注释上面的标记行时,它就起作用了。

我在这里想念什么?什么解释了这种行为?

4

2 回答 2

7

问题是 using 指令不会将成员添加到当前命名空间,因此这些std::成员仍可能被此命名空间中的声明隐藏。

using std::isnan相反,它的行为就像将导入的命名空间的成员添加到包含use-location 和导入的命名空间的命名空间一样。using 声明是命名空间中的普通声明,因此可以与后面的声明一起参与重载决议。

但是,正如评论中指出的那样,如果该函数不存在,则会产生错误。要解决这个问题,您需要将其从detail::命名空间中删除。这应该可行,因为导入的定义将与dummy重载处于同一级别。您可以将重载带到全局命名空间,或者您可以创建一个辅助命名空间(在全局命名空间中)并同时导入.

于 2012-01-06T11:04:55.487 回答
1

我为取代非线程安全标准函数的 POSIX 线程安全 API 集解决了这个问题:C++11 替代 localtime_r。此代码检测是否在全局命名空间中定义了 API,如果不存在,则选择自定义解决方法。

于 2012-01-06T14:33:10.177 回答