CS-Notes/docs/notes/Leetcode 题解 - 字符串.md

237 lines
7.2 KiB
Java
Raw Normal View History

2019-04-21 10:36:08 +08:00
<!-- GFM-TOC -->
2019-05-14 22:56:30 +08:00
* [1. 字符串循环移位包含](#1-字符串循环移位包含)
* [2. 字符串循环移位](#2-字符串循环移位)
* [3. 字符串中单词的翻转](#3-字符串中单词的翻转)
* [4. 两个字符串包含的字符是否完全相同](#4-两个字符串包含的字符是否完全相同)
* [5. 计算一组字符集合可以组成的回文字符串的最大长度](#5-计算一组字符集合可以组成的回文字符串的最大长度)
* [6. 字符串同构](#6-字符串同构)
* [7. 回文子字符串个数](#7-回文子字符串个数)
* [8. 判断一个整数是否是回文数](#8-判断一个整数是否是回文数)
* [9. 统计二进制字符串中连续 1 和连续 0 数量相同的子字符串个数](#9-统计二进制字符串中连续-1-和连续-0-数量相同的子字符串个数)
2019-04-21 10:36:08 +08:00
<!-- GFM-TOC -->
2019-03-08 20:31:07 +08:00
2019-03-27 20:57:37 +08:00
2019-05-14 22:56:30 +08:00
# 1. 字符串循环移位包含
2019-03-27 20:57:37 +08:00
[编程之美 3.1](#)
2019-03-08 20:31:07 +08:00
```html
2019-03-27 20:57:37 +08:00
s1 = AABCD, s2 = CDAA
Return : true
2019-03-08 20:31:07 +08:00
```
2019-03-27 20:57:37 +08:00
给定两个字符串 s1 s2要求判定 s2 是否能够被 s1 做循环移位得到的字符串包含
2019-03-08 20:31:07 +08:00
2019-03-27 20:57:37 +08:00
s1 进行循环移位的结果是 s1s1 的子字符串因此只要判断 s2 是否是 s1s1 的子字符串即可
2019-03-08 20:31:07 +08:00
2019-05-14 22:56:30 +08:00
# 2. 字符串循环移位
2019-03-08 20:31:07 +08:00
2019-03-27 20:57:37 +08:00
[编程之美 2.17](#)
2019-03-08 20:31:07 +08:00
```html
2019-03-27 20:57:37 +08:00
s = "abcd123" k = 3
Return "123abcd"
2019-03-08 20:31:07 +08:00
```
2019-03-27 20:57:37 +08:00
将字符串向右循环移动 k
2019-03-08 20:31:07 +08:00
2019-03-27 20:57:37 +08:00
abcd123 中的 abcd 123 单独翻转得到 dcba321然后对整个字符串进行翻转得到 123abcd
2019-03-08 20:31:07 +08:00
2019-05-14 22:56:30 +08:00
# 3. 字符串中单词的翻转
2019-03-08 20:31:07 +08:00
[程序员代码面试指南](#)
```html
2019-03-27 20:57:37 +08:00
s = "I am a student"
Return "student a am I"
2019-03-08 20:31:07 +08:00
```
将每个单词翻转然后将整个字符串翻转
2019-05-14 22:56:30 +08:00
# 4. 两个字符串包含的字符是否完全相同
2019-03-08 20:31:07 +08:00
2019-03-27 20:57:37 +08:00
[242. Valid Anagram (Easy)](https://leetcode.com/problems/valid-anagram/description/)
2019-03-08 20:31:07 +08:00
```html
2019-03-27 20:57:37 +08:00
s = "anagram", t = "nagaram", return true.
s = "rat", t = "car", return false.
2019-03-08 20:31:07 +08:00
```
2019-03-27 20:57:37 +08:00
可以用 HashMap 来映射字符与出现次数然后比较两个字符串出现的字符数量是否相同
2019-03-08 20:31:07 +08:00
2019-03-27 20:57:37 +08:00
由于本题的字符串只包含 26 个小写字符因此可以使用长度为 26 的整型数组对字符串出现的字符进行统计不再使用 HashMap
2019-03-08 20:31:07 +08:00
```java
2019-03-27 20:57:37 +08:00
public boolean isAnagram(String s, String t) {
int[] cnts = new int[26];
for (char c : s.toCharArray()) {
cnts[c - 'a']++;
}
for (char c : t.toCharArray()) {
cnts[c - 'a']--;
}
for (int cnt : cnts) {
if (cnt != 0) {
return false;
}
}
return true;
2019-03-08 20:31:07 +08:00
}
```
2019-05-14 22:56:30 +08:00
# 5. 计算一组字符集合可以组成的回文字符串的最大长度
2019-03-08 20:31:07 +08:00
2019-03-27 20:57:37 +08:00
[409. Longest Palindrome (Easy)](https://leetcode.com/problems/longest-palindrome/description/)
2019-03-08 20:31:07 +08:00
```html
2019-03-27 20:57:37 +08:00
Input : "abccccdd"
Output : 7
Explanation : One longest palindrome that can be built is "dccaccd", whose length is 7.
2019-03-08 20:31:07 +08:00
```
2019-03-27 20:57:37 +08:00
使用长度为 256 的整型数组来统计每个字符出现的个数每个字符有偶数个可以用来构成回文字符串
2019-03-08 20:31:07 +08:00
因为回文字符串最中间的那个字符可以单独出现所以如果有单独的字符就把它放到最中间
```java
2019-03-27 20:57:37 +08:00
public int longestPalindrome(String s) {
int[] cnts = new int[256];
for (char c : s.toCharArray()) {
cnts[c]++;
}
int palindrome = 0;
for (int cnt : cnts) {
palindrome += (cnt / 2) * 2;
}
if (palindrome < s.length()) {
palindrome++; // 这个条件下 s 中一定有单个未使用的字符存在,可以把这个字符放到回文的最中间
}
return palindrome;
2019-03-08 20:31:07 +08:00
}
```
2019-05-14 22:56:30 +08:00
# 6. 字符串同构
2019-03-08 20:31:07 +08:00
2019-03-27 20:57:37 +08:00
[205. Isomorphic Strings (Easy)](https://leetcode.com/problems/isomorphic-strings/description/)
2019-03-08 20:31:07 +08:00
```html
2019-03-27 20:57:37 +08:00
Given "egg", "add", return true.
Given "foo", "bar", return false.
Given "paper", "title", return true.
2019-03-08 20:31:07 +08:00
```
记录一个字符上次出现的位置如果两个字符串中的字符上次出现的位置一样那么就属于同构
```java
2019-03-27 20:57:37 +08:00
public boolean isIsomorphic(String s, String t) {
int[] preIndexOfS = new int[256];
int[] preIndexOfT = new int[256];
for (int i = 0; i < s.length(); i++) {
char sc = s.charAt(i), tc = t.charAt(i);
if (preIndexOfS[sc] != preIndexOfT[tc]) {
return false;
}
preIndexOfS[sc] = i + 1;
preIndexOfT[tc] = i + 1;
}
return true;
2019-03-08 20:31:07 +08:00
}
```
2019-05-14 22:56:30 +08:00
# 7. 回文子字符串个数
2019-03-08 20:31:07 +08:00
2019-03-27 20:57:37 +08:00
[647. Palindromic Substrings (Medium)](https://leetcode.com/problems/palindromic-substrings/description/)
2019-03-08 20:31:07 +08:00
```html
2019-03-27 20:57:37 +08:00
Input: "aaa"
Output: 6
Explanation: Six palindromic strings: "a", "a", "a", "aa", "aa", "aaa".
2019-03-08 20:31:07 +08:00
```
从字符串的某一位开始尝试着去扩展子字符串
```java
2019-03-27 20:57:37 +08:00
private int cnt = 0;
public int countSubstrings(String s) {
for (int i = 0; i < s.length(); i++) {
extendSubstrings(s, i, i); // 奇数长度
extendSubstrings(s, i, i + 1); // 偶数长度
}
return cnt;
2019-03-08 20:31:07 +08:00
}
2019-03-27 20:57:37 +08:00
private void extendSubstrings(String s, int start, int end) {
while (start >= 0 && end < s.length() && s.charAt(start) == s.charAt(end)) {
start--;
end++;
cnt++;
}
2019-03-08 20:31:07 +08:00
}
```
2019-05-14 22:56:30 +08:00
# 8. 判断一个整数是否是回文数
2019-03-08 20:31:07 +08:00
2019-03-27 20:57:37 +08:00
[9. Palindrome Number (Easy)](https://leetcode.com/problems/palindrome-number/description/)
2019-03-08 20:31:07 +08:00
要求不能使用额外空间也就不能将整数转换为字符串进行判断
将整数分成左右两部分右边那部分需要转置然后判断这两部分是否相等
```java
2019-03-27 20:57:37 +08:00
public boolean isPalindrome(int x) {
if (x == 0) {
return true;
}
if (x < 0 || x % 10 == 0) {
return false;
}
int right = 0;
while (x > right) {
right = right * 10 + x % 10;
x /= 10;
}
return x == right || x == right / 10;
2019-03-08 20:31:07 +08:00
}
```
2019-05-14 22:56:30 +08:00
# 9. 统计二进制字符串中连续 1 和连续 0 数量相同的子字符串个数
2019-03-08 20:31:07 +08:00
2019-03-27 20:57:37 +08:00
[696. Count Binary Substrings (Easy)](https://leetcode.com/problems/count-binary-substrings/description/)
2019-03-08 20:31:07 +08:00
```html
2019-03-27 20:57:37 +08:00
Input: "00110011"
Output: 6
Explanation: There are 6 substrings that have equal number of consecutive 1's and 0's: "0011", "01", "1100", "10", "0011", and "01".
2019-03-08 20:31:07 +08:00
```
```java
2019-03-27 20:57:37 +08:00
public int countBinarySubstrings(String s) {
int preLen = 0, curLen = 1, count = 0;
for (int i = 1; i < s.length(); i++) {
if (s.charAt(i) == s.charAt(i - 1)) {
curLen++;
} else {
preLen = curLen;
curLen = 1;
}
if (preLen >= curLen) {
count++;
}
}
return count;
2019-03-08 20:31:07 +08:00
}
```
2019-03-27 20:57:37 +08:00
2019-06-13 13:31:54 +08:00
# 微信公众号
2019-06-10 11:23:18 +08:00
2019-06-18 00:57:23 +08:00
更多精彩内容将发布在微信公众号 CyC2018 你也可以在公众号后台和我交流学习和求职相关的问题另外公众号提供了该项目的 PDF 等离线阅读版本后台回复 "下载" 即可领取公众号也提供了一份技术面试复习大纲不仅系统整理了面试知识点而且标注了各个知识点的重要程度从而帮你理清多而杂的面试知识点后台回复 "大纲" 即可领取我基本是按照这个大纲来进行复习的对我拿到了 BAT 头条等 Offer 起到很大的帮助你们完全可以和我一样根据大纲上列的知识点来进行复习就不用看很多不重要的内容也可以知道哪些内容很重要从而多安排一些复习时间
2019-06-10 11:23:18 +08:00
2019-06-19 23:02:26 +08:00
<div align="center"><img width="480px" src="https://cs-notes-1256109796.cos.ap-guangzhou.myqcloud.com/other/公众号海报3.png"></img></div>