🌠 作者:@阿亮joy.
🎆专栏:《阿亮爱刷题》
🎇 座右铭:每个优秀的人都有一段沉默的时光,那段时光是付出了很多努力却得不到结果的日子,我们把它叫做扎根
👉仅反转字母👈
给你一个字符串 s ,根据下述规则反转字符串:
- 所有非英文字母保留在原有位置。
- 所有英文字母(小写或大写)位置反转。
返回反转后的 s 。
思路:这道题可以借助快排的思想,左边找出字母,右边找出字母,然后进行交换。如果不是字母,就继续找。
class Solution
{
public:
string reverseOnlyLetters(string s)
{
size_t begin = 0;
size_t end = s.size() - 1;
while(begin < end)
{
// isalpha是判断是否是字母的函数接口
while(begin < end && !isalpha(s[begin]))
{
++begin;
}
while(begin < end && !isalpha(s[end]))
{
--end;
}
swap(s[begin], s[end]);
++begin;
--end;
}
return s;
}
};
👉字符串中的第一个唯一字符👈
给定一个字符串 s ,找到 它的第一个不重复的字符,并返回它的索引 。如果不存在,则返回 -1 。
思路:可以借助哈希映射的思想,用一个数组统计每个字母出现的次数,然后再遍历字符串就能够知道那个字母是第一个唯一字符。
class Solution
{
public:
int firstUniqChar(string s)
{
int count[26] = {0};
// 统计字母出现的次数
for(auto ch : s)
{
++count[ch - 'a'];
}
// 顺序遍历字符串,找出第一个唯一字母
for(int i = 0; i < s.size(); i++)
{
if(count[s[i] - 'a'] == 1)
return i;
}
return -1;
}
};
👉字符串相加👈
给定两个字符串形式的非负整数 num1 和num2 ,计算它们的和并同样以字符串形式返回。
你不能使用任何內建的用于处理大整数的库(比如 BigInteger), 也不能直接将输入的字符串转换为整数形式。
思路:这道题主要考察的就是竖式相加,我们可以从字符串的末尾依次往前相加,并保存进位信息。那如何将这个过程转换成代码呢?先定义两个变量end1
和end2
,并初始化为size - 1
,如何利用 while 循环从后往前遍历字符串。当end1
和end2
都小于0时,while 循环结束。在while 循环里,定义三个变量val1
、val2
和ret
,ret
等于val1 + val2 + carry
,其中carry
为进位信息,其值为ret / 10
。如果end
小于0,则val
的值为0。现在我们已经得到了相加的结果和进位信息了,就可以将数据插入到字符串。插入有两种方式,可以头插,也可以尾插。如果采用头插,效率会比较低,因为需要挪动数据。如果采用尾插,循环结束后需要将字符串逆置一下。
class Solution
{
public:
string addStrings(string num1, string num2)
{
int end1 = num1.size() - 1, end2 = num2.size() - 1;
int 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 ? num2[end2] - '0' : 0;
int ret = val1 + val2 + carry;
carry = ret / 10;
ret %= 10;
retStr.insert(0, 1, '0' + ret);
--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;
int 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 ? num2[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;
}
};
👉字符串相乘👈
给定两个以字符串形式表示的非负整数 num1 和 num2,返回 num1 和 num2 的乘积,它们的乘积也表示为字符串形式。
注意:不能使用任何内置的 BigInteger 库或直接将输入转换为整数。
思路:首先将补零,将 num2 的每位数字和 num1 相乘的结果存到 ret1 里,再将 ret1 逆置就能得到 num2 的低位到高位与 num1 相乘的结果,最后将 ret1 加到结果 ret 上,ret 就是最终相乘的结果了。
class Solution
{
public:
string addStrings(string num1, string num2)
{
int end1 = num1.size() - 1, end2 = num2.size() - 1;
int 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 ? num2[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;
}
string multiply(string num1, string num2)
{
// 思路:首先将补零,将num2的每位数字和num1相乘的结果存到ret1里
// 再将ret1逆置就能得到num2的个位、十位和百位与num1相乘的结果
// 再将ret1加到结果ret上,ret就是最终相乘的结果了
string ret("0");
if(num1[0] == '0' || num2[0] == '0')
{
return ret;
}
int size1 = (int)num1.size();
int size2 = (int)num2.size();
for(int i = size2 - 1; i >= 0; i--)
{
string ret1;
for(int j = 0; j < size2 - 1 - i; j++)
{
ret1.append(1, '0');
}
int carry = 0;
for(int j = size1 - 1; j >= 0; j--)
{
int val1 = num1[j] - '0';
int val2 = num2[i] - '0';
int ret2 = val1 * val2 + carry;
carry = ret2 / 10;
ret1.append(1, '0' + ret2 % 10);
}
if(carry > 0)
{
ret1.append(1, '0' + carry);
}
reverse(ret1.begin(), ret1.end());
ret = addStrings(ret, ret1);
}
return ret;
}
};
👉反转字符串中的单词 III👈
给定一个字符串 s ,你需要反转字符串中每个单词的字符顺序,同时仍保留空格和单词的初始顺序。
思路:定义两个变量pos
和prevpos
,pos
的初始值为第一个空格所在的位置,prevpos
的初始值为0。当pos != string::npos
时,进入 while 循环,将prevpos
到pos
之间的字符逆置,逆置之后,prevpos = pos + 1
,pos
等于下一个空格的位置;while 循环结束,将prevpos
到s.size()-1
之间的字符逆置,然后字符串的单词都逆置过来了。
class Solution
{
public:
// 逆置算法
void reverse(string& s, size_t begin, size_t end)
{
while(begin < end)
{
swap(s[begin++], s[end--]);
}
}
string reverseWords(string s)
{
size_t pos = s.find(' ', 0);
size_t prevpos = 0;
while(pos != string::npos)
{
reverse(s, prevpos, pos - 1);
prevpos = pos + 1;
pos = s.find(' ', pos + 1);
}
reverse(s, prevpos, s.size() - 1);
return s;
}
};
👉反转字符串 II👈
给定一个字符串 s 和一个整数 k,从字符串开头算起,每计数至 2k 个字符,就反转这 2k 字符中的前 k 个字符。
- 如果剩余字符少于 k 个,则将剩余字符全部反转。
- 如果剩余字符小于 2k 但大于或等于 k 个,则反转前 k 个字符,其余字符保持原样。
思路:先定义一个逆置算法的函数,然后判断剩余的字符有多少,然后进行相应的操作就行了。
class Solution
{
public:
//翻转start到end区间的字符串
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 reverseStr(string s, int k)
{
int len = s.size();
for(int i = 0; i < len; i += 2 * k)
{
// 剩余字符大于k个
if(i + k - 1 < len)
Reverse(s, i, i + k - 1);
else // 剩余字符小于或等于k个
Reverse(s, i, len - 1);
}
return s;
}
};
👉总结👈
好久没有写过刷题博客了,上次写的时候还是在上次。希望大家能够喜欢,那么以上就是本篇博客的全部内容了,如果大家觉得有收获的话,可以点个三连支持一下!谢谢大家!💖💝❣️