最終更新日時:
が更新

履歴 編集

function
<cmath>

std::rint(C++11)

namespace std {
  float rint(float x);
  double rint(double x);
  long double rint(long double x);

  double rint(Integral x);

  float rintf(float x);             // C++17 から
  long double rintl(long double x); // C++17 から
}

概要

引数 x を現在の丸めモードで整数値に丸めた値を得る。

戻り値

引数 x を現在の丸めモードで整数値に丸めた値

備考

  • 本関数と nearbyint は戻り値は同一であるが、本関数は引数 x が戻り値と異なってる場合 FE_INEXACT が発生する可能性があるが、nearbyint は発生しない点のみ動作が異なる。
  • 本関数は、C99 の規格にある rint(より正確には math.h ヘッダの rintrintfrintl の 3 つ。それぞれ C++ の doublefloatlong double バージョンに相当)と同等である。
  • C++11 以降では、処理系が IEC 60559 に準拠している場合(std::numeric_limits<T>::is_iec559() != false)、以下の規定が追加される。
    • x = ±0 の場合、±0 を返す。
    • x = ±∞ の場合、±∞ を返す。
    • 引数 x が戻り値と異なってる場合、FE_INEXACT が発生する。
  • C99 では、丸めモードの設定時には #pragma STDC FENV_ACCESS ON でなければなければならないと記載されているが、C++ には該当する記載を見つけることができなかった。
    なお、C99 でも FENV_ACCESS のデフォルトは処理系定義である。
  • 丸めモード FE_TONEAREST は単なる四捨五入ではないことに注意。

#include <cfenv>
#include <cmath>
#include <iostream>

void test(const char* title, int round_mode)
{
  std::feclearexcept(FE_ALL_EXCEPT);
  std::fesetround(round_mode);
  std::cout << title << std::endl;
  std::cout << "rint(2.5) = " << std::rint(2.5) << std::endl;
  std::cout << "rint(-2.5) = " << std::rint(-2.5) << std::endl;
  std::cout << "FE_INEXACT = " << std::boolalpha << (std::fetestexcept(FE_INEXACT) != 0) << std::endl << std::endl;
}

# define test(mode) test(#mode, mode)

int main()
{
# ifdef FE_DOWNWARD
  test(FE_DOWNWARD);
# endif
# ifdef FE_TONEAREST
  test(FE_TONEAREST);
# endif
# ifdef FE_TOWARDZERO
  test(FE_TOWARDZERO);
# endif
# ifdef FE_UPWARD
  test(FE_UPWARD);
# endif
}

出力例

FE_DOWNWARD
rint(2.5) = 2
rint(-2.5) = -3
FE_INEXACT = true

FE_TONEAREST
rint(2.5) = 2
rint(-2.5) = -2
FE_INEXACT = true

FE_TOWARDZERO
rint(2.5) = 2
rint(-2.5) = -2
FE_INEXACT = true

FE_UPWARD
rint(2.5) = 3
rint(-2.5) = -2
FE_INEXACT = true

処理系が ISO IEC 60559 に準拠していない場合、全ての丸めモードが利用可能とは限らない。
また、処理系が IEC60559 に準拠していない場合、FE_INEXACTfalse の可能性がある。

バージョン

言語

  • C++11

処理系

備考

本関数は C++11 で追加されたが、Clang(libc++) では C++11 モードでなくても使用可能である。