C++ 从cstring函数向string类成员函数迁移
C字符串和string类互转
#include <iostream>
#include <string>
int main(void)
{
//C字符串 转 string类
const char *s1 = "hello";
std::string s2 = std::string(s1);
std::cout<<s2<<std::endl;
//string类 转 C字符串
std::string s3 = "world";
char *s4 = (char*) s3.data(); //或 .c_str(),C++11后两者完全等价
std::cout<<s4<<std::endl;
return 0;
}
string、string.h、cstring、CString的区别与联系
⑴ string是C++标准库里面的string模板,也包含了C风格字符串操作的库函数,因此可以定义string类,使用时需声明namespace std。
⑵ string.h是C风格字符串操作的一个库函数,因此使用#include <string.h>时是不能定义string类的,使用时不需声明namespace std。
⑶ cstring是把string.h放到std中,它的功能和string.h一样,因为使用了std,所以使用时需要声明namespace std。
⑷ CString属于MFC的类,Visual C++中最常用的字符串类,继承自CSimpleStringT类,所以使用CString时要包含afx.h头文件。
string类和C字符串的简单对比
声明字串
string s;
char s[100];
字串长度
s.length() 或 s.size()
strlen(s)
首尾字符
s.front() 和 s.back() 或 s[0] 和 s[s.size()-1]
s[0] 和 s[strlen(s)-1]
字符引用
s[i] 或 s.at(i)
s[i]
读取一行
getline(cin, s);
gets(s);
字串赋值
s="ABCD";
strcpy(s, "ABCD");
字串连接
s=s+"ABCD"; 或 s+="ABCD";
strcat(s, "ABCD");
字串比较
s=="ABCD";
strcmp(s, "ABCD");
string类与C字符串对比,感觉前者操作比较直观, 另外还不用自己管理内存,长度几乎没有限制并且可以任意加长缩短。
字符串的最大长度
字符串char s[],char*s 的长度受限于操作系统给应用程序分派的栈内存、堆内存的大小。关于C++内存分配详见文末转载的附录。
string类字符串占用内存是动态的,它的上限可以通过string::max_size()函数获得。32位系统约4G字节,64位系统约4E字节(1E = 1KP = 1MT = 1GG = 2^60);甚至有些超大型服务器上是16E字节。不过实际上使用时最大长度也受限于计算机内存容量的,在内存可以承受的范围内,可以简单地认为string支持的字符串大小没有上限,或者上限大到几乎用不到。
#include <iostream>
#include <string>
using namespace std;
int main(void)
{
string str;
cout<<str.max_size()<<endl; //4611686018427387897<--64位系统 ≈4G个1G字节
cout<<unsigned(-1)<<endl; //4294967295<--32位系统通常会等于这个值,=4G-1字节
return 0;
}
提示: 打开网址 cpp.sh/8krms,点RUN查看运行结果。
cstring库函数和string类成员函数对比
strlen、strnlen
功能:串长度,从字符串的首字符起到第一个 '\0' 前的字符止的字符个数;如果仅声明没赋初值,结果是不定的,计数到 '\0' (不含)才会停止。
用法:size_t strlen(const char *s);
size_t strlen(const char *s, size_t n);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
size_t size=0;
const char* str="Hello,world!";
size=strlen(str);
cout<<"Size of the string is "<<size<<endl;
size_t n;
n=5;
size=strnlen(str,n);
cout<<"Size of the string is "<<size<<endl;
n=20;
size=strnlen(str,n);
cout<<"Size of the string is "<<size<<endl;
//strnlen 计数到n或遇到'\0'止
//返回值:size=(n>=strlen(str))?strlen(str):n
return 0;
}
提示: 打开网址 cpp.sh/6b6hj,点RUN查看运行结果。
string::size、string::length 串长
#include <iostream>
#include <string>
using namespace std;
int main ()
{
string str="Hello,world!";
cout<<str.size()<<endl;
cout<<str.length()<<endl;
//size() length() 两者完全等价
return 0;
}
提示: 打开网址 cpp.sh/3nm3z,点RUN查看运行结果。
strcpy、strncpy
功能:串复制,将源字符串src全部、或最多n个字符复制到目标字符数组dest中,返回指向dest的指针。
用法:char *strcpy(char *dest, char *source);
char *strncpy(char *dest, char *src, size_t n);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
const char *str1="123456789";
char deststring[20]; //目标字符串数组长度得足够容纳源字符串
strcpy(deststring,str1); //全部复制
cout<<deststring<<endl; //结果为 123456789
const char *str2="hello,world!";
strncpy(deststring,str2,5); //部分复制5个字符,注意没得到'\0'
cout<<deststring<<endl; //结果为 hello6789
cout<<strlen(deststring)<<endl;
deststring[5]='\0'; //设置字符串结束标志
cout<<deststring<<endl; //结果为 hello
cout<<strlen(deststring)<<endl;
return 0;
}
提示: 打开网址 cpp.sh/5yiqo,点RUN查看运行结果。
string::string、string::assign(string::operator=) 构造与赋值
#include <iostream>
#include <string>
#include <array>
#include <vector>
using namespace std;
int main ()
{
string dest,dest2;
string src="Hello,world!";
dest.assign(src); //对比strcpy
cout<<dest<<endl;
dest2=src; //或直接用=赋值
cout<<dest2<<endl;
int n=5;
dest=string(src.begin(),src.begin()+n); //对比strncpy
cout<<dest<<endl;
char buf[20]="Hello,world!";
dest=string(buf,buf+n); //从字符数组取值
cout<<dest<<endl;
n=2;
array<char,5>arr={'H','e','l','l','o'};
dest=string(arr.begin(),arr.begin()+n); //从array模板类取值
cout<<dest<<endl;
vector<char>vct={'H','e','l','l','o'};
dest=string(vct.begin(),vct.begin()+n); //从vector容器取值
cout<<dest<<endl;
//string类的本质和字符型vector或字符array类比较类同
return 0;
}
提示: 打开网址 cpp.sh/6klz7,点RUN查看运行结果。
strset、strnset
功能:置字符,将字符串str中全部或部分前n个字符设置为字符c,返回指向dest的指针。
用法:char *strcat(char *str, char c);
char *strncat(char *str, char c, size_t n);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
char str[]="Hello,world!";
char c='*';
char *p;
p=strnset(str,c,5);
cout<<p<<endl; //显示:*****,world!
cout<<str<<endl; //显示:*****,world!
p=strset(str,c);
cout<<p<<endl; //显示:************
cout<<str<<endl; //显示:************
return 0;
}
提示: 某些编译器已废止弃用这两个函数。
string::replace+string::string or string::assign+string::substr
无对应函数,可以用多种函数组合的方法实现类似功能。
#include <iostream>
#include <string>
using namespace std;
int main(void)
{
char c='*';
string str="Hello,world!";
str=str.replace(0,5,string(5,c));
cout<<str<<endl;
str="Hello,world!";
str=str.replace(str.begin(),str.begin()+5,string(5,c));
cout<<str<<endl;
//也能用其他函数组合实现:
str="Hello,world!";
str=str.assign(5,c)+str.substr(5,str.size());
cout<<str<<endl;
str="Hello,world!";
str=string(5,c)+str.substr(5,str.size());
cout<<str<<endl;
return 0;
}
提示: 打开网址 cpp.sh/3ez23,点RUN查看运行结果。
strcat、strncat
功能:串连接,把源串src的全部或前n个字符接到目标串dest尾,返回指向dest的指针。
用法:char *strcat(char *dest, char *src);
char *strncat(char *dest, char *src, size_t n);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
char dest[8];
const char *Dev="Dev-",*CPP="C++";
char *str;
strcpy(dest,Dev);
cout<<dest<<endl;
str=strcat(dest,CPP);
cout<<dest<<endl;
cout<<str<<endl;
return 0;
}
提示: 打开网址 cpp.sh/2mddy,点RUN查看运行结果。
std::operator+ 字符串连接
#include <iostream>
#include <string>
using namespace std;
int main(void)
{
string cpp,hello="Hello,",Dev="Dev-",c="C++";
cpp=hello+Dev+c; //直接用加号连接
cout<<cpp<<endl;
int n=5;
Dev+=c; //还能用 +=
cpp=hello+string(Dev.begin(),Dev.begin()+n); //对比 strncat()
cout<<cpp<<endl;
return 0;
}
提示: 打开网址 cpp.sh/7oumk,点RUN查看运行结果。
strcmp、strncmp
功能:串比较,从索引0位置开始比Ascii码,若str1>str2,返回值 > 0;反之 < 0;两串相等,返回0。
用法:int strcmp(char *str1, char *str2);
int strncmp(char *str1, char *str2, size_t n);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
const char *buf1="aaa", *buf2="bbb", *buf3="ccc";
int ptr;
ptr=strcmp(buf2,buf1);
if(ptr>0)
cout<<"buf2 is greater than buf1"<<endl;
else
cout<<"buf2 is less than buf1"<<endl;
ptr=strcmp(buf2,buf3);
if(ptr>0)
cout<<"buf2 is greater than buf3"<<endl;
else
cout<<"buf2 is less than buf3"<<endl;
return 0;
}
提示: 打开网址 cpp.sh/37okg,点RUN查看运行结果。
string::compare、std::operator== (>、<、>=、<=) 字符串比较
#include <iostream>
#include <string>
using namespace std;
int main(void)
{
string str1="abc";
string str2="aba";
string str3="abc";
cout << str1.compare(str2) <<endl; //1
cout << str1.compare(str3) <<endl; //0
cout << (str1==str2) <<endl; //0
cout << (str1>str2) <<endl; //1
cout << (str1<str2) <<endl; //0
cout << (str1>=str2) <<endl; //1
cout << (str1<=str2) <<endl<<endl; //0
cout << (str1==str3) <<endl; //1
cout << (str1>str3) <<endl; //0
cout << (str1<str3) <<endl; //0
cout << (str1>=str3) <<endl; //1
cout << (str1<=str3) <<endl; //1
return 0;
}
提示: 打开网址 cpp.sh/62ifh,点RUN查看运行结果。compare()在网页版上的返回值稍有不同。
strlwr、strupr
功能:串大小写,字符串的字母转化为小写字母或大写字母,字母之外的标点和数字不变。
用法:char *strlor(char *str); //字串中的字母都 tolower(c)
char *strupr(char *str); //字串中的字母都 toupper(c)
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
char buf[]="Dev-C++ 5.11";
cout<<strlwr(buf)<<endl; //显示:dev-c++ 5.11
cout<<strupr(buf)<<endl; //显示:DEV-C++ 5.11
return 0;
}
提示: 某些编译器已废止弃用这两个函数。
transform(s.begin(), s.end(), s.begin(), ::tolower); 、::toupper
无对应函数,用transform算法把全串字符逐个转小写或大写。需要#include<algorithm>
#include <iostream>
#include <string>
#include <algorithm>
using namespace std;
int main(void)
{
string t;
cout << "请输入字符串:";
cin >> t;
transform(t.begin(),t.end(),t.begin(),::tolower); //转小写
cout<<"转换成小写之后:" << t << endl;
transform(t.begin(),t.end(),t.begin(),::toupper); //转大写
cout<<"转换成大写之后:" << t << endl;
return 0;
}
提示: 打开网址 cpp.sh/8zs4o,点RUN查看运行结果。
stricmp\strcmpi、strnicmp
功能:串比较,以大小写不敏感方式比较。
用法:int stricmp(char *str1, char *str2); 也可写成strcmpi() 其宏定义
int strnicmp(char *str1, char *str2, size_t n);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
int ptr;
const char *buf1="BBB", *buf2="bbb";
ptr=stricmp(buf2,buf1);
if(ptr>0)
cout<<"buf2 is greater than buf1"<<endl;
else if(ptr<0)
cout<<"buf2 is less than buf1"<<endl;
else if(ptr==0)
cout<<"buf2 equals buf1"<<endl;
return 0;
}
注:无对应函数,先把两个字串同转大写或小写再进行比较。
strchr、strrchr
功能:查找指定字符在串中首次出现的位置,返回一个指向str中指定字符首次出现的指针,如找不到则返回一个空指针。前者为从左到右正方向查找,后者为从右向左反方向查找。
用法:char *strchr(char *str, char c);
char *strrchr(char *str, char c);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
char str[20];
char c='s';
strcpy(str,"It\'s a string"); //位置索引从0开始,3即是第4个字符
char *p=strchr(str,c);
if(p) //p-str==3 指针减法运算
cout<<"The character "<<c<<" is at position:"<<p-str<<endl;
else
cout<<"The character was not found!"<<endl;
cout<<p<<endl; //指针已移位,显示为:s a string
char *q=strrchr(str,c); //反向查找,位置索引还是从0开始,7即是第8个字符
if(q) //q-str==7 指针减法运算
cout<<"The character "<<c<<" is at position:"<<q-str<<endl;
else
cout<<"The character was not found!"<<endl;
cout<<q<<endl; //指针已移位,显示为:string
return 0;
}
提示: 打开网址 cpp.sh/4m2on,点RUN查看运行结果。
strstr
功能:查找子串,指定子串在字符串中首次出现的位置。返回一个指向str中首次出现substr的指针,如找不到则返回一个空指针。
用法:char *strstr(char *str, char *substr);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
char str[20];
char s[4]="str";
strcpy(str,"It\'s a string"); //位置索引从0开始,7即是第8个字符开始
char *p=strstr(str,s);
if(p) //p-str==7 指针减法运算
cout<<"The substring "<<s<<" is at position:"<<p-str<<endl;
else
cout<<"The substring was not found!"<<endl;
cout<<p<<endl; //指针已移位,显示为:string
return 0;
}
提示: 打开网址 cpp.sh/4tkva ,点RUN查看运行结果。
string::find、string::rfind 查找字符或子串
#include <iostream>
#include <string>
using namespace std;
int main(void)
{
string str="It\'s a string.";
char c='s';
string::size_type pos; //或声明为 size_t pos; 如串长不是巨大的话可用int代替
//C++函数重载,查找字符或子串用同一函数名
pos=str.find(c); //显示:3 正向查找,类似于strchr(str,c)
cout<<pos<<endl;
pos=str.rfind(c); //显示:7 反向查找,类似于strrchr(str,c)
cout<<pos<<endl;
string substr="string";
pos=str.find(substr); //显示:7
cout<<pos<<endl;
pos=str.rfind(substr); //显示:7
cout<<pos<<endl;
//另可以指定查找位置,返回的还在全串的位置
pos=str.find(substr,5); //显示:7
cout<<pos<<endl;
pos=str.find(substr,8); //没找到则返回一个很大的数=Maximum value for size_t
//(实际上就是成员常量string::npos) 声明int型则返回-1
cout<<pos<<endl;
// string::npos 的用法:
if (pos != string::npos) //string::npos 可以用 str.npos 替代
cout << "position is:" << pos << endl;
else
cout << "Not found the substring:" + substr; //此处没找到的原因,设置了开始位置 8
return 0;
}
提示: 打开网址 cpp.sh/6skc7,点RUN查看运行结果。
strspn、strcspn
功能:查找前缀,计算字符串str中从头开始连续有几个字符都 [属于/不属于] 字符串control。前者返回第一次失配前匹配的字符数,后者返回第一次匹配前失配的字符数。
用法:size_t strspn (const char *str, const char *control);
size_t strcspn (const char *str, const char *control);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
const char *str="(3+3/7)*7 or (4-4/7)*7 = 24";
cout<<strspn(str,"()")<<endl; //返回1,str前部连续的有1个字符匹配,即数字3之前的字符数
cout<<strspn(str,"37()+-*/")<<endl; //返回9,str前部连续有9个字符匹配,即空格前的字符数
cout<<strspn(str,"1234567890")<<endl<<endl; //返回0,字符串str非数字起始
cout<<strcspn(str," ")<<endl; //返回9,str前部9个字符未出现空格
cout<<strcspn(str,"+-*/")<<endl; //返回2,str前部2个字符未出现+-*/
cout<<strcspn(str,"1234567890")<<endl; //返回1,即第1个字符不是数字
return 0;
}
提示: 打开网址 cpp.sh/34vlm,点RUN查看运行结果。
string::find_first_of、string::find_first_not_of
#include <iostream>
#include <string>
using namespace std;
int main(void)
{
size_t index,last=0;
string s="(3+3/7)*7 or (4-4/7)*7 = 24";
string separator="37()+-*/";
index=s.find_first_of(separator,last);
while (index==last++)
index=s.find_first_of(separator,last);
cout<<"第一次失配前已匹配的字符数:"<<last-1<<endl; //显示9,即最后一个匹配的位置索引+1
last=0;
separator="or";
do index=s.find_first_not_of(separator,last++); //注意:do-while和while-do的区别
while (index==last-1);
cout<<"第一次匹配前已失配的字符数:"<<last-1<<endl; //显示10,即最后一个失配的位置索引+1
return 0;
}
提示: 打开网址 cpp.sh/4fi7x,点RUN查看运行结果。
strpbrk
功能:查找字符集,查找breakset里任一字符第一次出现在dest里的位置,返回dest中第一次出现breakset中字符的指针,如果不存在返回NULL。
用法:char *strpbrk(const char *dest, const char *breakset);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
const char *str="24 = (3+3/7)*7 or (4-4/7)*7";
const char *breakset="()^%"; //分界符中至少有一个字符属于str,否则返回NULL
str=strpbrk(str,breakset); //左(最先出现,此时str=="(3+3/7)*7 or (4-4/7)*7"
if (str) cout<<str<<endl; //cout输出NULL,程序会异常退出
//用do-while计算除分界符之外的字符数
unsigned int cnt=0;
const char *separator="()+-*/ or";
do {
str=strpbrk(str,separator);
if (str==NULL) break; //找不到了就中断循环,否则以下操作会退出
cout<<cnt<<":"<<str<<endl; //本行测试用,可去掉
str+=strspn(str,separator);
++cnt;
} while(str && *str);
//cout<<str; //此时str==NULL,此处显示str就会中止退出
cout<<"\nThere are "<<cnt<<" digits."<<endl;
return 0;
}
提示: 打开网址 cpp.sh/9tgqj,点RUN查看运行结果。
string::find + string::substr
无对应函数,以上两个函数组合可以模拟strpbrk例程的效果。
#include <iostream>
#include <string>
using namespace std;
int main(void)
{
string str="(3+3/7)*7 or (4-4/7)*7";
string sep="()+-*/ or";
int index=0,cnt=0;
for(auto s:str){
if (sep.find(s)==string::npos)
cout<<cnt++<<":"<<str.substr(index-1,str.size())<<endl;
index++;
}
//cout<<str; //此时str还是原值,string类的优点之一
cout<<"\nThere are "<<cnt<<" digits."<<endl;
return 0;
}
提示: 打开网址 cpp.sh/2iuoy,点RUN查看运行结果。
strtok、(strtok_r、strtok_s)
功能:串分割,把字符串中用分界符分割成多个子串。saveptr记录的是每次被分割后剩余的部分。
用法:char *strtok(char *s, const char *delim);
char *strtok_r(char *str, const char *delim, char **saveptr); //linux版本
char *strtok_s(char *strToken, const char *strDelimit, char **buf); //windows版本
#include <iostream>
#include <cstring>
#include <vector>
using namespace std;
int main(void)
{
const char *tmp = "hello,Dev-C++";
char str[20];
strcpy(str,tmp);
cout<<"str初值:"<<str<<endl;
char *token = strtok(str,",");
cout<<token<<endl;
token=strtok(NULL,","); //str必须包含分界符,否则此行出错
cout<<token<< endl;
cout<<"str终值:"<<str<<endl; //str被改变了!
//strtok(str,delim)允许同时有多个分界符,但delim中至少有一个字符是str的子串
cout<<"\n多个子串可用while循环来分隔"<<endl;
strcpy(str,tmp); //重新取值
token=strtok(str,",-");
while(token!=NULL){
cout<<token<<endl;
token=strtok(NULL,",-");
}
cout<<"\n把分割出来的子串装入容器中"<<endl;
strcpy(str,tmp); //重新取值
vector<string>vect;
token=strtok(str,",-");
while(token!=NULL){
vect.push_back(token);
token=strtok(NULL,",-");
}
for (auto v:vect) cout<<v<<endl;
return 0;
}
提示: 打开网址 cpp.sh/5fa2d,点RUN查看运行结果。
string::find_first_of + string::substr 串分割
#include <iostream>
#include <string>
#include <vector>
using namespace std;
int main(void)
{
vector<string>vect;
size_t index,last=0;
string s="Hello,Dev-C++";
string delim=",-";
index=s.find_first_of(delim,last);
while (index!=s.npos){
vect.push_back(s.substr(last,index-last));
last=index+1;
index=s.find_first_of(delim,last);
}
if (s.length()-last>0)
vect.push_back(s.substr(last));
for(auto v:vect) cout<<v<<endl;
return 0;
}
提示: 打开网址 cpp.sh/92cka,点RUN查看运行结果。
strrev
功能:串倒序,所有字符反转顺序。
用法:char *strrev(char *str);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
char str[15];
strcpy(str,"Hello,world!");
cout<<str<<endl;
char *p=strrev(str);
cout<<p<<endl; //显示: !dlrow,olleH
return 0;
}
提示: 某些编译器已废止弃用这个函数。
std::reverse(string::begin, string::end) 串倒序
或 string::assign(string::rbegin, string::rend)
用reverse算法全串倒序,需要#include<algorithm>;或用assign()函数反向赋值。
#include <iostream>
#include <string>
#include <cstring>
#include <algorithm>
using namespace std;
int main(void)
{
string str="Hello,world!";
reverse(str.begin(),str.end()); //必须包括<algorithm>头文件
cout<<str<<endl; //显示: !dlrow,olleH
//或用string::assign + string::rbegin rend
str.assign(str.rbegin(),str.rend());
cout<<str<<endl; //显示: Hello,world!
//reverse()算法也适用于C字符串、字符数组
char *c=(char*)str.c_str();
reverse(c,c+strlen(c));
cout<<c<<endl; //显示: !dlrow,olleH
char s[20];
strcpy(s,c);
reverse(s,s+strlen(s));
cout<<s<<endl; //显示: Hello,world!
return 0;
}
提示: 打开网址 cpp.sh/6vzuk ,点RUN查看运行结果。
swab
功能:交换字节,奇偶数位置的字符相互交换位置,nbytes通常要求是偶数。
用法:void swab (char *from, char *to, int nbytes);
#include <iostream>
#include <cstring>
using namespace std;
int main(void)
{
char source[11]="1234567890";
char target[11];
swab(source,target,strlen(source));
cout<<source<<" | "<<target<<endl; //显示:1234567890 | 2143658709
return 0;
}
string::swap
无对应函数,有个string::swap(string str1,string str2)函数,但它的功能是交换两个字符串的值。
#include <iostream>
#include <string>
using namespace std;
int main(void)
{
string source="1234567890";
string target;
target.swap(source);
cout<<source<<" | "<<target<<endl;
return 0;
}
string Swab(string str);
自定义函数模拟实现swab()
#include<iostream>
#include<string>
using namespace std;
string Swab(string);
int main(void)
{
string str="1234567890";
cout<<str<<" | "<<Swab(str)<<endl;
return 0;
}
string Swab(string str)
{
int i,j=0;
for (auto&s:str){
if (j%2==1){
i=str.at(j-1);
str.at(j-1)=s;
s=i;
}
j++;
}
return str;
}
提示: 打开网址 cpp.sh/8giax,点RUN查看运行结果。
strtol、strtod、strtoul
功能:字符串转整数或浮点数,转换前会跳过空格字符,如有不合条件的字符由endptr传回。非<cstring>库函数,包含在头文件 <cstdlib> 或 <stdlib.h>
用法:long strtol(char *str, char **endptr, int base); //base为整数的进制基数,十进制base==10
double strtod(char *str, char **endptr); //转浮点数,可以转换的字符有+-Ee.及数字
unsigned long strtoul(const char *nptr,char **endptr,int base); //转无符号长整型
#include <iostream>
#include <cstdlib>
using namespace std;
int main()
{
char *endptr;
char a[] = "123.4567890";
char b[] = "123.456Hann";
char c[] = "-123.456E-7";
cout<<strtol(a,NULL,10)<<endl;
cout<<a<<endl<<endl;
cout<<strtol(a,&endptr,10)<<endl;
cout<<a<<endl; //a未被改变
cout<<endptr<<endl<<endl;
cout<<strtod(b,&endptr)<<endl;
cout<<b<<endl; //b未被改变
cout<<endptr<<endl<<endl;
cout<<strtod(c,NULL)<<endl;
}
提示: 打开网址 cpp.sh/94bjb,点RUN查看运行结果。
用stringstream类转数字
#include <iostream>
#include <sstream>
#include <string>
using namespace std;
template<typename _Type>
_Type str2number(string s)
{
_Type num;
stringstream ss;
ss<<s;
ss>>num;
ss.clear();
return num;
}
int main(void)
{
string str="1234.56789e-5";
long double a;
long int b;
double c;
int d;
a=str2number<long double>(str);
b=str2number<long int>(str);
c=str2number<double>(str);
d=str2number<int>(str);
cout<<a<<endl;
cout<<b<<endl;
cout<<c<<endl;
cout<<d<<endl;
return 0;
}
提示: 打开网址 cpp.sh/6s6as,点RUN查看运行结果。
以上代码均在Dev-C++ 5.11上通过无错编译,其他字符串相关文章请阅《C++ <cstring>字符串库函数的自定义实现》
附录:
【转载】C++内存分配方式详解——堆、栈、自由存储区、全局/静态存储区和常量存储区
栈,就是那些由编译器在需要的时候分配,在不需要的时候自动清除的变量的存储区。里面的变量通常是局部变量、函数参数等。在一个进程中,位于用户虚拟地址空间顶部的是用户栈,编译器用它来实现函数的调用。和堆一样,用户栈在程序执行期间可以动态地扩展和收缩。
堆,就是那些由 new 分配的内存块,他们的释放编译器不去管,由我们的应用程序去控制,一般一个 new 就要对应一个 delete。如果程序员没有释放掉,那么在程序结束后,操作系统会自动回收。堆可以动态地扩展和收缩。
自由存储区,就是那些由 malloc 等分配的内存块,他和堆是十分相似的,不过它是用 free 来结束自己的生命的。
全局/静态存储区,全局变量和静态变量被分配到同一块内存中,在以前的 C 语言中,全局变量又分为初始化的和未初始化的(初始化的全局变量和静态变量在一块区域,未初始化的全局变量与静态变量在相邻的另一块区域,同时未被初始化的对象存储区可以通过 void* 来访问和操纵,程序结束后由系统自行释放),在 C++ 里面没有这个区分了,他们共同占用同一块内存区。
常量存储区,这是一块比较特殊的存储区,他们里面存放的是常量,不允许修改(当然,你要通过非正当手段也可以修改,而且方法很多)
明确区分堆与栈
在 BBS 上,堆与栈的区分问题,似乎是一个永恒的话题,由此可见,初学者对此往往是混淆不清的,所以我决定拿他第一个开刀。
首先,我们举一个例子: void f() { int* p=newint[5]; }
这条短短的一句话就包含了堆与栈,看到 new,我们首先就应该想到,我们分配了一块堆内存,那么指针 p 呢?他分配的是一块栈内存,所以这句话的意思就是:在栈内存中存放了一个指向一块堆内存的指针 p。在程序会先确定在堆中分配内存的大小,然后调用 operator new 分配内存,然后返回这块内存的首地址,放入栈中,他在 VC6 下的汇编代码如下:
00401028push 14h
0040102Acall operator new (00401060)
0040102Fadd esp,4
00401032mov dword ptr [ebp-8],eax
00401035mov eax,dword ptr [ebp-8]
00401038mov dword ptr [ebp-4],eax这里,我们为了简单并没有释放内存,那么该怎么去释放呢?是 delete p 么?噢,错了,应该是 delete []p,这是为了告诉编译器:我删除的是一个数组,VC6 就会根据相应的 Cookie 信息去进行释放内存的工作。
好了,我们回到我们的主题:堆和栈究竟有什么区别?
主要的区别由以下几点:
1、管理方式不同;
2、空间大小不同;
3、能否产生碎片不同;
4、生长方向不同;
5、分配方式不同;
6、分配效率不同;管理方式:对于栈来讲,是由编译器自动管理,无需我们手工控制;对于堆来说,释放工作由程序员控制,容易产生memory leak。
空间大小:一般来讲在 32 位系统下,堆内存可以达到4G的空间,从这个角度来看堆内存几乎是没有什么限制的。但是对于栈来讲,一般都是有一定的空间大小的,例如,在VC6下面,默认的栈空间大小是1M(好像是,记不清楚了)。当然,我们可以修改:打开工程,依次操作菜单如下:Project->Setting->Link,在 Category 中选中 Output,然后在 Reserve 中设定堆栈的最大值和 commit。注意:reserve 最小值为 4Byte;commit 是保留在虚拟内存的页文件里面,它设置的较大会使栈开辟较大的值,可能增加内存的开销和启动时间。
碎片问题:对于堆来讲,频繁的 new/delete 势必会造成内存空间的不连续,从而造成大量的碎片,使程序效率降低。对于栈来讲,则不会存在这个问题,因为栈是先进后出的队列,他们是如此的一一对应,以至于永远都不可能有一个内存块从栈中间弹出,在他弹出之前,在他上面的后进的栈内容已经被弹出,详细的可以参考数据结构,这里我们就不再一一讨论了。
生长方向:对于堆来讲,生长方向是向上的,也就是向着内存地址增加的方向;对于栈来讲,它的生长方向是向下的,是向着内存地址减小的方向增长。
分配方式:堆都是动态分配的,没有静态分配的堆。栈有2种分配方式:静态分配和动态分配。静态分配是编译器完成的,比如局部变量的分配。动态分配由 malloc 函数进行分配,但是栈的动态分配和堆是不同的,他的动态分配是由编译器进行释放,无需我们手工实现。
分配效率:栈是机器系统提供的数据结构,计算机会在底层对栈提供支持:分配专门的寄存器存放栈的地址,压栈出栈都有专门的指令执行,这就决定了栈的效率比较高。堆则是 C/C++ 函数库提供的,它的机制是很复杂的,例如为了分配一块内存,库函数会按照一定的算法(具体的算法可以参考数据结构/操作系统)在堆内存中搜索可用的足够大小的空间,如果没有足够大小的空间(可能是由于内存碎片太多),就有可能调用系统功能去增加程序数据段的内存空间,这样就有机会分到足够大小的内存,然后进行返回。显然,堆的效率比栈要低得多。
从这里我们可以看到,堆和栈相比,由于大量 new/delete 的使用,容易造成大量的内存碎片;由于没有专门的系统支持,效率很低;由于可能引发用户态和核心态的切换,内存的申请,代价变得更加昂贵。所以栈在程序中是应用最广泛的,就算是函数的调用也利用栈去完成,函数调用过程中的参数,返回地址,EBP 和局部变量都采用栈的方式存放。所以,我们推荐大家尽量用栈,而不是用堆。
虽然栈有如此众多的好处,但是由于和堆相比不是那么灵活,有时候分配大量的内存空间,还是用堆好一些。
无论是堆还是栈,都要防止越界现象的发生(除非你是故意使其越界),因为越界的结果要么是程序崩溃,要么是摧毁程序的堆、栈结构,产生以想不到的结果,就算是在你的程序运行过程中,没有发生上面的问题,你还是要小心,说不定什么时候就崩掉,那时候 debug 可是相当困难的 :)
对了,还有一件事,如果有人把堆栈合起来说,那它的意思是栈,可不是堆,呵呵,清楚了?
static 用来控制变量的存储方式和可见性
函数内部定义的变量,在程序执行到它的定义处时,编译器为它在栈上分配空间,函数在栈上分配的空间在此函数执行结束时会释放掉,这样就产生了一个问题: 如果想将函数中此变量的值保存至下一次调用时,如何实现? 最容易想到的方法是定义一个全局的变量,但定义为一个全局变量有许多缺点,最明显的缺点是破坏了此变量的访问范围(使得在此函数中定义的变量,不仅仅受此 函数控制)。需要一个数据对象为整个类而非某个对象服务,同时又力求不破坏类的封装性,即要求此成员隐藏在类的内部,对外不可见。
static 的内部机制:
静态数据成员要在程序一开始运行时就必须存在。因为函数在程序运行中被调用,所以静态数据成员不能在任何函数内分配空间和初始化。这样,它的空间分配有三个可能的地方,一是作为类的外部接口的头文件,那里有类声明;二是类定义的内部实现,那里有类的成员函数定义;三是应用程序的 main()函数前的全局数据声明和定义处。
静态数据成员要实际地分配空间,故不能在类的声明中定义(只能声明数据成员)。类声明只声明一个类的“尺寸和规格”,并不进行实际的内存分配,所以在类声明中写成定义是错误的。它也不能在头文件中类声明的外部定义,因为那会造成在多个使用该类的源文件中,对其重复定义。
static 被引入以告知编译器,将变量存储在程序的静态存储区而非栈上空间,静态数据成员按定义出现的先后顺序依次初始化,注意静态成员嵌套时,要保证所嵌套的成员已经初始化了。消除时的顺序是初始化的反顺序。
static 的优势:
可以节省内存,因为它是所有对象所公有的,因此,对多个对象来说,静态数据成员只存储一处,供所有对象共用。静态数据成员的值对每个对象都是一样,但它的 值是可以更新的。只要对静态数据成员的值更新一次,保证所有对象存取更新后的相同的值,这样可以提高时间效率。引用静态数据成员时,采用如下格式:
<类名>::<静态成员名>
如果静态数据成员的访问权限允许的话(即 public 的成员),可在程序中,按上述格式来引用静态数据成员。
Ps:
(1) 类的静态成员函数是属于整个类而非类的对象,所以它没有this指针,这就导致了它仅能访问类的静态数据和静态成员函数。
(2) 不能将静态成员函数定义为虚函数。
(3) 由于静态成员声明于类中,操作于其外,所以对其取地址操作,就多少有些特殊,变量地址是指向其数据类型的指针,函数地址类型是一个“nonmember 函数指针”。
(4) 由于静态成员函数没有 this 指针,所以就差不多等同于 nonmember 函数,结果就产生了一个意想不到的好处:成为一个 callback 函数,使得我们得以将 c++ 和 c-based x window 系统结合,同时也成功的应用于线程函数身上。
(5) static 并没有增加程序的时空开销,相反她还缩短了子类对父类静态成员的访问时间,节省了子类的内存空间。
(6) 静态数据成员在<定义或说明>时前面加关键字 static。
(7) 静态数据成员是静态存储的,所以必须对它进行初始化。
(8) 静态成员初始化与一般数据成员初始化不同:
初始化在类体外进行,而前面不加 static,以免与一般静态变量或对象相混淆;
初始化时不加该成员的访问权限控制符 private、public;
初始化时使用作用域运算符来标明它所属类;
所以我们得出静态数据成员初始化的格式: <数据类型><类名>::<静态数据成员名>=<值>
(9) 为了防止父类的影响,可以在子类定义一个与父类相同的静态变量,以屏蔽父类的影响。这里有一点需要注意:我们说静态成员为父类和子类共享,但我们有重复定义了静态成员,这会不会引起错误呢?不会,我们的编译器采用了一种绝妙的手法:name-mangling 用以生成唯一的标志。附录内容转自http://www.cnblogs.com/daocaoren/archive/2011/06/29/2092957.html