C++數值類型與string、CString之間的轉換


數值范圍

首先看一下各個數值類型的內存大小、取值范圍,便於后面測試類型轉換功能。數值類型的范圍(最大最小值)在<limits>中有定義,可以通過宏定義(INT_MIN、INT_MAX)或類模板的min、max方法(“numeric_limits<T>::max)()”)得到各個數值類型的最大、最小值,代碼如下(將后續用到的頭文件都包含進來):

#include <iostream>   //標准IO
using namespace std;  //標准庫命名空間(cout、string)
#include <limits>     //數值范圍
#include "atlstr.h"   //使用CString類型
#include <string>     //使用string類型
#include <iomanip>    //補齊字符串
#include <sstream>    //使用stringstream需要引入這個頭文件 

int main()
{   
    cout << "char              :" << "\t所占字節數:"<<sizeof(char              ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<char              >::min)() << "\t最大值:" << (numeric_limits<char              >::max)()  << endl;
    cout << "signed char       :" << "\t所占字節數:"<<sizeof(signed char       ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<signed char       >::min)() << "\t最大值:" << (numeric_limits<signed char       >::max)()  << endl;
    cout << "unsigned char     :" << "\t所占字節數:"<<sizeof(unsigned char     ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<unsigned char     >::min)() << "\t最大值:" << (numeric_limits<unsigned char     >::max)()  << endl;
    cout << "short             :" << "\t所占字節數:"<<sizeof(short             ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<short             >::min)() << "\t最大值:" << (numeric_limits<short             >::max)()  << endl;
    cout << "unsigned short    :" << "\t所占字節數:"<<sizeof(unsigned short    ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<unsigned short    >::min)() << "\t最大值:" << (numeric_limits<unsigned short    >::max)()  << endl;
    cout << "int               :" << "\t所占字節數:"<<sizeof(int               ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<int               >::min)() << "\t最大值:" << (numeric_limits<int               >::max)()  << endl;
    cout << "unsigned int      :" << "\t所占字節數:"<<sizeof(unsigned int      ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<unsigned int      >::min)() << "\t最大值:" << (numeric_limits<unsigned int      >::max)()  << endl;
    cout << "long              :" << "\t所占字節數:"<<sizeof(long              ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<long              >::min)() << "\t最大值:" << (numeric_limits<long              >::max)()  << endl;
    cout << "unsigned long     :" << "\t所占字節數:"<<sizeof(unsigned long     ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<unsigned long     >::min)() << "\t最大值:" << (numeric_limits<unsigned long     >::max)()  << endl;
    cout << "long long         :" << "\t所占字節數:"<<sizeof(long long         ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<long long         >::min)() << "\t最大值:" << (numeric_limits<long long         >::max)()  << endl;
    cout << "unsigned long long:" << "\t所占字節數:"<<sizeof(unsigned long long) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<unsigned long long>::min)() << "\t最大值:" << (numeric_limits<unsigned long long>::max)()  << endl;
    cout << "float             :" << "\t所占字節數:"<<sizeof(float             ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<float             >::min)() << "\t最大值:" << (numeric_limits<float             >::max)()  << endl;
    cout << "double            :" << "\t所占字節數:"<<sizeof(double            ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<double            >::min)() << "\t最大值:" << (numeric_limits<double            >::max)()  << endl;
    cout << "long double       :" << "\t所占字節數:"<<sizeof(long double       ) << "\t最小值:"<< setw(20)<< setiosflags(ios::left)<< (numeric_limits<long double       >::min)() << "\t最大值:" << (numeric_limits<long double       >::max)()  << endl;
}

運行結果:

char              :    所占字節數:1   最小值:€                       最大值:
signed char       :    所占字節數:1   最小值:€                       最大值:
unsigned char     :    所占字節數:1   最小值:                        最大值:
short             :    所占字節數:2   最小值:-32768                  最大值:32767
unsigned short    :    所占字節數:2   最小值:0                       最大值:65535
int               :    所占字節數:4   最小值:-2147483648             最大值:2147483647
unsigned int      :    所占字節數:4   最小值:0                       最大值:4294967295
long              :    所占字節數:4   最小值:-2147483648             最大值:2147483647
unsigned long     :    所占字節數:4   最小值:0                       最大值:4294967295
long long         :    所占字節數:8   最小值:-9223372036854775808    最大值:9223372036854775807
unsigned long long:    所占字節數:8   最小值:0                       最大值:18446744073709551615
float             :    所占字節數:4   最小值:1.17549e-38             最大值:3.40282e+38
double            :    所占字節數:8   最小值:2.22507e-308            最大值:1.79769e+308
long double       :    所占字節數:8   最小值:2.22507e-308            最大值:1.79769e+308

