atol() v/s. strtol()

前端 未结 7 1697
伪装坚强ぢ
伪装坚强ぢ 2020-11-30 19:13

What is the difference between atol() & strtol()?

According to their man pages, they seem to have the same effect as well as matching arguments:

         


        
相关标签:
7条回答
  • 2020-11-30 19:46

    According to the atoi man page, it has been deprecated by strtol.

    IMPLEMENTATION NOTES
    The atoi() and atoi_l() functions have been deprecated by strtol() and strtol_l() 
    and should not be used in new code.
    
    0 讨论(0)
  • 2020-11-30 19:48

    In new code I would always use strtol. It has error handling and the endptr argument allows you to see which part of the string was used.

    The C99 standard states about the ato* functions:

    Except for the behavior on error,they equivalent to

    atoi: (int)strtol(nptr,(char **)NULL, 10)
    atol: strtol(nptr,(char **)NULL, 10)
    atoll: strtoll(nptr, (char **)NULL, 10)

    0 讨论(0)
  • 2020-11-30 20:05

    If memory serves, strtol() has the added benefit to set the (optional) endptr to point to the first character that could not be converted. If NULL, it is ignored. That way if you're processing a string containing numbers and characters mixed, you could continue.

    e.g.,

    char buf[] = "213982 and the rest";
    char *theRest;
    long int num = strtol(buf, &theRest, 10);
    printf("%ld\n", num);    /* 213982 */
    printf("%s\n", theRest); /* " and the rest" */
    
    0 讨论(0)
  • 2020-11-30 20:11

    strtol provides you with more flexibility, as it can actually tell you if the whole string was converted to an integer or not. atol, when unable to convert the string to a number (like in atol("help")), returns 0, which is indistinguishable from atol("0"):

    int main()
    {
      int res_help = atol("help");
      int res_zero = atol("0");
    
      printf("Got from help: %d, from zero: %d\n", res_help, res_zero);
      return 0;
    }
    

    Outputs:

    Got from help: 0, from zero: 0
    

    strtol will specify, using its endptr argument, where the conversion failed.

    int main()
    {
      char* end;
      int res_help = strtol("help", &end, 10);
    
      if (!*end)
        printf("Converted successfully\n");
      else
        printf("Conversion error, non-convertible part: %s", end);
    
      return 0;
    }
    

    Outputs:

    Conversion error, non-convertible part: help
    

    Therefore, for any serious programming, I definitely recommend using strtol. It's a bit more tricky to use but this has a good reason, as I explained above.

    atol may be suitable only for very simple and controlled cases.

    0 讨论(0)
  • 2020-11-30 20:11

    atol(str) is equivalent to

    strtol(str, (char **)NULL, 10);
    

    Use strtol if you want the end pointer (to check whether there are more characters to read or if in fact you have read any at all) or a base other than 10. Otherwise, atol is fine.

    0 讨论(0)
  • 2020-11-30 20:11

    The man page of strtol gives the following:

    ERRORS
       EINVAL (not in C99) The given base contains an unsupported value.
       ERANGE The resulting value was out of range.
       The implementation may also set errno to EINVAL in case no conversion was performed (no digits seen, and 0 returned).
    

    The following code checks for range errors. (Modified Eli's code a bit)

    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include <errno.h>
    
    int main()
    {
       errno = 0;
       char* end = 0;
       long res = strtol("83459299999999999K997", &end, 10);
    
       if(errno != 0)
       {
          printf("Conversion error, %s\n", strerror(errno));
       }
       else if (*end)
       {
          printf("Converted partially: %i, non-convertible part: %s\n", res, end);
       }
       else
       {
          printf("Converted successfully: %i\n", res);
       }
    
       return 0;
    }
    
    0 讨论(0)
提交回复
热议问题