數值范圍
首先看一下各個數值類型的內存大小、取值范圍,便於后面測試類型轉換功能。數值類型的范圍(最大最小值)在<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
}
}