注:浮點數的值是關於符號對稱的,其最大值、最小值都是正數,參考 為什么Double.MIN_VALUE不為負

數值類型與string互相轉換

數值類型轉換為string

數值類型轉換為string有使用函數模板+ostringstream、使用標准庫函數std::to_string()兩種方法,前者功能更加通用、使用方便,后者性能更高、應用范圍廣。

使用函數模板+ostringstream

使用函數模板+ostringstream將數值類型(整型、字符型、實型、布爾型)轉換成string,代碼如下:

//函數模板:將常用的數值類型轉換為string類型變量(此方法具有普遍適用性) 
template<typename T> 
string ToString(const T& t)
{
    ostringstream oss;  //創建一個格式化輸出流
    oss << t;           //把值傳遞入流中
    return oss.str();
}

注:使用stringstream需要引入<sstream>頭文件(#include <sstream>)。

測試代碼:

    cout << "ToString函數模板測試結果:" << endl;
    cout << "char              :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<char              >::min)()) << "\t最大值:" << ToString((numeric_limits<char              >::max)()) << endl;
    cout << "signed char       :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<signed char       >::min)()) << "\t最大值:" << ToString((numeric_limits<signed char       >::max)()) << endl;
    cout << "unsigned char     :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<unsigned char     >::min)()) << "\t最大值:" << ToString((numeric_limits<unsigned char     >::max)()) << endl;
    cout << "short             :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<short             >::min)()) << "\t最大值:" << ToString((numeric_limits<short             >::max)()) << endl;
    cout << "unsigned short    :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<unsigned short    >::min)()) << "\t最大值:" << ToString((numeric_limits<unsigned short    >::max)()) << endl;
    cout << "int               :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<int               >::min)()) << "\t最大值:" << ToString((numeric_limits<int               >::max)()) << endl;
    cout << "unsigned int      :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<unsigned int      >::min)()) << "\t最大值:" << ToString((numeric_limits<unsigned int      >::max)()) << endl;
    cout << "long              :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<long              >::min)()) << "\t最大值:" << ToString((numeric_limits<long              >::max)()) << endl;
    cout << "unsigned long     :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<unsigned long     >::min)()) << "\t最大值:" << ToString((numeric_limits<unsigned long     >::max)()) << endl;
    cout << "long long         :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<long long         >::min)()) << "\t最大值:" << ToString((numeric_limits<long long         >::max)()) << endl;
    cout << "unsigned long long:" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<unsigned long long>::min)()) << "\t最大值:" << ToString((numeric_limits<unsigned long long>::max)()) << endl;
    cout << "float             :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<float             >::min)()) << "\t最大值:" << ToString((numeric_limits<float             >::max)()) << endl;
    cout << "double            :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<double            >::min)()) << "\t最大值:" << ToString((numeric_limits<double            >::max)()) << endl;
    cout << "long double       :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << ToString((numeric_limits<long double       >::min)()) << "\t最大值:" << ToString((numeric_limits<long double       >::max)()) << endl;

測試結果:

ToString函數模板測試結果:
char              :    最小值:€                       最大值:
signed char       :    最小值:€                       最大值:
unsigned char     :    最小值:                        最大值:
short             :    最小值:-32768                  最大值:32767
unsigned short    :    最小值:0                       最大值:65535
int               :    最小值:-2147483648             最大值:2147483647
unsigned int      :    最小值:0                       最大值:4294967295
long              :    最小值:-2147483648             最大值:2147483647
unsigned long     :    最小值:0                       最大值:4294967295
long long         :    最小值:-9223372036854775808    最大值:9223372036854775807
unsigned long long:    最小值:0                       最大值:18446744073709551615
float             :    最小值:1.17549e-38             最大值:3.40282e+38
double            :    最小值:2.22507e-308            最大值:1.79769e+308
long double       :    最小值:2.22507e-308            最大值:1.79769e+308

使用標准庫函數std::to_string()

std命令空間下有一個C++標准庫函數std::to_string()可以將數值類型轉換為string,使用時需要include頭文件<string>。參考 to_string ,函數如下:

string to_string(int value);
string to_string(unsigned int value);
string to_string(long value);
string to_string(unsigned long value);
string to_string(long long value);
string to_string(unsigned long long value);
string to_string(float value);
string to_string(double value);
string to_string(long double value);

測試代碼:

    cout << "ToString標准庫函數測試結果:" << endl;
    //char、short類型會自動調用string to_string(int value);
    cout << "char              :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<char              >::min)()) << "\t最大值:" << to_string((numeric_limits<char              >::max)()) << endl;
    cout << "signed char       :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<signed char       >::min)()) << "\t最大值:" << to_string((numeric_limits<signed char       >::max)()) << endl;
    cout << "unsigned char     :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<unsigned char     >::min)()) << "\t最大值:" << to_string((numeric_limits<unsigned char     >::max)()) << endl;
    cout << "short             :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<short             >::min)()) << "\t最大值:" << to_string((numeric_limits<short             >::max)()) << endl;
    cout << "unsigned short    :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<unsigned short    >::min)()) << "\t最大值:" << to_string((numeric_limits<unsigned short    >::max)()) << endl;
    cout << "int               :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<int               >::min)()) << "\t最大值:" << to_string((numeric_limits<int               >::max)()) << endl;
    
    cout << "unsigned int      :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<unsigned int      >::min)()) << "\t最大值:" << to_string((numeric_limits<unsigned int      >::max)()) << endl;
    cout << "long              :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<long              >::min)()) << "\t最大值:" << to_string((numeric_limits<long              >::max)()) << endl;
    cout << "unsigned long     :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<unsigned long     >::min)()) << "\t最大值:" << to_string((numeric_limits<unsigned long     >::max)()) << endl;
    cout << "long long         :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<long long         >::min)()) << "\t最大值:" << to_string((numeric_limits<long long         >::max)()) << endl;
    cout << "unsigned long long:" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<unsigned long long>::min)()) << "\t最大值:" << to_string((numeric_limits<unsigned long long>::max)()) << endl;
    cout << "float             :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<float             >::min)()) << "\t最大值:" << to_string((numeric_limits<float             >::max)()) << endl;
    cout << "double            :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<double            >::min)()) << "\t最大值:" << to_string((numeric_limits<double            >::max)()) << endl;
    cout << "long double       :" << "\t最小值:" << setw(20) << setiosflags(ios::left) << to_string((numeric_limits<long double       >::min)()) << "\t最大值:" << to_string((numeric_limits<long double       >::max)()) << endl;

測試結果:

ToString標准庫函數測試結果:
char              :    最小值:-128                    最大值:127
signed char       :    最小值:-128                    最大值:127
unsigned char     :    最小值:0                       最大值:255
short             :    最小值:-32768                  最大值:32767
unsigned short    :    最小值:0                       最大值:65535
int               :    最小值:-2147483648             最大值:2147483647
unsigned int      :    最小值:0                       最大值:4294967295
long              :    最小值:-2147483648             最大值:2147483647
unsigned long     :    最小值:0                       最大值:4294967295
long long         :    最小值:-9223372036854775808    最大值:9223372036854775807
unsigned long long:    最小值:0                       最大值:18446744073709551615
float             :    最小值:0.000000                最大值:340282(此處省略部分..)16925440.000000
double            :    最小值:0.000000                最大值:179769(此處省略部分..)24858368.000000
long double       :    最小值:0.000000                最大值:179769(此處省略部分..)24858368.000000

注:使用標准庫函數時,char、short類型會調用int類型的轉行函數,浮點數不會啟用科學計數法

string轉換為數值類型

同理,string轉換為數值類型有使用函數模板+istringstream、使用C++標准庫函數兩種方法(不考慮使用C標准庫函數)。

使用函數模板+istringstream

上面使用的是ostringstream,這里要使用的是istringstream,兩者都屬於stringstream。代碼如下:

