C++大小字母的转换方式
C++大小字母转换
因为所有小写字母的ASCⅡ值要比对应的大写字母的ASCⅡ值大32,所以c1减去32后便得到原来字母的大写形式;反之,c2加上32后便得到原来字母的小写形式。
#include<iostream> using namespace std; //大小写字母转换通过ASCⅡ码值进行改变,小写-大写=32; int main() { char c1 = 'a'; char c2 = 'A'; cout << c1 << " " << c2 << endl; c1 = c1 - 32;//小写转换成大写 c2 = c2 + 32;//大写转换成小写 cout << c1 << " " << c2 << endl; }
运行结果:
a A
A a
C++常用的大小写转换方法
思路1:根据字母的ASCII表进行转换
由表格可以看出,对应大小写字母之间相差32,由此可以衍生出以下编程的思路:
程序1.1
#include <iostream> using namespace std; int main() { char a[20]; int i = 0; cout<<"请输入一串字符:\n"; cin>>a; for(;a[i];i++) { if(a[i] >= 'a'&&a[i] <= 'z') a[i] -= 32; else if(a[i] >= 'A'&&a[i] <= 'Z') a[i] += 32; } for(i = 0;a[i];i++) cout<<a[i]; cout<<endl; system("pause"); return 0; }
程序1.2
#include <iostream> using namespace std; void main(void) { char i; cout<<"Input,'#'for an end: "<<endl; while(1) { cin >> i; if ((i>=65)&&(i<=90)) { i=i+32; cout << i; } else if((i>=97)&&(i<=122)) { i=i-32; cout << i; } else cout << (int)i; if(i=='#') break; } }
思路2:利用大小写字母转换函数
由此可以衍生出以下几种编程的思路:
程序2.1 简易版
#include <iostream> using namespace std; int main() { cout<<(char)toupper(97)<<'\n'; cout<<(char)toupper('a')<<'\n'; cout<<(char)tolower(66)<<'\n'; cout<<(char)tolower('B')<<'\n'; return 0; }
程序2.2 利用函数strupr、strlwr
#include<iostream> #include<string> using namespace std; int main(int argc, char* argv[]) { //声明字符数组 char str[80],*p; int i; //转换字符串中的小写为大写 cout<<"将字符串中的小写字母转换为大写"<<endl; cout<<"请输入原字符串:"<<endl; cin>>str; p=strupr(str); cout<<"p:"<<p<<endl; cout<<"string:"<<str<<endl; cout<<"___________________"<<endl; //转换字符串中的大写为小写 cout<<"将字符串中的大写字母转换为小写"<<endl; cout<<"请输入原字符串:"<<endl; cin>>str; p=strlwr(str); cout<<"p:"<<p<<endl; cout<<"string:"<<str<<endl; cout<<"___________________"<<endl; system("pause"); return 0; }
程序2.3 利用函数toupper、tolower
#include<iostream> #include<cctype> #include<vector> using namespace std; int main() { vector<char> vch; int n; char elem; cout<<"请输入大小写字符的个数:"; cin>>n; cout<<"请输入"<<n<<"个大小写字符:"; for(int i = 0;i<n;++i) { cin>>elem; vch.push_back(elem); } vector<char>::iterator it = vch.begin(); for(it;it != vch.end();++it) { if(*it >= 'a'&&(*it) <='z') *it = toupper(*it); else if(*it >= 'A'&& (*it) <= 'Z') *it = tolower(*it); } cout<<"大小写转化之后的结果:"; vector<char>::iterator itera = vch.begin(); for(itera;itera != vch.end();++itera) cout<<*itera; cout<<endl; return 0; }
程序2.4 利用transform和tolower及toupper进行结合
#include<iostream> #include<algorithm> #include<string> #include<cctype> using namespace std; int main() { cout<<"请输入一个全部大写的字符串:"; string str; cin>>str; ///转小写 transform(str.begin(),str.end(),str.begin(),tolower); ///transform(wstr.begin(), wstr.end(), wstr.begin(), towlower); cout<<"转化为小写后为:"<<str<<endl; ///转大写 cout<<"请再输入一个全部小写的字符串:"; string s; cin>>s; transform(s.begin(), s.end(), s.begin(), toupper); ///transform(wstr.begin(), wstr.end(), wstr.begin(), towupper); cout<<"转化为大写后为:"<<s; wstring wstr =L"Abc"; transform(wstr.begin(), wstr.end(), wstr.begin(), towupper); cout<<wstr; return 0; }
程序2.5 注意wmain(),另一种编程方法
#include <iostream> #include <cstring> #include <windows.h> #include <cctype> #include <algorithm> using namespace std; int wmain(int argc, WCHAR* argv[]) { char ch = 'a'; ch = toupper(ch); cout<<ch<<endl; WCHAR wch = 'a'; wch = towupper(wch); cout<<char(wch)<<endl; WCHAR wideStr[] = L"Abc"; _wcslwr_s(wideStr, wcslen(wideStr) + 1); _wcsupr_s(wideStr, wcslen(wideStr) + 1); wstring wstr =L"Abc"; transform(wstr.begin(), wstr.end(), wstr.begin(), towupper); return 0; }
程序2.6 写成convert函数,利用|=和&=进行变换
#include <iostream> #include <cassert> using namespace std; char* convert(char *src) { char *p = src; assert(p != NULL); while(*p) { if ('A' <= *p && *p < 'Z') *p |= 0x20; else if ('a' <= *p && *p < 'z') *p &= ~0x20; p++; } return src; } int main() { char src; cin>>src; convert(&src); cout<<src; return 0; }
其中,在用到transform时,可能遇到如下错误提示:
error: no matching function for call to ‘transform(__gnu_cxx::__normal_iterator<char*, std::basic_string<char, std::char_traits<char>, std::allocator<char> > >, __gnu_cxx::__normal_iterator<char*, std::basic_string<char, std::char_traits<char>, std::allocator<char> > >, __gnu_cxx::__normal_iterator<char*, std::basic_string<char, std::char_traits<char>, std::allocator<char> > >, <unknown type>)’
这里说明了原因:
The problem is that the version of std::tolower inherited from the C standard library is a non-template function, but there are other versions of std::tolower that are function templates, and it is possible for them to be included depending on the standard library implementation. You actually want to use the non-template function, but there is ambiguity when just tolower is provided as the predicate.
翻译过来就是说,既有C版本的toupper/tolower函数,又有STL模板函数toupper/tolower,二者存在冲突。
解决办法:
在toupper/tolower前面加::,强制指定是C版本的(这时也不要include <cctype>了):
#include <iostream> #include <string> #include <algorithm> // transform using namespace std; int main() { string str = "abcdADcdeFDde!@234"; transform(str.begin(), str.end(), str.begin(), ::toupper); cout << str << endl; transform(str.begin(), str.end(), str.begin(), ::tolower); cout << str << endl; return 0; }
关于transform
transform() 可以将函数应用到序列的元素上,并将这个函数返回的值保存到另一个序列中,它返回的迭代器指向输出序列所保存的最后一个元素的下一个位置。
这个算法有一个版本和 for_each() 相似,可以将一个一元函数应用到元素序列上来改变它们的值,但这里有很大的区别。for_each() 中使用的函数的返回类型必须为 void,而且可以通过这个函数的引用参数来修改输入序列中的值;而 transform() 的二元函数必须返回一个值,并且也能够将应用函数后得到的结果保存到另一个序列中。
不仅如此,输出序列中的元素类型可以和输入序列中的元素类型不同。对于 for_each(),函数总是会被应用序列的元素上,但对于 transform(),这一点无法保证。
第二个版本的 transform() 允许将二元函数应用到两个序列相应的元素上,但先来看一下如何将一元函数应用到序列上。在这个算法的这个版本中,它的前两个参数是定义输入序列的输入迭代器,第 3 个参数是目的位置的第一个元素的输出迭代器,第 4 个参数是一个二元函数。这个函数必须接受来自输入序列的一个元素为参数,并且必须返回一个可以保存在输出序列中的值。
总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。
最新评论