美文网首页
字符串转数字

字符串转数字

作者: XDgbh | 来源:发表于2018-08-18 14:50 被阅读21次
    /*
    问题:把一个字符串转换成整数。(发散思维:通常在存储非常非常大的数时,会考虑存储为字符串或者数组)
    提前想好测试用例:
    1、输入空字符串
    2、输入的字符串包含非数字字符
    3、输入是全数字的字符串,但是前面全是0
    4、正常的全数字字符串输入,但要注意正数还是负数
    注意点:
    要考虑字符串太长,用多大的整数类型来存入。
    int和long和(long int)都是4字节只能存32位,最高位0代表正数,1代表负数,实际只有31位,只能存-(2^32-1)到2^32-1,即-4294967295到4294967295
    (long long)是8字节可以存64位,实际可用63位,能存-(2^64-1)到2^64-1,几乎是一个长度为20的数字
    优化:
    考虑输入字符串长度小于10位是,可以用int来存放。大于10位时用 long long来存放
    只是数据类型不同,算法思想应该一样,因此,可以用C++的函数模板来做。
    */
    #include<iostream>
    #include<assert.h>
    #include<string>
    
    using namespace std;
    
    //函数模板,泛型编程,输入字符串小于10位时,用int型,10位及以上用long long型整数
    // 所有函数都可以加inline关键字,做成内联函数每次调用的时候不远程而是把所有函数代码复制过去,
    // 可以提升速率,但是更耗内存,我们不用考虑inline函数的内存影响,编译器会自动判断是否执行inline
    template <typename NumType>
    inline NumType StrToNum(const char *inputStr)
    {
        int sign_flag = 0;  //数字的正负性,0为正,1为负
        NumType number = 0;
        //首先判断输入的合法性
        assert(inputStr);
        if (*inputStr == '-')
        {
            //排除负数的干扰
            inputStr++;
            sign_flag = 1;
        }
        else if (*inputStr == '+')
        {
            //排除正数符号的干扰
            inputStr++;
        }
        //首数字等于0则跳过
        while (*inputStr && *inputStr == '0')
        {
            inputStr++;
        }
        //开始处理正常的数字
        while (*inputStr)
        {
            assert(*inputStr >= '0' && *inputStr <= '9');
            number = number * 10 + *inputStr - '0';
            inputStr++;
        }
        if (sign_flag)
        {
            //是负数
            number = -number;
        }
        return  number;
    }
    
    //方法二:先确认字符串都是合法的数字类字符,然后可用<string>的字符串转字符函数直接转
    // stoi(strNum)是将字符串转成int型和 stoll(strNum)转成long long型,若非数字开头则直接中断,否则输出直到非数字后正常退出
    
    int main()
    {
        int intSize = sizeof(int);
        int longSize = sizeof(long);
        int longintSize = sizeof(long int);
        int longlongSize = sizeof(long long);
        cout << intSize << endl;    //打印4 (字节)
        cout << longSize << endl;   //4
        cout << longintSize << endl;//4
        cout << longlongSize << endl;//8
        cout << "please input string of numbers no more then 30lens:" ;
        char strNum[30] = "";
        //string strNum(30, 0); //从键盘读入一般都针对char指针
        //cin >> strNum;        //不可行。不会判断是否溢出,如果溢出则程序崩溃。遇空格、回车、TAB则结束读入。常用于读入一个字符
        //cin.get(strNum, 10, 'a'); //可行。指定接收的字符个数,输入的超出会自动舍弃。默认'\n'结束,可以换为其他字符
        cin.getline(strNum, 30);    //可行,而且可以指定以什么字符结束,默认是'\n'。与cin.get()的区别是最后输入的换行符会被清除,不会留在缓冲区
        cout << "get input char[30]_string:"<<strNum<<endl;
        string strNum_obj;
        cout << "input string numbers:";
        getline(cin, strNum_obj);   //用这个读入字符串存入string对象,就可以不提前指定长度,很方便。可以理解string对象就是动态长度的字符串
        if (strlen(strNum_obj.data()) < 10)     //或者可以直接用strNum_obj.length()得到字符串长度
        {
            //cout << "stoi:" << stoi(strNum) << endl;
    
            //数字位数小于10,模板函数使用int型
            cout << "return number:" << StrToNum<int>(strNum_obj.data());
        }
        else if (strNum_obj.length() < 20)
        {
            //cout << "stoll:" << stoll(strNum) << endl;
    
            //数字位数10到20,模板函数使用long long型
            cout << "return number:" << StrToNum<long long>(strNum_obj.data());
        }
        else
        {
            cout << "the string are too long to convert to numbers !"<<endl;
        }
    
        return 0;
    }
    

    相关文章

      网友评论

          本文标题:字符串转数字

          本文链接:https://www.haomeiwen.com/subject/zjnhsftx.html