基于我面试别人和被别人面试的经验,我将整理一份清单。我面别人也好别人面我也好,都有一些总会被提及的问题。要记住,有些公司(比如Google)更加关注你是否可以做出高效的算法设计,所以如果你想去那儿工作,除了我下面列出的问题外,你还需要多加练习!
我会在这里添加或是更新这些问题的答案(欢迎你提出有建设性的需求!)
我将问题分为以下几个大类:
请用清晰准确的语句解释如下名词(不需要编码):
this
如何工作,代表什么?实现以下功能:
isPrime
- 返回true
或false
, 表示输入的数是否为质数:isPrime(0) // false
isPrime(1) // false
isPrime(17) // true
isPrime(10000000000000) // false
factorial
- 返回给定数的阶乘的值:factorial(0) // 1
factorial(1) // 1
factorial(6) // 720
fib
-返回斐波那契数列的前n项的和(n为给定) 斐波那契数列.fib(0) // 0
fib(1) // 1
fib(10) // 55
fib(20) // 6765
isSorted
- 返回true
或false
,表示给定的数组是否被排序过:isSorted([]) // true
isSorted([-Infinity, -5, 0, 3, 9]) // true
isSorted([3, 9, -3, 10]) // false
filter
- 实现过滤器功能.filter([1, 2, 3, 4], n => n < 3) // [1, 2]
reduce
- 实现reduce 函数.reduce([1, 2, 3, 4], (a, b) => a + b, 0) // 10
reverse
- 反转给定字符串 (用已封装好的 reverse 是一个cheat,要自己实现).reverse('') // ''
reverse('abcdef') // 'fedcba'
indexOf
- 实现数组的 indexOf方法.indexOf([1, 2, 3], 1) // 0
indexOf([1, 2, 3], 4) // -1
isPalindrome
- 返回true或false判断给定字符串是否是一个回文(palindrome)(大小写不敏感)isPalindrome('') // true
isPalindrome('abcdcba') // true
isPalindrome('abcd') // false
isPalindrome('A man a plan a canal Panama') // true
missing
- 一个数字1至n不重复且未排序过的数字组成的数组,从数字1至数字n选取出不重复且未排序过的数字组成数组(n为最大的数),调用方法后补全数组里缺失的数字。是否可以设计出时间复杂度为O(n)的算法?提示:有个聪明的方法供你使用。missing([]) // undefined
missing([1, 4, 3]) // 2
missing([2, 3, 4]) // 1
missing([5, 1, 4, 2]) // 3
missing([1, 2, 3, 4]) // undefined
isBalanced
- 用true
或false
表示给定的字符串的花括号是否平衡(一一对应)isBalanced('}{') // false
isBalanced('{
{}') // false
isBalanced('{}{}') // true
isBalanced('foo { bar { baz } boo }') // true
isBalanced('foo { bar { baz }') // false
isBalanced('foo { bar } }') // false
fib2
- 实现像上面的fib
函数一样的功能,但是要能够算出数列中前50位以上的数的和。(小提示: 从内存中查询).fib2(0) // 0
fib2(1) // 1
fib2(10) // 55
fib2(50) // 12586269025
isBalanced2
- 实现同上面的isBalanced
函数相同的功能,但是要支持三种类型的括号{},[],和()。带有交错括号的字符串应该返回false。isBalanced2('(foo { bar (baz) [boo] })') // true
isBalanced2('foo { bar { baz }') // false
isBalanced2('foo { (bar [baz] } )') // false
uniq
- 选取一个由数字组成的数组,为其去重,返回去重后的数组。可以实现出时间复杂度为O(n)的算法吗?uniq([]) // []
uniq([1, 4, 2, 2, 3, 4, 8]) // [1, 4, 2, 3, 8]
intersection
- 算出两个数组的交集(公共部分)。可以实现时间复杂度为O(M+N)(M和N为两个数组的长度)的方法吗?intersection([1, 5, 4, 2], [8, 91, 4, 1, 3]) // [4, 1]
intersection([1, 5, 4, 2], [7, 12]) // []
sort
-实现 sort 方法,用于排序元素为数字的数组, 且时间复杂度为O(N×log(N)).sort([]) // []
sort([-4, 1, Infinity, 3, 3, 0]) // [-4, 0, 1, 3, 3, Infinity]
includes
- 判断给定的数字是否出现在给定的已排列好的数组中,返回true
或false
。是否能设计出时间复杂度为O(log(N))的算法?includes([1, 3, 8, 10], 8) // true
includes([1, 3, 8, 8, 15], 15) // true
includes([1, 3, 8, 10, 15], 9) // false
assignDeep
- 仿照 Object.assign, 但是要深度合并对象。为了简单起见,可以假设对象只包含数字或是什么别的(而不是数组、函数等)。assignDeep({ a: 1 }, {}) // { a: 1 }
assignDeep({ a: 1 }, { a: 2 }) // { a: 2 }
assignDeep({ a: 1 }, { a: { b: 2 } }) // { a: { b: 2 } }
assignDeep({ a: { b: { c: 1 }}}, { a: { b: { d: 2 }}, e: 3 })
// { a: { b: { c: 1, d: 2 }}, e: 3 }
reduceAsync
- 仿照reduce 你在“简单”部分中完成了,但每个条目都必须在进行下一步之前被解决。let a = () => Promise.resolve('a')
let b = () => Promise.resolve('b')
let c = () => new Promise(resolve => setTimeout(() => resolve('c'), 100))
await reduceAsync([a, b, c], (acc, value) => [...acc, value], [])
// ['a', 'b', 'c']
await reduceAsync([a, c, b], (acc, value) => [...acc, value], ['d'])
// ['d', 'a', 'c', 'b']
reduceAsync
来实现seq
。seq
使用一个可返回 promise
的函数体内使用数组的函数,然后逐一的解决。let a = () => Promise.resolve('a')
let b = () => Promise.resolve('b')
let c = () => Promise.resolve('c')
await seq([a, b, c]) // ['a', 'b', 'c']
await seq([a, c, b]) // ['a', 'c', 'b']
注意:下面你要实现的数据结构问题,意不在让你记住它们,而只是希望你去查看给出的API,Google是怎么考虑且实现它们的,然后站在一个较高的角度去思考这些实现和其他数据结构相比如何。
permute
- 返回一个字符串数组,包含给定的字符串的所有的排列。permute('') // []
permute('abc') // ['abc', 'acb', 'bac', 'bca', 'cab', 'cba']
debounce
- 实现 debounce 函数.let a = () => console.log('foo')
let b = debounce(a, 100)
b()
b()
b() // only this call should invoke a()
add
和has
。class LinkedList {...}
let list = new LinkedList(1, 2, 3)
list.add(4) // undefined
list.add(5) // undefined
list.has(1) // true
list.has(4) // true
list.has(6) // false
hash()
函数,传入一个字符串,返回一个数字(大多数情况下数字都是唯一的,有时候两个不同的字符串会返回一样的数字):function hash (string) {
return string
.split('')
.reduce((a, b) => ((a << 5) + a) + b.charCodeAt(0), 5381)
}
你有的哈希表需要支持两种方法,get
和set
:
let map = new HashMap
map.set('abc', 123) // undefined
map.set('foo', 'bar') // undefined
map.set('foo', 'baz') // undefined
map.get('abc') // 123
map.get('foo') // 'baz'
map.get('def') // undefined
add
,has
,remove
, size
:let tree = new BinarySearchTree
tree.add(1, 2, 3, 4)
tree.add(5)
tree.has(2) // true
tree.has(5) // true
tree.remove(3) // undefined
tree.size() // 4
let tree = new BinaryTree
let fn = value => console.log(value)
tree.add(1, 2, 3, 4)
tree.bfs(fn) // undefined
tree.inorder(fn) // undefined
tree.preorder(fn) // undefined
tree.postorder(fn) // undefined
下面的每一个问题,先要弄明白为什么给出的代码块无法正常实现功能,然后想出解决方案,编写代码,正常实现功能:
“hey amy”
,但是它打印的是“hey arnold”
,为什么呢?function greet(person) {
if (person == { name: 'amy' }) {
return 'hey amy'
} else {
return 'hey arnold'
}
}
greet({ name: 'amy' })
2.我希望代码按顺序打印出数字0,1,2,3
,但是现在并不是这样的输出(这一度被认为是一个小bug,很多人喜欢在面试的时候提问为什么)
for (var i = 0; i < 4; i++) {
setTimeout(() => console.log(i), 0)
}
“doggo”
,但是现在打出来是undefined
let dog = {
name: 'doggo',
sayName() {
console.log(this.name)
}
}
let sayName = dog.sayName
sayName()
bark()
,但是我得到的确实error
,为什么呢?function Dog(name) {
this.name = name
}
Dog.bark = function() {
console.log(this.name + ' says woof')
}
let fido = new Dog('fido')
fido.bark()
function isBig(thing) {
if (thing == 0 || thing == 1 || thing == 2) {
return false
}
return true
}
isBig(1) // false
isBig([2]) // false
isBig([3]) // true
checkbox, radio button, icon
以及普通的列表项-列表项从很多的表单中得到