mirror of
				https://github.com/saitohirga/WSJT-X.git
				synced 2025-11-04 05:50:31 -05:00 
			
		
		
		
	
		
			
	
	
		
			292 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			292 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| 
								 | 
							
								/** nonfinite_num_facet.cpp
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Copyright (c) 2011 Francois Mauger
							 | 
						||
| 
								 | 
							
								 * Copyright (c) 2011 Paul A. Bristow
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Distributed under the Boost Software License, Version 1.0.
							 | 
						||
| 
								 | 
							
								 * (See accompanying file LICENSE_1_0.txt
							 | 
						||
| 
								 | 
							
								 * or copy at http://www.boost.org/LICENSE_1_0.txt)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * This simple program illustrates how to use the
							 | 
						||
| 
								 | 
							
								 * `boost/math/nonfinite_num_facets.hpp' material from the original
							 | 
						||
| 
								 | 
							
								 * Floating Point  Utilities contribution by Johan Rade.
							 | 
						||
| 
								 | 
							
								 * Floating Point Utility library has been accepted into Boost,
							 | 
						||
| 
								 | 
							
								 * but the utilities have been/will be incorporated into Boost.Math library.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								\file
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								\brief A fairly simple example of using non_finite_num facet for
							 | 
						||
| 
								 | 
							
								C99 standard output of infinity and NaN.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								\detail  This program illustrates how to use the
							 | 
						||
| 
								 | 
							
								 `boost/math/nonfinite_num_facets.hpp' material from the original
							 | 
						||
| 
								 | 
							
								  Floating Point  Utilities contribution by Johan Rade.
							 | 
						||
| 
								 | 
							
								  Floating Point Utility library has been accepted into Boost,
							 | 
						||
| 
								 | 
							
								  but the utilities have been/will be incorporated into Boost.Math library.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  Based on an example from Francois Mauger.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  Double and float variables are assigned ordinary finite values (pi),
							 | 
						||
| 
								 | 
							
								  and nonfinite like infinity and NaN.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  These values are then output and read back in, and then redisplayed.
							 | 
						||
| 
								 | 
							
								  
							 | 
						||
| 
								 | 
							
								*/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#ifdef _MSC_VER
							 | 
						||
| 
								 | 
							
								#   pragma warning(disable : 4127) // conditional expression is constant.
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <iostream>
							 | 
						||
| 
								 | 
							
								#include <iomanip>
							 | 
						||
| 
								 | 
							
								using std::cout;
							 | 
						||
| 
								 | 
							
								using std::endl;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <limits> // numeric_limits
							 | 
						||
| 
								 | 
							
								using std::numeric_limits;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <boost/cstdint.hpp>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <boost/math/special_functions/nonfinite_num_facets.hpp>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static const char sep = ','; // Separator of bracketed float and double values.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// Use max_digits10 (or equivalent) to obtain 
							 | 
						||
| 
								 | 
							
								// all potentially significant decimal digits for the floating-point types.
							 | 
						||
| 
								 | 
							
								    
							 | 
						||
| 
								 | 
							
								#ifdef BOOST_NO_CXX11_NUMERIC_LIMITS
							 | 
						||
| 
								 | 
							
								  std::streamsize  max_digits10_float = 2 + std::numeric_limits<float>::digits * 30103UL / 100000UL;
							 | 
						||
| 
								 | 
							
								  std::streamsize  max_digits10_double = 2 + std::numeric_limits<double>::digits * 30103UL / 100000UL;
							 | 
						||
| 
								 | 
							
								#else
							 | 
						||
| 
								 | 
							
								  // Can use new C++0X max_digits10 (the maximum potentially significant digits).
							 | 
						||
| 
								 | 
							
								  std::streamsize  max_digits10_float = std::numeric_limits<float>::max_digits10;
							 | 
						||
| 
								 | 
							
								  std::streamsize  max_digits10_double = std::numeric_limits<double>::max_digits10;
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/* A class with a float and a double */
							 | 
						||
| 
								 | 
							
								struct foo
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  foo () : fvalue (3.1415927F), dvalue (3.1415926535897931)
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  // Set both the values to -infinity :
							 | 
						||
| 
								 | 
							
								  void minus_infinity ()
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    fvalue = -std::numeric_limits<float>::infinity ();
							 | 
						||
| 
								 | 
							
								    dvalue = -std::numeric_limits<double>::infinity ();
							 | 
						||
| 
								 | 
							
								    return;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  // Set the values to +infinity :
							 | 
						||
| 
								 | 
							
								  void plus_infinity ()
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    fvalue = +std::numeric_limits<float>::infinity ();
							 | 
						||
| 
								 | 
							
								    dvalue = +std::numeric_limits<double>::infinity ();
							 | 
						||
| 
								 | 
							
								    return;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  // Set the values to NaN :
							 | 
						||
| 
								 | 
							
								  void nan ()
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    fvalue = +std::numeric_limits<float>::quiet_NaN ();
							 | 
						||
| 
								 | 
							
								    dvalue = +std::numeric_limits<double>::quiet_NaN ();
							 | 
						||
| 
								 | 
							
								    return;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  // Print a foo:
							 | 
						||
| 
								 | 
							
								  void print (std::ostream & a_out, const std::string & a_title)
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    if (a_title.empty ()) a_out << "foo";
							 | 
						||
| 
								 | 
							
								    else a_out << a_title;
							 | 
						||
| 
								 | 
							
								    a_out << " : " << std::endl;
							 | 
						||
| 
								 | 
							
								    a_out << "|-- " << "fvalue = ";
							 | 
						||
| 
								 | 
							
								    
							 | 
						||
| 
								 | 
							
								    a_out.precision (max_digits10_float);
							 | 
						||
| 
								 | 
							
								    a_out << fvalue << std::endl;
							 | 
						||
| 
								 | 
							
								    a_out << "`-- " << "dvalue = ";
							 | 
						||
| 
								 | 
							
								    a_out.precision (max_digits10_double);
							 | 
						||
| 
								 | 
							
								    a_out << dvalue << std::endl;
							 | 
						||
| 
								 | 
							
								    return;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // I/O operators for a foo structure of a float and a double :
							 | 
						||
| 
								 | 
							
								  friend std::ostream & operator<< (std::ostream & a_out, const foo & a_foo);
							 | 
						||
| 
								 | 
							
								  friend std::istream & operator>> (std::istream & a_in, foo & a_foo);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // Attributes :
							 | 
						||
| 
								 | 
							
								  float  fvalue; // Single precision floating number.
							 | 
						||
| 
								 | 
							
								  double dvalue; // Double precision floating number.
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								std::ostream & operator<< (std::ostream & a_out, const foo & a_foo)
							 | 
						||
| 
								 | 
							
								{ // Output bracketed FPs, for example "(3.1415927,3.1415926535897931)"
							 | 
						||
| 
								 | 
							
								  a_out.precision (max_digits10_float);
							 | 
						||
| 
								 | 
							
								  a_out << "(" << a_foo.fvalue << sep ;
							 | 
						||
| 
								 | 
							
								  a_out.precision (max_digits10_double);
							 | 
						||
| 
								 | 
							
								  a_out << a_foo.dvalue << ")";
							 | 
						||
| 
								 | 
							
								  return a_out;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								std::istream & operator>> (std::istream & a_in, foo & a_foo)
							 | 
						||
| 
								 | 
							
								{ // Input bracketed floating-point values into a foo structure,
							 | 
						||
| 
								 | 
							
								  // for example from "(3.1415927,3.1415926535897931)"
							 | 
						||
| 
								 | 
							
								  char c = 0;
							 | 
						||
| 
								 | 
							
								  a_in.get (c);
							 | 
						||
| 
								 | 
							
								  if (c != '(')
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    std::cerr << "ERROR: operator>> No ( " << std::endl;
							 | 
						||
| 
								 | 
							
								    a_in.setstate(std::ios::failbit);
							 | 
						||
| 
								 | 
							
								    return a_in;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  float f;
							 | 
						||
| 
								 | 
							
								  a_in >> std::ws >> f;
							 | 
						||
| 
								 | 
							
								  if (! a_in)
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    return a_in;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  a_in >> std::ws;
							 | 
						||
| 
								 | 
							
								  a_in.get (c);
							 | 
						||
| 
								 | 
							
								  if (c != sep)
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    std::cerr << "ERROR: operator>> c='" << c << "'" << std::endl;
							 | 
						||
| 
								 | 
							
								    std::cerr << "ERROR: operator>> No '" << sep << "'" << std::endl;
							 | 
						||
| 
								 | 
							
								    a_in.setstate(std::ios::failbit);
							 | 
						||
| 
								 | 
							
								    return a_in;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  double d;
							 | 
						||
| 
								 | 
							
								  a_in >> std::ws >> d;
							 | 
						||
| 
								 | 
							
								  if (! a_in)
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    return a_in;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  a_in >> std::ws;
							 | 
						||
| 
								 | 
							
								  a_in.get (c);
							 | 
						||
| 
								 | 
							
								  if (c != ')')
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    std::cerr << "ERROR: operator>> No ) " << std::endl;
							 | 
						||
| 
								 | 
							
								    a_in.setstate(std::ios::failbit);
							 | 
						||
| 
								 | 
							
								    return a_in;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  a_foo.fvalue = f;
							 | 
						||
| 
								 | 
							
								  a_foo.dvalue = d;
							 | 
						||
| 
								 | 
							
								  return a_in;
							 | 
						||
| 
								 | 
							
								} // std::istream & operator>> (std::istream & a_in, foo & a_foo)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								int main ()
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								  std::cout << "nonfinite_num_facet simple example." << std::endl;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								   if((std::numeric_limits<double>::has_infinity == false) || (std::numeric_limits<double>::infinity() == 0))
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    std::cout << "Infinity not supported on this platform." << std::endl;
							 | 
						||
| 
								 | 
							
								    return 0;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if((std::numeric_limits<double>::has_quiet_NaN == false) || (std::numeric_limits<double>::quiet_NaN() == 0))
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    std::cout << "NaN not supported on this platform." << std::endl;
							 | 
						||
| 
								 | 
							
								    return 0;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#ifdef BOOST_NO_CXX11_NUMERIC_LIMITS
							 | 
						||
| 
								 | 
							
								  cout << "BOOST_NO_CXX11_NUMERIC_LIMITS is defined, so no max_digits10 available either:"
							 | 
						||
| 
								 | 
							
								     "\n we'll have to calculate our own version." << endl;
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								  std::cout << "std::numeric_limits<float>::max_digits10 is " << max_digits10_float << endl;
							 | 
						||
| 
								 | 
							
								  std::cout << "std::numeric_limits<double>::max_digits10 is " << max_digits10_double << endl;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								   std::locale the_default_locale (std::locale::classic ());
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    std::cout << "Write to a string buffer (using default locale) :" << std::endl;
							 | 
						||
| 
								 | 
							
								    foo f0; // pi
							 | 
						||
| 
								 | 
							
								    foo f1; f1.minus_infinity ();
							 | 
						||
| 
								 | 
							
								    foo f2; f2.plus_infinity ();
							 | 
						||
| 
								 | 
							
								    foo f3; f3.nan ();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    f0.print (std::cout, "f0"); // pi
							 | 
						||
| 
								 | 
							
								    f1.print (std::cout, "f1"); // +inf
							 | 
						||
| 
								 | 
							
								    f2.print (std::cout, "f2"); // -inf
							 | 
						||
| 
								 | 
							
								    f3.print (std::cout, "f3"); // NaN
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    std::ostringstream oss;
							 | 
						||
| 
								 | 
							
								    std::locale C99_out_locale (the_default_locale, new boost::math::nonfinite_num_put<char>);
							 | 
						||
| 
								 | 
							
								    oss.imbue (C99_out_locale);
							 | 
						||
| 
								 | 
							
								    oss.precision (15);
							 | 
						||
| 
								 | 
							
								    oss << f0 << f1 << f2 << f3;
							 | 
						||
| 
								 | 
							
								    std::cout << "Output in C99 format is: \"" << oss.str () << "\"" << std::endl;
							 | 
						||
| 
								 | 
							
								    std::cout << "Output done." << std::endl;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  {
							 | 
						||
| 
								 | 
							
								    std::string the_string = "(3.1415927,3.1415926535897931)(-inf,-inf)(inf,inf)(nan,nan)"; // C99 format
							 | 
						||
| 
								 | 
							
								    // Must have correct separator!
							 | 
						||
| 
								 | 
							
								    std::cout << "Read C99 format from a string buffer containing \"" << the_string << "\""<< std::endl;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    std::locale C99_in_locale (the_default_locale, new boost::math::nonfinite_num_get<char>);
							 | 
						||
| 
								 | 
							
								    std::istringstream iss (the_string);
							 | 
						||
| 
								 | 
							
								    iss.imbue (C99_in_locale);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    foo f0, f1, f2, f3;
							 | 
						||
| 
								 | 
							
								    iss >> f0 >> f1 >> f2 >> f3;
							 | 
						||
| 
								 | 
							
								    if (! iss)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								       std::cerr << "Input Format error !" << std::endl;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    else
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								      std::cerr << "Input OK." << std::endl;
							 | 
						||
| 
								 | 
							
								      cout << "Display in default locale format " << endl;
							 | 
						||
| 
								 | 
							
								      f0.print (std::cout, "f0");
							 | 
						||
| 
								 | 
							
								      f1.print (std::cout, "f1");
							 | 
						||
| 
								 | 
							
								      f2.print (std::cout, "f2");
							 | 
						||
| 
								 | 
							
								      f3.print (std::cout, "f3");
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    std::cout << "Input done." << std::endl;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  
							 | 
						||
| 
								 | 
							
								  std::cout << "End nonfinite_num_facet.cpp" << std::endl;
							 | 
						||
| 
								 | 
							
								  return 0;
							 | 
						||
| 
								 | 
							
								} // int main()
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								 // end of test_nonfinite_num_facets.cpp
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								Output:
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								nonfinite_num_facet simple example.
							 | 
						||
| 
								 | 
							
								  std::numeric_limits<float>::max_digits10 is 8
							 | 
						||
| 
								 | 
							
								  std::numeric_limits<double>::max_digits10 is 17
							 | 
						||
| 
								 | 
							
								  Write to a string buffer (using default locale) :
							 | 
						||
| 
								 | 
							
								  f0 : 
							 | 
						||
| 
								 | 
							
								  |-- fvalue = 3.1415927
							 | 
						||
| 
								 | 
							
								  `-- dvalue = 3.1415926535897931
							 | 
						||
| 
								 | 
							
								  f1 : 
							 | 
						||
| 
								 | 
							
								  |-- fvalue = -1.#INF
							 | 
						||
| 
								 | 
							
								  `-- dvalue = -1.#INF
							 | 
						||
| 
								 | 
							
								  f2 : 
							 | 
						||
| 
								 | 
							
								  |-- fvalue = 1.#INF
							 | 
						||
| 
								 | 
							
								  `-- dvalue = 1.#INF
							 | 
						||
| 
								 | 
							
								  f3 : 
							 | 
						||
| 
								 | 
							
								  |-- fvalue = 1.#QNAN
							 | 
						||
| 
								 | 
							
								  `-- dvalue = 1.#QNAN
							 | 
						||
| 
								 | 
							
								  Output in C99 format is: "(3.1415927,3.1415926535897931)(-inf,-inf)(inf,inf)(nan,nan)"
							 | 
						||
| 
								 | 
							
								  Output done.
							 | 
						||
| 
								 | 
							
								  Read C99 format from a string buffer containing "(3.1415927,3.1415926535897931)(-inf,-inf)(inf,inf)(nan,nan)"
							 | 
						||
| 
								 | 
							
								  Display in default locale format 
							 | 
						||
| 
								 | 
							
								  f0 : 
							 | 
						||
| 
								 | 
							
								  |-- fvalue = 3.1415927
							 | 
						||
| 
								 | 
							
								  `-- dvalue = 3.1415926535897931
							 | 
						||
| 
								 | 
							
								  f1 : 
							 | 
						||
| 
								 | 
							
								  |-- fvalue = -1.#INF
							 | 
						||
| 
								 | 
							
								  `-- dvalue = -1.#INF
							 | 
						||
| 
								 | 
							
								  f2 : 
							 | 
						||
| 
								 | 
							
								  |-- fvalue = 1.#INF
							 | 
						||
| 
								 | 
							
								  `-- dvalue = 1.#INF
							 | 
						||
| 
								 | 
							
								  f3 : 
							 | 
						||
| 
								 | 
							
								  |-- fvalue = 1.#QNAN
							 | 
						||
| 
								 | 
							
								  `-- dvalue = 1.#QNAN
							 | 
						||
| 
								 | 
							
								  Input done.
							 | 
						||
| 
								 | 
							
								  End nonfinite_num_facet.cpp
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								*/
							 |