如題
「外觀數列」是一個整數序列,從數字 1 開始,序列中的每一項都是對前一項的描述。前五項如下:
1. 1
2. 11
3. 21
4. 1211
5. 111221
1 被讀作 "one 1" ("一個一") , 即 11。
11 被讀作 "two 1s" ("兩個一"), 即 21。
21 被讀作 "one 2", "one 1" ("一個二" , "一個一") , 即 1211。
給定一個正整數 n(1 ≤ n ≤ 30),輸出外觀數列的第 n 項。
注意:整數序列中的每一項將表示爲一個字符串。
示例 1:
輸入: 1
輸出: "1"
解釋:這是一個基本樣例。
示例 2:
輸入: 4
輸出: "1211"
解釋:當 n = 3 時,序列是 "21",其中我們有 "2" 和 "1" 兩組,"2" 可以讀作 "12",也就是出現頻次 = 1 而 值 = 2;類似 "1" 可以讀作 "11"。所以答案是 "12" 和 "11" 組合在一起,也就是 "1211"。
來源:力扣(LeetCode)
鏈接:https://leetcode-cn.com/problems/count-and-say
著作權歸領釦網絡所有。商業轉載請聯繫官方授權,非商業轉載請註明出處。
代碼實現
/**
* @param {number} n
* @return {string}
*/
var countAndSay = function(n) {
// 方案一 正則解釋在下方
let prev = '1'
for(let i = 1; i < n; i++){
prev = prev.replace(/(\d)\1*/g, item =>`${item.length}${item[0]}`)
}
return prev;
// 方案二 循環遍歷法
if (n == 1) {
return "1";
}
var str = "1";
var sum = '';
for (var i = 1; i < n; i++) {
var sum = '';
for (var j = 0; j < str.length; j++) {
var num = 1;
for (var k = j; k < str.length; k++) {
if (str[k+1] && str[k + 1] == str[j]) {
num += 1;
} else {
break;
}
}
if (num) {
sum += `${num}${str[j]}`;
j += num - 1;
} else {
sum += `1${str[j]}`
}
}
str = sum;
}
return sum;
};
關於正則 /(\d)\1*/g
\1必須與小括號配合使用。
正則表達式中的小括號"()"。是代表分組的意思。 如果再其後面出現\1則是代表與第一個小括號中要匹配的內容相同
\2 則是代表與第二個小括號中要匹配的內容相同
再舉例
1. 驗證6個相同的數字 /^(\d)\1{5}/g 2.驗證3個相同的數字+3個相同的字母 /^(\d)\1{2}([a-z])\2{2}/g