//函數模板:將string類型變量轉換為常用的數值類型(此方法具有普遍適用性) 
template <class T>
T ToValue(const string& str) {
    istringstream iss(str);
    T num;
    iss >> num;
    return num;
}

使用C++標准庫函數

使用C++11引入的C++庫函數將string轉換為數值類型,函數位於頭文件<string>中,參考string 函數

  • stod:將字符序列轉換為 double 。
  • stof:將字符序列轉換為 float。
  • stoi:將字符序列轉換為 int 。
  • stol:將字符序列轉換為 long 。
  • stold:將字符序列轉換為 long double 。
  • stoll:將字符序列轉換為 long long 。
  • stoul:將字符序列轉換為 unsigned long 。
  • stoull:將字符序列轉換為 unsigned long long 。

CString與string間的互相轉換

CString和string的相互轉換需要考慮多字節和unicode兩種環境,代碼如下:


string CStringToString(CString cs) {
#ifdef _UNICODE

    //如果是unicode工程
    USES_CONVERSION;
    std::string str(W2A(cs));
    return str;
#else
    //如果是多字節工程 
    std::string str(cs.GetBuffer());
    cs.ReleaseBuffer();
    return str;

#endif // _UNICODE 
}

CString StringToCString(string str) {
#ifdef _UNICODE
    //如果是unicode工程
    USES_CONVERSION; 
    CString ans(str.c_str());
    return ans;
#else
    //如果是多字節工程 
    CString ans;
    ans.Format("%s", str.c_str());
    return ans;
#endif // _UNICODE  
}

CString與數值類型間的互相轉換

CString與數值類型有使用string做中轉、使用標准庫函數兩種方法,前者使用更方便一些。

使用string做中轉

改一下函數名稱便於區分,代碼如下:

//模板函數:將CString類型變量轉換為常用的數值類型(此方法具有普遍適用性)
template <class T>
T CStringToValue(const CString& cStr)
{
    string str = ToString(cStr);
    return ToValue<T>(str);
}

//模板函數:將常用的數值類型轉換為CString類型變量(此方法具有普遍適用性)    
template<typename T>
static CString ValueToCString(const T& t)
{
    string str = ToString<T>(t);
    return StringToCString(str);
}

使用標准庫函數

數值類型轉CString可以使用CString.Format函數,可以使用一下格式化字符串:

格式字符串 含義
% c 單個字符
% d 十進制整數(int)
% ld 十進制整數(long)
% f 十進制浮點數(float)
% lf 十進制浮點數(double)
% o 八進制數
% s 字符串
% u 無符號十進制數
% x 十六進制數
代碼如下:
CString str;
str.Format(_T("%d"  ), 123);

CString轉數值類型可以使用定義好的宏函數_ttof 、_ttof 、_ttoi 、_ttol 、_ttoll 、_tcstold 、_tcstoul 、_tcstoull,它們會根據多字節和unicode環境進行切換。

轉換工具類

新建一個Convert的頭文件、命名空間,把上面的方法放在一起便於調用,標准庫函數需要重新包裝一些。代碼如下:

#pragma once
#include <iostream>   //標准IO
#include "atlstr.h"   //使用CString類型
#include <string>     //使用string類型
#include <sstream>    //使用stringstream需要引入這個頭文件 
using namespace std;  //標准庫命名空間(cout、string)

/*
* 源博客頁:https://www.cnblogs.com/timefiles/p/ConvertInValueCStringString.html
* 博客主頁:https://www.cnblogs.com/timefiles
* 創建時間:2021/6/14
*/

namespace Convert
{
    //函數模板:將常用的數值類型轉換為string類型變量(此方法具有普遍適用性) 
    template<typename T>
    string ToString(const T& t)
    {
        ostringstream oss;  //創建一個格式化輸出流
        oss << t;           //把值傳遞入流中
        return oss.str();
    }
    string ToString(const int                &val){ return to_string(val);}                    
    string ToString(const long               &val){ return to_string(val);}
    string ToString(const long long          &val){ return to_string(val);}
    string ToString(const unsigned int       &val){ return to_string(val);}
    string ToString(const unsigned long      &val){ return to_string(val);}
    string ToString(const unsigned long long &val){ return to_string(val);}
    string ToString(const float              &val){ return to_string(val);}
    string ToString(const double             &val){ return to_string(val);}
    string ToString(const long double        &val){ return to_string(val);}


