[C++]string的使用
目录
string的使用::
1.string类介绍
2.string常用接口说明
string相关习题训练::
1.仅仅反转字母
2.找字符串中第一个只出现一次的字符
3.字符串里面最后一个单词的长度
4.验证一个字符串是否是回文
5.字符串相加
6.翻转字符串II:区间部分翻转
7.翻转字符串III:翻转字符串中的单词
8.字符串相乘
string及其模拟实现::
1.string类介绍
1.string类是basic_string模板类的一个实例,它使用char来实例化basic_string模板类,并用char_traits和allocator作为basic_string的默认参数。
2.string是表示字符串的字符类。
3.该类的接口与常规容器的接口基本相同,再添加了一些专门用来操作string的常规操作。
4.注意这个类独立于所使用的编码来处理字节:如果用来处理多字节或变长字符(如UTF-8)的序列,这个类的所有成员(如长度或大小)以及它的迭代器,将仍然按照字节来操作而不是实际编码的字符来操作。
5.string在底层实际是:basic_string模板类的别名,typedef basic_string<char,char_traits,allocator> string。
6.不能操作多字节或者变长字符的序列。
template<class T>
class basic_string
{
private:T* _str;size_t size;size_t capacity;
};
typedef basic_string<char> string;
扩展:
字符集:是一个系统支持的所有抽象字符的集合,也就是一系列字符的集合。字符是各种文字和符号的总称,包括各国家文字、标点符号、图形符号、数字等,常见的字符集有:ASCII字符集、GB2312字符集(主要用于处理中文汉字)、GBK字符集(主要用于处理中文汉字)、Unicode字符集等。
字符编码:是一套法则,使用该法则能够对自然语言的字符的一个字符集(如字母表或音节表)与计算机能识别的二进制数字进行配对。即它能在符号集合与数字系统之间建立对应关系,是信息处理的一项基本技术。通常人们用符号集合(一般情况下就是文字)来表达信息,而计算机的信息处理系统则是以二进制的数字来存储和处理信息的。字符编码就是将符号转换为计算机能识别的二进制编码。
一般一个字符集等同于一个编码方式,ANSI体系(ANSI是一种字符代码,为使计算机支持更多语言,通常使用0x80~0xFF范围的2个字节来表示一个字符)的字符集如ASCII、ISO 8859-1、GB2312、GBK等等都是如此。一般我们说一种编码都是针对某一特定的字符集。
一个字符集也可以有多种编码方式,例如UCS字符集(也是Unicode使用的字符集)上有UTF-8、UTF-16、UTF-32等编码方式。
从计算机字符编码的发展历史角度来看,大概经历了三个阶段:
第一个阶段:ASCII字符集和ASCII编码
计算机刚开始只支持英语,其他语言不能够在计算机上存储和显示。ASCII用一个字节的7个比特位表示一个字符,第一个位置0.后来为了更多的欧洲常用字符又对ASCII进行了扩展,又有了EASCII,EASCII用8位表示一个字符,使它能多表示128个字符,支持了部分西欧字符。
第二个阶段:ANSI编码(本地化)
为使计算机支持更多语言,通常使用0x80~0xFF范围的两个字节来表示一个字符。比如汉字 '中' 在中文操作系统中使用 [0xD6,0xD0] 这两个字节存储。不同的国家和地区制定了不同的标准,由此产生了GB2312,BIG5,JIS等各自的编码标准。这些使用2个字节来代表一个字符的各种汉字延申编码方式称为ANSI编码,在简体中文系统下,ANSI编码代表GB2312编码,在日文操作系统下,ANSI编码代表JIS编码。不同ANSI编码之间互不兼容,当信息在国际化间交流时,无法将属于两种语言的文字,存出在同一段ANSi编码的文本中。
第三个阶段:Unicode(国际化)
为了使国际间信息交流更加方便,国际组织制定了Unicode字符集,为各种语言中的每一个字符设定了统一并且唯一的数字编号,以满足跨语言、跨平台进行文本交换、处理的要求。Unicode常见的有三种编码方式:UTF-8(1、2、3、4个字节表示)、UTF-16(2个字节表示)、UTF-32(4个字节表示)。
为什么string类要实现成模板?
我们印象中,单个字符就是一个字节,也就是char类型,但是由于编码问题,会有2字节和4字节的字符类型。
类型 | 编码 | 类型 |
string | UTF-8 | char |
wstring | Unicode | wchar_t |
u16string | UTF-16 | char16_t |
u32string | UTF-32 | char32_t |
正是由于了字符串的多种类型,所以存在了basic_strng模板。
2.string常用接口说明
在使用string类时,必须包含#include头文件以及using namespace std;
1.string类对象的常见构造
函数名称 | 功能说明 |
string() | 构造空的string类对象,即空字符串 |
string(const char* s) | 用C-string来构造string类对象 |
string(size_t n,char c) | string类对象中包含n个字符c |
string(const string& s) | 拷贝构造函数 |
void Teststring()
{string s1; //构造空的string类对象s1string s2("hello world"); //用C格式字符串构造string类对象s2string s3(3,"'x');//用3个字符创建对象 string s4(s2);//拷贝构造
}
2.string类对象的容量操作:
函数名称 | 功能说明 |
size | 返回字符串有效字符长度 |
length | 返回字符串有效字符长度 |
capacity | 返回总空间大小 |
empty | 检测字符串是否为空串,是返回true,否则返回false |
clear | 清空有效字符 |
reserve | 为字符串预留空间 |
resize | 将有效字符的个数改成n个,外出的多余空间用字符c填充 |
// 测试string容量相关的接口
// size/clear/resize
void Teststring1()
{// 注意:string类对象支持直接用cin和cout进行输入和输出string s("hello, world!!!");cout << s.size() << endl;cout << s.length() << endl;cout << s.capacity() << endl;cout << s << endl;// 将s中的字符串清空,注意清空时只是将size清0,不改变底层空间的大小s.clear();cout << s.size() << endl;cout << s.capacity() << endl;// 将s中有效字符个数增加到10个,多出位置用'a'进行填充// “aaaaaaaaaa”s.resize(10, 'a');cout << s.size() << endl;cout << s.capacity() << endl;// 将s中有效字符个数增加到15个,多出位置用缺省值'\0'进行填充// "aaaaaaaaaa\0\0\0\0\0"// 注意此时s中有效字符个数已经增加到15个s.resize(15);cout << s.size() << endl;cout << s.capacity() << endl;cout << s << endl;// 将s中有效字符个数缩小到5个s.resize(5);cout << s.size() << endl;cout << s.capacity() << endl;cout << s << endl;
}//====================================================================================
void Teststring2()
{string s;// 测试reserve是否会改变string中有效元素个数s.reserve(100);cout << s.size() << endl;cout << s.capacity() << endl;// 测试reserve参数小于string的底层空间大小时,是否会将空间缩小s.reserve(50);cout << s.size() << endl;cout << s.capacity() << endl;
}// 利用reserve提高插入数据的效率,避免增容带来的开销
//====================================================================================
void TestPushBack()
{string s;size_t sz = s.capacity();cout << "making s grow:\n";for (int i = 0; i < 100; ++i){s.push_back('c');if (sz != s.capacity()){sz = s.capacity();cout << "capacity changed: " << sz << '\n';}}
}// 构建vector时,如果提前已经知道string中大概要放多少个元素,可以提前将string中空间设置好
void TestPushBackReserve()
{string s;s.reserve(100);size_t sz = s.capacity();cout << "making s grow:\n";for (int i = 0; i < 100; ++i){s.push_back('c');if (sz != s.capacity()){sz = s.capacity();cout << "capacity changed: " << sz << '\n';}}
}
注意:
1.size()与length()方法底层实现原理完全相同,引入size()的原因是为了和其他容器的接口保持一致,一般情况下基本都是用size()。
2.clear()只是将string中有效字符清空,不改变底层空间大小。
3.resize(size_t n)与resize(size_t n,char c)都是将字符串中有效字符个数改变到n个,不同的是当字符个数增多时:resize(n)用0来填充多出的元素空间,resize(size_t n,char c)用字符c来填充多出的元素空间。注意:resize在改变元素个数时,如果是将元素个数增多,可能会改变底层容量的大小,如果是将元素个数减少,底层空间总大小不变。
4.resize是改变它的size,reserve是改变它的capacity。但要注意resize既会改变size也可能影响capacity。
5.reserve(size_t res_arg = 0):为string预留空间,不改变有效元素个数,当reserve的参数小于string的底层空间总大小时,reserve不会改变容量大小。
3.string类对象的访问及遍历操作:
函数名称 | 功能说明 |
operator[] | 返回pos位置的字符 |
begin + end | begin()获取第一个字符的迭代器 + end()获取最后一个字符下一个位置的迭代器 |
rbegin + rend | rbegin():获取倒数第一个有效字符的迭代器 + rend()获取第一个字符的迭代器 |
范围for | C++11支持更简洁的范围for的新遍历方式 |
// string的遍历
// begin()+end() for+[] 范围for
// 注意:string遍历时使用最多的还是for+下标 或者 范围for(C++11后才支持)
// begin()+end()大多数使用在需要使用STL提供的算法操作string时,比如:采用reverse逆置string
void Teststring3()
{string s1("hello world");const string s2("Hello world");cout << s1 << " " << s2 << endl;cout << s1[0] << " " << s2[0] << endl;s1[0] = 'H';cout << s1 << endl;// s2[0] = 'h'; 代码编译失败,因为const类型对象不能修改
}void Teststring4()
{string s("hello Bit");// 3种遍历方式:// 需要注意的以下三种方式除了遍历string对象,还可以遍历是修改string中的字符,// 另外以下三种方式对于string而言,第一种使用最多// 1. for+operator[]for (size_t i = 0; i < s.size(); ++i)cout << s[i] << endl;// 2.迭代器string::iterator it = s.begin();while (it != s.end()){cout << *it << endl;++it;}// string::reverse_iterator rit = s.rbegin();// C++11之后,直接使用auto定义迭代器,让编译器推到迭代器的类型auto rit = s.rbegin();while (rit != s.rend()){cout << *rit << endl;}// 3.范围forfor (auto ch : s){cout << ch << endl;}
}
//const正向迭代器(不能改变*it)
void test(const string& s)
{string::const_iterator it = s.begin();while (it != s.end()){cout << *it;++it;}
}
//const反向迭代器(不能改变*it)
void test(const string& s)
{string::const_reverse_iterator rit = s.rbegin();while (rit != s.rend()){cout << *rit;++rit;}
}
void Print(const string& s)
{//普通的正向/反向迭代器支持读写容器数据 不是it不可以修改 是it指向的内容不可以修改//const的正向/反向只支持读不支持修改容器数据string::const_iterator it = s.begin();while (it != s.end()){cout << *it << " ";++it;}cout << endl;string const_reverse_iterator rit = s.rbegin();//auto rit = s.rbegin();while (rit != s.rend()){cout << *rit << " ";++rit;}cout << endl;
}
4.string类对象的修改操作:
函数名称 | 功能说明 |
push_back | 在字符串后尾插字符c |
append | 在字符串后追加一个字符串 |
operator+= | 在字符串后追加字符串str |
c_str | 返回C格式字符串 |
find + npos | 从字符串pos位置开始往后找字符c,返回该字符在字符串中的位置 |
rfind | 从字符串pos位置开始往前找字符c,返回该字符在字符串中的位置 |
substr | 在str中从pos位置开始,截取n个字符,然后将其返回 |
注意:
1.在string尾部追加字符时,s.push_back(c) / s.append(1,c) / s += 'c'三种的实现方式差不多,一般情况下string类的+=操作用的比较多,+=操作不仅可以连接单个字符,还可以连接字符串。
2.对string操作时,如果能够大概预估到放多少字符,可以先通过reserve把空间预留好。
void test_string1()
{string s1("hello world hello world");string s2("hello world hello world");string s3(s2);string s4(s3);//assign可以理解为将原字符对象清空,重新进行赋值s1.assign("hello bit", 5);cout << s1 << endl;//replace是对字符对象的部分取代s2.replace(6, 5, "bit");cout << s2 << endl;//将' '替换成20%size_t pos = s3.find(' ');while (pos != strinng::npos){s3.replace(pos, 1, "20%");pos = s3.find(' ', pos + 3);}cout << s3 << endl;string ret;//s4的长度+空格个数*2//ret.reserve(s4.size() + space_size*2);for (auto ch: s4){if (ch != ' '){ret += ch;}else{ret += "%20";}}cout << ret << endl;
}
//用C语言形式读取文件
void test_string2()
{string file("test.cpp");//c_str返回C语言形式的字符串char*FILE* fout = fopen(file.c_str(), "r");assert(fout);char ch = fgetc(fout);while (ch != EOF){cout << ch;ch = fgetc(fout);}fclose(fout);
}
//要求取出一个文件的后缀
void test_string3()
{//"Test.cpp"string file;cin >> file;//要求取后缀//size_t pos = file.find('.');//rfind倒着找 适用于Test.zip.tarsize_t pos = file.rfind('.');if (pos != string::npos){//substr:取出一个对象的一部分构建成string对象进行返回//string suffix = file.substr(pos, file.size() - pos);//substr的函数形式:string substr(size_t pos = 0,size_t len = npos) 默认取到结尾string suffix = file.substr(pos);cout < suffix << endl;}
}
// 测试string:
// 1. 插入(拼接)方式:push_back append operator+=
// 2. 正向和反向查找:find() + rfind()
// 3. 截取子串:substr()
// 4. 删除:erase
void Teststring5()
{string str;str.push_back(' '); // 在str后插入空格str.append("hello"); // 在str后追加一个字符"hello"str += 'b'; // 在str后追加一个字符'b' str += "it"; // 在str后追加一个字符串"it"cout << str << endl;cout << str.c_str() << endl; // 以C语言的方式打印字符串// 获取file的后缀//一闭一开才是距离string file("string.cpp");size_t pos = file.rfind('.');string suffix(file.substr(pos, file.size() - pos));cout << suffix << endl;// npos是string里面的一个静态成员变量// static const size_t npos = -1;// 取出url中的域名string url("http://www.cplusplus.com/reference/string/string/find/");cout << url << endl;size_t start = url.find("://");if (start == string::npos){cout << "invalid url" << endl;return;}start += 3;size_t finish = url.find('/', start);string address = url.substr(start, finish - start);cout << address << endl;// 删除url的协议前缀pos = url.find("://");url.erase(0, pos + 3);cout << url << endl;
}
5.string类非成员函数:
函数名称 | 功能说明 |
operator+ | 尽量少用,因为传值返回,导致深拷贝效率低 |
operator>> | 输入运算符重载 |
operator<< | 输出运算符重载 |
getline | 获取一行字符串 |
relational operators | 大小比较 |
注:流提取是不能接收到空格和换行的,需要接收一行的时候需要使用getline
6.vs和g++下string结构的说明
注意:下述结构是在32位平台下进行验证,32位平台下指针占4个字节。
vs下string的结构:
string总共占28个字节,内部结构稍微复杂一点,先是有一个联合体,联合体用来定义string中字符串的存储空间:
1.当字符串长度小于16时,使用内部固定的字符数组来存放
2.当字符串长度大于等于16时,从堆上开辟空间
union _Bxty
{ // storage for small buffer or pointer to larger onevalue_type _Buf[_BUF_SIZE];pointer _Ptr;char _Alias[_BUF_SIZE]; // to permit aliasing
} _Bx;
这种设计也是有一定道理的,大多数情况下字符串的长度都小于16,那string对象创建好之后,内部已经有了16个字符数组的固定空间,不需要通过堆创建,效率高
其次:还有一个size_t字段保存字符串长度,一个size_t字段保存从堆上开辟空间总的容量
最后:还有一个指针做一些其他事情。
故总共占16+4+4+4 = 28个字节
g++ 下string的结构:
g++下,string是通过写时拷贝实现的,string对象总共占4个字节,内部只包含了一个指针,该指针将来指向一块堆空间,内部包含了如下字段:
1.空间总大小
2.字符串有效长度
3.引用计数
4.指向堆空间的指针,用来存储字符串
struct _Rep_base
{size_type _M_length;size_type _M_capacity;_Atomic_word _M_refcount;
};
写时拷贝是在浅拷贝的基础之上增加了引用计数的方式实现的。
引用计数:用来记录资源使用者的个数。在构造时,将资源的计数给成1,每增加一个对象使用该资源,就给计数增加1,当某个对象销毁时,先给该计数减一,然后再检查是否需要释放资源,如果计数为1,说明该对象是资源的最后一个使用者,将该资源释放,否则就不能释放,因为还有其他对象在使用该资源。
像begin(), end(), rbegin(), rend(), operator[]的非const版本和const版本也都实现了
但像size这种类型的函数就只实现了const版本 push_back只实现了非const版本
那么哪些函数只实现const/非const版本 哪些函数两者都实现了呢
1.只读功能的函数实现const版本即可
2.只写功能的函数提供非const版本
3.读写功能的函数实现const+非const版本
string相关习题训练::
1.仅仅反转字母
代码1:
class Solution
{
public:string reverseOnlyLetters(string s){size_t begin = 0, end = s.size() - 1;while (begin < end){while (begin < end && !isalpha(s[begin])){++begin;}while (begin < end && !isalpha(s[end])){--end;}swap(s[begin], s[end]);++begin;--end;}return s;}
};
代码2:
class Solution
{
public:bool isLetter(char ch){if (ch >= 'a' && ch <= 'z')return true;if (ch >= 'A' && ch <= 'Z')return true;return false;}string reverseOnlyLetters(string S) {if (S.empty())return S;size_t begin = 0, end = S.size() - 1;while (begin < end){while (begin < end && !isLetter(S[begin]))++begin;while (begin < end && !isLetter(S[end]))--end;swap(S[begin], S[end]);++begin;--end;}return S;}
};
2.找字符串中第一个只出现一次的字符
class Solution
{
public:int firstUniqChar(string s){int count[26] = { 0 };//统计次数for (auto ch : s){count[ch - 'a']++;}//找第一个只出现一次的字符for (size_t i = 0; i < s.size(); ++i){if (count[s[i]] - 'a' == 1){return i;}}return -1;}
};
3.字符串里面最后一个单词的长度
//报错原因:cin遇到空格 一次提取结束 空格之后的字符串内容仍然在缓冲区中
//int main()
//{
// string str;
// cin >> str;
// size_t pos = str.rfind(' ');
// //下标相减 左闭右开就是个数
// cout << str.size() - pos - 1 << endl;
// return 0;
//}
//getline函数:
//istream& getline(istream& is,string& str,char delim) 可以指定字符提取结束
//istream& getline(istream& is, string& str) 默认换行提取结束
代码1
int main()
{string str;getline(cin, str);size_t pos = str.rfind(' ');//下标相减 左闭右开就是个数cout << str.size() - pos - 1 << endl;return 0;
}
代码2
int main()
{string line;// 不要使用cin>>line,因为会它遇到空格就结束了// while(cin>>line)while(getline(cin, line)){size_t pos = line.rfind(' ');cout<<line.size()-pos-1<<endl;}return 0;
}
4.验证一个字符串是否是回文
class Solution
{
public:bool isLetterOrNumber(char ch){return (ch >= '0' && ch <= '9')|| (ch >= 'a' && ch <= 'z')|| (ch >= 'A' && ch <= 'Z');}bool isPalindrome(string s){//先小写字母转换成大写 再进行判断for (auto : ch:s){if (ch >= 'a' && ch <= 'z'){ch -= 32;}}int begin = 0, end = s.size() - 1;while (begin < end){while (begin < end && !isLetterOrNumber(s[begin])){++begin;}while (begin < end && !isLetterOrNumber(s[end])){--end;}if (s[begin] != s[end]){return false;}else{++begin;--end;}}return true;}
};
5.字符串相加
方法一:
class Solution
{
public:string addStrings(string num1, string num2){int end1 = num1.size() - 1, end2 = num2.size() - 1;//最开始进位赋值为0int carry = 0;string retStr;while (end1 >= 0 || end2 >= 0){int val1 = end1 >= 0 ? num1[end1] - '0' : 0;int val2 = end2 >= 0 ? nums[end2] - '0' : 0;int ret = val1 + val2 + carry;carry = ret / 10;ret %= 10;/*if (ret > 9){ret -= 10;carry = 1;}else{carry = 0;}*/retStr.insert(0, 1, ret + '0');--end1;--end2;}if (carry == 1){retStr.insert(0, 1, '1');}return retStr;}
};
方法二:
class Solution
{
public:string addStrings(string num1, string num2){int end1 = num1.size() - 1, end2 = num2.size() - 1;//最开始进位赋值为0int carry = 0;string retStr;retStr.reserve(max(num1.size(), num2.size()) + 1);while (end1 >= 0 || end2 >= 0){int val1 = end1 >= 0 ? num1[end1] - '0' : 0;int val2 = end2 >= 0 ? nums[end2] - '0' : 0;int ret = val1 + val2 + carry;carry = ret / 10;ret %= 10;retStr += ('0' + ret);--end1;--end2;}if (carry == 1){retStr += '1';}reverse(retStr.begin(), retStr.end());return retStr;}
};
6.翻转字符串
//思路:
//收尾交换,进行翻转
class Solution
{
public:void reverseString(vector<char>& s){if (s.empty()){return;}int start = 0;int end = s.size() - 1;while (start < end){swap(s[start], s[end]);start++;end--;}}
};
7.翻转字符串II:区间部分翻转
class Solution
{
public://翻转start到end区间的字符串void Reverse(string& s, int start, int end){char tmp;end--;while (start < end){tmp = s[start];s[start] = s[end];s[end] = tmp;start++;end--;}}string reverseStr(string s, int k){int len = s.size();for (int i = 0; i < len; i += 2 * k){if (i + k < len){Reverse(s, i, i + k);}else{Reverse(s, i, len);}}return s;}
};
8.翻转字符串III:翻转字符串中的单词
//思路:
//1. 通过查找空格,分割单词
//2. 针对分割的单词进行翻转
class Solution
{
public:void Reverse(string& s, int start, int end){char tmp;while (start < end){tmp = s[start];s[start] = s[end];s[end] = tmp;start++;end--;}}string reverseWords(string s){size_t start = 0;size_t end = 0;while (start < s.size()){end = s.find(' ', start);if (end == string::npos){end = s.size();break;}Reverse(s, start, end - 1);start = end + 1;}Reverse(s, start, end - 1);return s;}
};
9.字符串相乘
//思路:
//1. 下翻转数据
//2. 按位相乘
//3. 将乘得的结果进行错位相加,模拟乘法的笔算过程
class Solution
{
public:void MulItem(string& tmp, string& num1, char a){int i = 0, sign = 0;int mul = 0;while (i < num1.size()){mul = (num1[i] - '0') * (a - '0') + sign;if (mul >= 10){sign = mul / 10;mul %= 10;}else{sign = 0;}tmp.push_back(mul + '0');i++;}if (sign > 0){tmp.push_back(sign + '0');}}//对应为相加,sign进位采用引用传递int AddItem(int a, int b, int& sign){int add = a + b + sign;if (add >= 10){sign = 1;add -= 10;}else{sign = 0;}return add;}//错位相加void MoveAdd(string& result, string& tmp, int k){int i, j;i = k;j = 0;int sign = 0;while (i < result.size() && j < tmp.size()){result[i] = AddItem(result[i] - '0', tmp[j] - '0', sign) + '0';i++;j++;}while (i < result.size() && sign){result[i] = AddItem(result[i] - '0', 0, sign) + '0';i++;}while (j < tmp.size()){int v = AddItem(0, tmp[j] - '0', sign);result.push_back(v + '0');j++;}if (sign){result.push_back(sign + '0');}}string multiply(string num1, string num2){//先翻转数据,方便进位处理reverse(num1.begin(), num1.end());reverse(num2.begin(), num2.end());string tmp, result;for (int i = 0; i < num2.size(); ++i){//使用num2的每一个数据乘以num1MulItem(tmp, num1, num2[i]);//将乘得的结果进行错位相加MoveAdd(result, tmp, i);tmp.clear();}while (result.size() != 1 && result.back() == '0'){result.pop_back();}//翻转数据,恢复数据reverse(result.begin(), result.end());return result;}
};
10.字符串转换成整数
//思路:
//1. 要考虑正负数
//2. 要考虑数据是否溢出
class Solution
{
public:int StrToInt(string str){int len = str.size();int flag = 1;if (len == 0){return 0;}const char* cstr = str.c_str();if (cstr == NULL){return 0;}int i = 0;if (cstr[i] == '+'){i++;flag = 1;//如果str[i]为'+',str[i]顺序后移,并令标志flag为1,表示为正数 }else if (cstr[i] == '-'){i++;flag = -1;//如果str[i]为'-',str[i]顺序后移,并令标志flag为-1,表示为负数 }long long num = 0;while (cstr[i] != '\0'){if (cstr[i] >= '0' && cstr[i] <= '9'){//每遍历一个在0-9间的字符,就将其输入到num中 num = num * 10 + (cstr[i] - '0');//下一次输入到num中时要加上上一次*10的结果,即上一次的数左移一位(十进制下) //如果数据溢出,则返回0if ((flag > 0 && num > 0x7fffffff) || (flag < 0 && num>0x80000000)){return 0;}i++;}else{return 0;}}if (flag < 0){num = num * -1;}return (int)num;}
};
相关文章:

[C++]string的使用
目录 string的使用:: 1.string类介绍 2.string常用接口说明 string相关习题训练:: 1.仅仅反转字母 2.找字符串中第一个只出现一次的字符 3.字符串里面最后一个单词的长度 4.验证一个字符串是否是回文 5.字符串相加 6.翻转字符串…...

Kali Linux 操作系统安装详细步骤——基于 VMware 虚拟机
1. Kali 操作系统简介 Kali Linux 是一个基于 Debian 的 Linux 发行版,旨在进行高级渗透测试和安全审计。Kali Linux 包含数百种工具,适用于各种信息安全任务,如渗透测试,安全研究,计算机取证和逆向工程。Kali Linux 由…...
R语言APSIM模型应用及批量模拟实践技术
查看原文>>>基于R语言APSIM模型高级应用及批量模拟实践技术 目录 专题一、APSIM模型应用与R语言数据清洗 专题二、APSIM气象文件准备与R语言融合应用 专题三、APSIM模型的物候发育和光合生产模块 专题四、APSIM物质分配与产量模拟 专题五、APSIM土壤水平衡模块 …...

破解马赛克有多「容易」?
刷短视频时,估计大家都看过下面这类视频,各家营销号争相曝光「一分钟解码苹果笔刷背后内容」的秘密。换汤不换药,自媒体们戏称其为「破解马赛克」,殊不知让多少不明真相的用户建立起了错误的认知,也让苹果笔刷第 10086…...

【.NET基础加强第八课--委托】
.NET基础加强第八课--委托 委托(Delegate)委托操作顺序实例多播委托—委托链实例实例委托传值 委托(Delegate) 委托(Delegate) 是存有对某个方法的引用的一种引用类型变量 委托操作顺序 1,定义一个委托类…...

jetcache:阿里这款多级缓存框架一定要掌握
0. 引言 之前我们讲解了本地缓存ehcache组件,在实际应用中,并不是单一的使用本地缓存或者redis,更多是组合使用来满足不同的业务场景,于是如何优雅的组合本地缓存和远程缓存就成了我们要研究的问题,而这一点ÿ…...

干货 | 如何做一个简单的访谈研究?
Hello,大家好! 这里是壹脑云科研圈,我是喵君姐姐~ 心理学中研究中,大家常用的研究方法大多是实验法、问卷调查法等,这些均是定量研究。 其实,作为质性研究中常用的访谈法,可对个体的内心想法进…...

4年外包出来,5次面试全挂....
我的情况 大概介绍一下个人情况,男,毕业于普通二本院校非计算机专业,18年跨专业入行测试,第一份工作在湖南某软件公司,做了接近4年的外包测试工程师,今年年初,感觉自己不能够再这样下去了&…...

基于遗传算法优化的核极限学习机(KELM)分类算法-附代码
基于遗传算法优化的核极限学习机(KELM)分类算法 文章目录 基于遗传算法优化的核极限学习机(KELM)分类算法1.KELM理论基础2.分类问题3.基于遗传算法优化的KELM4.测试结果5.Matlab代码 摘要:本文利用遗传算法对核极限学习机(KELM)进行优化,并用于分类 1.KE…...

评判需求优先级5大规则和方法(纯干货):
在划分用户需求时,需秉承需求任务紧跟核心业务指标,按照一定的规则和方法进行优先级的划分。 常见评判需求优先级规则有:四象限法则、KANO模型、二八原则、产品生命周期法、ROI评估法。 一、四象限法则 四象限法则是以【重要】和【紧急】程度…...
c++ 11标准模板(STL) std::vector (七)
定义于头文件 <vector> template< class T, class Allocator std::allocator<T> > class vector;(1)namespace pmr { template <class T> using vector std::vector<T, std::pmr::polymorphic_allocator<T>>; }(2)(C17…...

Contest3137 - 2022-2023-2 ACM集训队每月程序设计竞赛(1)五月月赛
A 1! 5! 46 169 有一种数字,我们称它为 纯真数。 它等于自身每一个数位的阶乘之和。请你求出不超过n的所有 纯真数。(注:纯真数不含有前导0)数据范围1e18 纯真数只有四个,注意0!1 1,2,145,40585 int n;cin>>n;int res[]{…...

如何使用 YOLOv8 神经网络检测图像中的物体
对象检测是一项计算机视觉任务,涉及识别和定位图像或视频中的对象。它是许多应用的重要组成部分,例如自动驾驶汽车、机器人和视频监控。 多年来,已经开发了许多方法和算法来查找图像中的对象及其位置。执行这些任务的最佳质量来自使用卷积神经网络。 YOLO 是这项任务最流行的…...

Python每日一练:小艺读书醉酒的狱卒非降序数组(详解快排)
文章目录 前言一、小艺读书二、醉酒的狱卒三、非降序数组总结 前言 今天这个非降序数组,阅读解理小学水平,说起来都是泪啊。我折腾了一天都没搞定,从冒泡写到快速排序。换了几种都还不行,我又给快排加上插入排序。结果还是不能全…...

手麻系统源码,PHP手术麻醉临床信息系统源码,手术前管理模块功能
手麻系统源码,PHP手术麻醉临床信息系统源码,手术前管理模块功能 术前管理模块主要有手术排班、手术申请单、手术通知单、手术知情同意书、输血血液同意书、术前查房记录、术前访视、风险评估、手术计划等功能。 功能: 手术排班:…...
AUTOSAR - ComM - 学习一 :基础知识+配置
目录 1、概述 1.1、总览 1.2、功能描述 1.3、依赖关系 2、功能SPEC 2.1、PNC...
手把手教你搭建ROS阿克曼转向小车之(增量式PID代码实现)
在上一篇文章中我们已经成功的把编码器的反馈值给计算出来,这篇文章将会讲解怎么使用反馈回来的速度值进行PID计算,从而闭环控制电机的速度。 PID算法介绍 1.开环控制系统 开环控制系统(open-loop control system)是指被控对象的输出(被控制量)对控制器…...

C语言函数大全-- t 开头的函数
C语言函数大全 本篇介绍C语言函数大全-- t 开头的函数 1. tan,tanf,tanl 1.1 函数说明 函数声明函数功能double tan(double x)计算 以弧度 x 为单位的角度的正切值(double)float tanf(float x)计算 以弧度 x 为单位的角度的正…...
安卓系统APP稳定性测试分析的研究报告
目录 第一章:概念 第二章:重要性 第三章:意义和作用 第四章:行业现状 第五章:常见测试方法和工具 第六章:实际测试场景 第七章:测试方案 第八章:测试方法 第九章࿱…...

【Java基础】集合
一、集合概述 为了方便对多个对象进行存储和操作,集合是一种Java容器,可以动态地把多个对象引用放入容器中 数组存储的特点 一旦初始化后,长度不可改变,元素类型不可改变提供的方法很少,对于添加、删除、获取实际元…...

Flask RESTful 示例
目录 1. 环境准备2. 安装依赖3. 修改main.py4. 运行应用5. API使用示例获取所有任务获取单个任务创建新任务更新任务删除任务 中文乱码问题: 下面创建一个简单的Flask RESTful API示例。首先,我们需要创建环境,安装必要的依赖,然后…...
论文解读:交大港大上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化学习框架(二)
HoST框架核心实现方法详解 - 论文深度解读(第二部分) 《Learning Humanoid Standing-up Control across Diverse Postures》 系列文章: 论文深度解读 + 算法与代码分析(二) 作者机构: 上海AI Lab, 上海交通大学, 香港大学, 浙江大学, 香港中文大学 论文主题: 人形机器人…...

工业安全零事故的智能守护者:一体化AI智能安防平台
前言: 通过AI视觉技术,为船厂提供全面的安全监控解决方案,涵盖交通违规检测、起重机轨道安全、非法入侵检测、盗窃防范、安全规范执行监控等多个方面,能够实现对应负责人反馈机制,并最终实现数据的统计报表。提升船厂…...

uniapp微信小程序视频实时流+pc端预览方案
方案类型技术实现是否免费优点缺点适用场景延迟范围开发复杂度WebSocket图片帧定时拍照Base64传输✅ 完全免费无需服务器 纯前端实现高延迟高流量 帧率极低个人demo测试 超低频监控500ms-2s⭐⭐RTMP推流TRTC/即构SDK推流❌ 付费方案 (部分有免费额度&#x…...

EtherNet/IP转DeviceNet协议网关详解
一,设备主要功能 疆鸿智能JH-DVN-EIP本产品是自主研发的一款EtherNet/IP从站功能的通讯网关。该产品主要功能是连接DeviceNet总线和EtherNet/IP网络,本网关连接到EtherNet/IP总线中做为从站使用,连接到DeviceNet总线中做为从站使用。 在自动…...

【论文阅读28】-CNN-BiLSTM-Attention-(2024)
本文把滑坡位移序列拆开、筛优质因子,再用 CNN-BiLSTM-Attention 来动态预测每个子序列,最后重构出总位移,预测效果超越传统模型。 文章目录 1 引言2 方法2.1 位移时间序列加性模型2.2 变分模态分解 (VMD) 具体步骤2.3.1 样本熵(S…...

Mysql中select查询语句的执行过程
目录 1、介绍 1.1、组件介绍 1.2、Sql执行顺序 2、执行流程 2.1. 连接与认证 2.2. 查询缓存 2.3. 语法解析(Parser) 2.4、执行sql 1. 预处理(Preprocessor) 2. 查询优化器(Optimizer) 3. 执行器…...

【电力电子】基于STM32F103C8T6单片机双极性SPWM逆变(硬件篇)
本项目是基于 STM32F103C8T6 微控制器的 SPWM(正弦脉宽调制)电源模块,能够生成可调频率和幅值的正弦波交流电源输出。该项目适用于逆变器、UPS电源、变频器等应用场景。 供电电源 输入电压采集 上图为本设计的电源电路,图中 D1 为二极管, 其目的是防止正负极电源反接, …...

【C++进阶篇】智能指针
C内存管理终极指南:智能指针从入门到源码剖析 一. 智能指针1.1 auto_ptr1.2 unique_ptr1.3 shared_ptr1.4 make_shared 二. 原理三. shared_ptr循环引用问题三. 线程安全问题四. 内存泄漏4.1 什么是内存泄漏4.2 危害4.3 避免内存泄漏 五. 最后 一. 智能指针 智能指…...

免费数学几何作图web平台
光锐软件免费数学工具,maths,数学制图,数学作图,几何作图,几何,AR开发,AR教育,增强现实,软件公司,XR,MR,VR,虚拟仿真,虚拟现实,混合现实,教育科技产品,职业模拟培训,高保真VR场景,结构互动课件,元宇宙http://xaglare.c…...