美文网首页
字符串转数字

字符串转数字

作者: 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