在转换为较小的数字类型之前执行范围检查的安全、跨平台方法是什么?
What is a safe, cross-platform way to perform range checks before casting to a smaller numeric type?
Here 是我能找到的最接近的副本。
尽管有标签,问题似乎是关于 C 的,可用的 answer 引用了 C99 规范。
在不使用 Boost 或其他库的情况下,在 C++98 中处理此检查的正确方法是什么?
您可以从 gsl::narrow()
复制代码并稍微调整一下,将其变成 can_narrow()
返回 bool
而不是 throw
ing:
// narrow_cast(): a searchable way to do narrowing casts of values
template<class T, class U>
inline constexpr T narrow_cast(U u) noexcept
{ return static_cast<T>(u); }
namespace details
{
template<class T, class U>
struct is_same_signedness : public std::integral_constant<bool, std::is_signed<T>::value == std::is_signed<U>::value>
{};
}
template<class T, class U>
inline bool can_narrow(U u)
{
T t = narrow_cast<T>(u);
if (static_cast<U>(t) != u)
return false;
if (!details::is_same_signedness<T, U>::value && ((t < T{}) != (u < U{})))
return false;
return true;
}
Here 是我能找到的最接近的副本。
尽管有标签,问题似乎是关于 C 的,可用的 answer 引用了 C99 规范。
在不使用 Boost 或其他库的情况下,在 C++98 中处理此检查的正确方法是什么?
您可以从 gsl::narrow()
复制代码并稍微调整一下,将其变成 can_narrow()
返回 bool
而不是 throw
ing:
// narrow_cast(): a searchable way to do narrowing casts of values
template<class T, class U>
inline constexpr T narrow_cast(U u) noexcept
{ return static_cast<T>(u); }
namespace details
{
template<class T, class U>
struct is_same_signedness : public std::integral_constant<bool, std::is_signed<T>::value == std::is_signed<U>::value>
{};
}
template<class T, class U>
inline bool can_narrow(U u)
{
T t = narrow_cast<T>(u);
if (static_cast<U>(t) != u)
return false;
if (!details::is_same_signedness<T, U>::value && ((t < T{}) != (u < U{})))
return false;
return true;
}