    //函數模板:將string類型變量轉換為常用的數值類型(此方法具有普遍適用性) 
    template <class T>
    T ToValue(const string& str) {
        istringstream iss(str);
        T num;
        iss >> num;
        return num;
    }
    double             ToDouble     (const string& str){ return  stod   (str);}
    float              ToFloat      (const string& str){ return  stof   (str);}
    int                ToInt        (const string& str){ return  stoi   (str);}
    long               ToLong       (const string& str){ return  stol   (str);}
    long double        ToLongDouble (const string& str){ return  stold  (str);}
    long long          ToLongLong   (const string& str){ return  stoll  (str);}
    unsigned long      ToULong      (const string& str){ return  stoul  (str);}
    unsigned long long ToULongLong  (const string& str){ return  stoull (str);}

    namespace MFC 
    {
        
        //將CString類型變量轉換為string類型
        string CStringToString(CString cs) {
#ifdef _UNICODE

            //如果是unicode工程
            USES_CONVERSION;
            std::string str(W2A(cs));
            return str;
#else
            //如果是多字節工程 
            std::string str(cs.GetBuffer());
            cs.ReleaseBuffer();
            return str;

#endif // _UNICODE 
        }

        //將string類型變量轉換為CString類型
        CString StringToCString(string str) {
#ifdef _UNICODE
            //如果是unicode工程
            USES_CONVERSION;
            CString ans(str.c_str());
            return ans;
#else
            //如果是多字節工程             
            CString ans;
            ans.Format("%s", str.c_str());
            return ans;
#endif // _UNICODE  
        }


        //模板函數:將CString類型變量轉換為常用的數值類型(此方法具有普遍適用性)
        template <class T>
        T ToValue(const CString& cStr)
        {
            string str = CStringToString(cStr);
            return  Convert::ToValue<T>(str);
        }

        //模板函數:將常用的數值類型轉換為CString類型變量(此方法具有普遍適用性)    
        template<typename T>
        static CString ToCString(const T& t)
        {
            string str = Convert::ToString<T>(t);
            return StringToCString(str);
        }


        CString ToString(const int                &val){ CString str;str.Format(_T("%d"  ), val); return str;}                    
        CString ToString(const long               &val){ CString str;str.Format(_T("%ld" ), val); return str;} 
        CString ToString(const unsigned int       &val){ CString str;str.Format(_T("%ud" ), val); return str;}
        CString ToString(const unsigned long      &val){ CString str;str.Format(_T("%uld"), val); return str;}  
        CString ToString(const float              &val){ CString str;str.Format(_T("%f"  ), val); return str;}
        CString ToString(const double             &val){ CString str;str.Format(_T("%lf" ), val); return str;}


        double             ToDouble     (const CString& str){ return  _ttof     (str);}
        float              ToFloat      (const CString& str){ return  _ttof     (str);}
        int                ToInt        (const CString& str){ return  _ttoi     (str);}
        long               ToLong       (const CString& str){ return  _ttol     (str);}
        long long          ToLongLong   (const CString& str){ return  _ttoll    (str);}       

#ifdef _UNICODE
        //如果是unicode工程
        long double        ToLongDouble (const CString& str){ wchar_t * pEnd; return  _tcstold  (str, & pEnd);}
        unsigned long      ToULong      (const CString& str){ wchar_t * pEnd; return  _tcstoul  (str, & pEnd,10);}
        unsigned long long ToULongLong  (const CString& str){ wchar_t * pEnd; return  _tcstoull (str, & pEnd,10);}
#else
        //如果是多字節工程             
        long double        ToLongDouble (const CString& str){ char * pEnd; return  _tcstold  (str, & pEnd);}
        unsigned long      ToULong      (const CString& str){ char * pEnd; return  _tcstoul  (str, & pEnd,10);}
        unsigned long long ToULongLong  (const CString& str){ char * pEnd; return  _tcstoull (str, & pEnd,10);}
#endif // _UNICODE  
        
    }
}

參考資料


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM