⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 cpl_strtod.cpp

📁 用于读取TAB、MIF、SHP文件的类
💻 CPP
📖 第 1 页 / 共 2 页
字号:
/****************************************************************************** * $Id: cpl_strtod.cpp,v 1.8 2006/06/30 15:05:46 dron Exp $ * * Project:  CPL - Common Portability Library * Purpose:  Functions to convert ASCII string to floating point number. * Author:   Andrey Kiselev, dron@ak4719.spb.edu. This code is adopted version *           of implementation written by Gregory Pietsch. * ****************************************************************************** * Copyright (c) 2006, Andrey Kiselev * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included * in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. ****************************************************************************** * * $Log: cpl_strtod.cpp,v $ * Revision 1.8  2006/06/30 15:05:46  dron * Avoid macro redefinition. * * Revision 1.7  2006/04/21 14:43:52  dron * Added test for copysignf() presence. * * Revision 1.6  2006/04/21 12:50:09  dron * Eliminated all the code related to long double floats. * * Revision 1.5  2006/04/04 16:11:06  fwarmerdam * temporarily avoid use of long double math funcs not on macos 10.3 * * Revision 1.4  2006/04/02 14:45:38  fwarmerdam * Avoid VS8 problem with const char * / char *. * * Revision 1.3  2006/03/29 14:30:13  dron * Added forgotten extern "C" statements. * * Revision 1.2  2006/03/29 10:58:30  dron * Declare nan/nanf/nanl functions if not declared in system headers. * * Revision 1.1  2006/03/18 16:35:35  dron * New. * *//* strtold.c - code for the atof, strtod, strtof, and strtold functions   AUTHOR: Gregory Pietsch      DESCRIPTION:      The strtod and strtof functions convert the initial portion of the   string pointed to by nptr to double and float representations,   respectively. First, they decompose the input string into three parts: an   initial, possibly empty, sequence of white-space characters (as specified by   the isspace function), a subject sequence resembling a floating-point   constant or representing infinity or NaN, and a final sequence of one or   more unrecognized characters, including the terminating null character of   the input string. Then, they attempt to convert the subject string to a   floating-point number, and return the result.      The expected form of the subject sequence is an optional plus or minus sign,   then one of the following:      - a nonempty sequence of decimal digits optionally containing a decimal-     point character, then an optional exponent part as defined in 6.4.4.2;        - a 0x or 0X, then a nonempty sequence of hexadecimal digits optionally     containing a decimal-point character, then an optional binary exponent     part as defined in 6.4.4.2;        - one of INF or INFINITY, ignoring case      - one of NAN or NAN(n-char-sequence_opt), ignoring case in the NAN part,     where:            n-char-sequence:         digit         nondigit         n-char-sequence digit         n-char-sequence non-digit        The subject sequence is defined as the longest initial subsequence of the   input string, starting with the first non-white-space character, that is of   the expected form. The subject sequence contains no characters if the input   string is not of the expected form.      If the subject sequence has the expected form for a floating-point number,   the sequence of characters starting with the first digit or the decimal-   point character (whichever comes first) is interpreted as a floating    constant according to the rules of 6.4.4.2, except that the decimal-point   character is used in place of a period, and that if neither an exponent part   nor a decimal-point character appears in a decimal floating-point number,   or if a binary exponent part does not appear in a hexadecimal floating-point   number, an exponent part of the appropriate type with value zero is assumed   to follow the last digit in the string. If the subject sequence begins with   a minus sign, the sequence is interpreted as negated. (It is unspecified   whether a minus-signed sequence is converted to a negative number directly   or by negating the value resulting from converting the corresponding   unsigned sequence [see F.5]; the two methods may yield different results if   rounding is toward positive or negative infinity. In either case, the   functions honor the sign of zero if floating-point arithmetic supports    signed zeros.) A character sequence INF or INFINITY is interpreted as an   infinity, if representable in the return type, else like a floating constant   that is too large for the range of the return type. A character sequence   NAN or NAN(n-char-sequence_opt), is interpreted as a quiet NaN, if supported   in the return type, else like a subject sequence part that does not have the   expected form; the meaning of the n-char-sequences is implementation-defined   (an implementation may use the n-char-sequence to determine extra   information to be represented in the NaN's significand). A pointer to the   final string is stored in the object pointed to by endptr, provided that   endptr is not a null pointer.      If the subject string has the hexadecimal form and FLT_RADIX is a power of   2, the value resulting from the conversion is correctly rounded.      In other than the "C" locale, additional locale-specific subject sequences   may be accepted.   If the subject sequence is empty or does not have the expected form, no   conversion is performed; the value of nptr is stored in the object pointed   to by endptr, provided that endptr is not a null pointer.      RECOMMENDED PRACTICE:      If the subject sequence has the hexadecimal form, FLT_RADIX is not a power   of 2, and the result is not exactly representable, the result should be one   of the two numbers in the appropriate internal format that are adjacent to   the hexadecimal floating source value, with the stipulation that the error   should have the correct sign for the current rounding direction.      If the subject sequence has the decimal form and at most DECIMAL_DIG   (defined in <float.h>) significant digits, the result should be correctly   rounded. If the subject sequence D has the decimal form and more than    DECIMAL_DIG significant digits, consider the two bounding, adjacent   decimal strings L and U, both having DECIMAL_DIG significant digits, such   that the values of L, D, and U satisfy L <= D <= U. The result should be   one of the (equal or adjacent) values that would be obtained by correctly   rounding L and U according to the current rounding direction, with the extra   stipulation that the error with respect to D should have a correct sign for   the current rounding direction. (DECIMAL_DIG, defined in <float.h>, should   be significantly large that L and U will usually round to the same internal   floating value, but if not will round to adjacent values.)      RETURNS:      The functions return the converted value, if any. If no conversion could be   performed, zero is returned. If the correct value is outside the range of   representable values, plus or minus HUGE_VAL, HUGE_VALF, or HUGE_VALL is   returned (according to the return type and sign of the value), and the value   of the macro ERANGE is stored in errno. If the result underflows (7.12.1),   the functions return a value whose magnitude is no greater than the smallest   normalized positive number in the return type; whether errno acquires the   value ERANGE is implementation-defined.      COPYRIGHT NOTICE:      This file is placed into the public domain by the author, Gregory Pietsch.   Do with it what you wish; just don't pretend that you wrote it.*//* includes needed by this file */#include <ctype.h>#include <errno.h>#include <float.h>#include <limits.h>#include <math.h>#include <stdlib.h>#include <string.h>#include "cpl_conv.h"CPL_CVSID("$Id: cpl_strtod.cpp,v 1.8 2006/06/30 15:05:46 dron Exp $");/* defines *//* SIG_MAX: the maximum number of significant digits we have. The maximum   number of significant digits in a long double must be at least 35,   which is the number of significant digits in an IEEE quad precision   floating-point number (113 * log10(2.0), rounded up). */#define SIG_MAX 40/* Define C99 constants in case they are not defined in <math.h> */#ifndef FP_NAN# define FP_NAN         0#endif /* FP_NAN */#ifndef FP_INFINITE# define FP_INFINITE    1#endif /* FP_INFINITE */#ifndef FP_ZERO# define FP_ZERO        2#endif /* FP_ZERO */#ifndef FP_SUBNORMAL# define FP_SUBNORMAL   3#endif /* FP_SUBNORMAL */#ifndef FP_NORMAL# define FP_NORMAL      4#endif /* FP_NORMAL */#ifndef HUGE_VALF# define HUGE_VALF      HUGE_VAL#endif /* HUGE_VALF */#ifndef HUGE_VALL# define HUGE_VALL      HUGE_VAL#endif /* HUGE_VALL *//* Declare C99 floating point functions, if they are missing in <math.h>. * For the moment we do not have an actual replacements for these functions * and suppose they are exist, even not declared in <math.h>. */#if !defined(HAVE_COPYSIGNF) && !defined(copysignf)# define copysignf copysign#endif /* HAVE_COPYSIGNF */#ifndef HAVE_STRTOF# define strtof strtod#endif /* HAVE_STRTOF */#ifndef HAVE_NAN# define nan(tagp) strtod("NAN(tagp)", NULL)#else# if defined(HAVE_DECL_NAN) && !HAVE_DECL_NANCPL_C_STARTextern  double nan(const char *);CPL_C_END# endif#endif /* HAVE_NAN */#ifndef HAVE_NANF# define nanf(tagp) strtof("NAN(tagp)", NULL)#else# if defined(HAVE_DECL_NANF) && !HAVE_DECL_NANFCPL_C_STARTextern  float nanf(const char *);CPL_C_END# endif#endif /* HAVE_NANF *//* _Memcasecmp: return a case-insensitive comparison of two areas of   memory. */static int_Memcasecmp (const void *s1, const void *s2, size_t n){  const unsigned char *su1, *su2;  for (su1 = (const unsigned char *)s1, su2 = (const unsigned char *)s2;       0 < n;       ++su1, ++su2, --n)    {      if (toupper (*su1) != toupper (*su2))	return (*su1 < *su2 ? -1 : +1);    }  return 0;}/* _Ldmul: multiply y by *px with checking */static int_Ldmul(double *px, double y){  int lexp;  double ld;  ld = frexp(*px, &lexp) * y;  errno = 0;  *px = ldexp(ld, lexp);	/* ldexpl can overflow */  return errno != 0 ? -1 : 0;}    /* _Ldtento: compute x * 10^n, paranoid style */static double_Ldtento(double x, int n){  double factor, fac10;  int errx;  unsigned nu;    if (n == 0 || x == 0.0L)    return x;  factor = 1.0L;  if (n < 0)    {      /* scale down */      nu = -(unsigned)n;      for (fac10 = 10.0L; 0 < nu; nu >>= 1, fac10 *= fac10)        {          if (nu & 1)            factor *= fac10;        }      errx = _Ldmul(&x, 1.0L/factor);    }  else     {      /* scale up */      for (fac10 = 10.0L; 0 < n; n >>= 1, fac10 *= fac10)        {          if (n & 1)            factor *= fac10;        }      errx = _Ldmul(&x, factor);    }  if (0 < errx)    errno = ERANGE;  return x;}/* _Stold: return the classification of the floating-point number (FP_ZERO   if there's no number; if there's a zero, return FP_NORMAL).    The pld parameter is a possibly-returned double;   the pnan parameter is the n-char-sequence returned if we're   returning FP_NAN; point is a symbol used as a decimal delimiter. */static int_Stold (const char *s, char **endptr, double *pld, char **pnan,        const char point){  static const char hexits[] = {    '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e',    'f'  };  char *sc;  char buf[SIG_MAX];  char sign;  double x, fac, facpow;  int ndigit, nsig, nzero, olead, opoint, base;  size_t m;  char *p;  char *pc;  int n;  unsigned long lo[SIG_MAX / 8 + 1], *pl;  short sexp;  long lexp;  char *scsav, esign;  /* Get past the spaces.  */  for (sc = (char *)s; isspace(*sc); ++sc)    ;  /* Get past the initial sign.  */  sign = (*sc == '-' || *sc == '+') ? *sc++ : '+';  /* Do we have INFINITY or INF? */  if (_Memcasecmp (sc, "INFINITY", m = 8) == 0      || _Memcasecmp (sc, "INF", m = 3) == 0)    {      if (endptr)	*endptr = sc + m;      *pld = (sign == '-') ? -1.0L : +1.0L;      return FP_INFINITE;    }  /* Do we have NAN or NAN(nan-char-sequence_opt)? */  if (_Memcasecmp (sc, "NAN(", m = 4) == 0      || _Memcasecmp (sc, "NAN", m = 3) == 0)    {      sc += m;      if (m == 4)	{

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -