快乐数
题目
编写一个算法来判断一个数 n 是不是快乐数.
快乐数定义为:
- 对于一个正整数, 每一次将该数替换为它每个位置上的数字的平方和.
- 然后重复这个过程直到这个数变为 1, 也可能是无限循环但始终变不到 1.
- 如果可以变为 1, 那么这个数就是快乐数.
- 如果 n 是快乐数就返回 true; 不是, 则返回 false.
示例
输入: 19
输出: true
解释:
- 1²+ 9² = 82
- 8² + 2² = 68
- 6² + 8² = 100
- 1² + 0² + 0² = 1
题解
首先把求一个数上每个位置平方和的函数写出来, 这个没啥说的:
var compute = function (n) {
let num = 0
while (n >= 1) {
const remainder = n % 10
num += remainder * remainder
n = (n - remainder) / 10
}
return num
}
然后要明白一个定理: 就是说一个数字, 按每个位置求平方和, 得到的新数字再次按每个位置求平方和...这样往复计算, 总会遇到重复的.
比如说 12 -> 5 -> 25 -> 29 -> 85 -> [89 -> 145 -> 42 -> 20 -> 4 -> 16 -> 37 -> 58 -> 89].
如果你做过 141. 环形链表, 应该一眼看出这是一个环状结构没跑了, 所以快慢指针大法好, HashMap 大法也好, 都可以使出来了.
快慢指针
/**
* @param {number} n
* @return {boolean}
*/
var isHappy = function (n) {
let fast = n,
slow = n
do {
slow = compute(slow)
fast = compute(compute(fast))
} while (fast !== slow)
return slow === 1
}
HashMap
var isHappy = function (n) {
let curNum = n
const set = new Set()
set.add(curNum)
while (curNum !== 1) {
curNum = compute(curNum)
if (set.has(curNum)) {
return false
}
set.add(curNum)
}
return true
}