复盘一次失败的技术面试后,我成功拿到了 5 个offer
扫描二维码
随时随地手机看文章
对于开发者来说,找工作最难的莫过于技术面试。Fredrik Strand Oseberg在freeCodeCamp上发表了一篇文章,介绍了自己学习编程6个月后找工作获得5个offer的经历。
我先给大家介绍一下我的工作背景。
在过去的6个月里,我一直在孜孜不倦地研究我的投资组合和个人项目。最值得注意的是,我创建了CryptoDasher,这是一种能够实时跟踪加密货币和投资组合价值的工具。我还参加了一家名为Loopring的中国区块链公司举办的设计竞赛。
我觉得已经准备好了。我向挪威一家大型咨询公司投了简历,应聘其前端开发人员的工作,并引起了他们的注意——至少我是这么认为的。
通过笔试和第一轮面试后,我被邀请参加技术面试。这是最重要的一个环节。
我很紧张。
“你应该怎们准备这个技术面试呢?”我问自己。我四处询问,疯狂地在互联网上搜索相关资料。并在YouTube上看了一些模拟面试。以下是我使用的一些资源:
拆解前端面试(freeCodeCamp上的文章)
David Shariff 的2017年为Web前端开发面试做准备
每个JavaScript开发者都应该知道的10个面试问题
Toptal的JavaScript面试问题列表
Mozilla开发者网络(MDN)
Pramp - 与其他人进行模拟面试的工具
Github前端开发者问题合集
YouTube JS模拟面试1
YouTube JS模拟面试2
我在这些材料上花了好几个小时的时间,尽力为即将到来的面试做好准备。如果我在面试前没有尽最大的努力,我会感到很不舒服,我相信你明白那种感受。
面试的那天到了。我早上4点就醒了。
很害怕,很好奇,也很兴奋。
我在公司的大厅里遇到了面试官,和他一起去了他们的办公室。
我们之间的交谈很愉快,并相互交换了联系方式。我比较擅长软技能,所以我希望能早点展示这种能力。很快,我又见到了另一位面试官,然后去了会议室。
面试刚开始很顺利。我们每个人都进行了自我介绍,他们开始问我一些关于我的背景的问题。
他们问我,开始学习编程时,最困难的部分是什么,我想学习什么样的技术,我想要教别人什么技术,以及我觉得令人兴奋的东西。
在这一点上,我觉得面试很顺利。我很想知道更多关于这家公司的信息,我觉得已经与面试官在某些方面有了共鸣。
然后技术部分开始了。
首先,我被要求解释我在笔试中的代码。是一个为数据集创建分页,并将其显示在列表中的任务。我用React编写了它,然后我开始检查代码。当我们浏览这些代码时,我的面试官会问我一些问题。我将试着把他们提出的问题列出来,以及给出我认为的面试官想要的答案。
你知道单元测试是什么吗?代码的哪一部分可以进行单元测试?
说实话,我想我回答错了。单元测试是一段代码,用于验证一个单元或源代码的特定部分是否执行了它的预期目的,而不会产生不必要的副作用。我不记得我说了什么,但我可能把它和集成测试混在一起了。在面试之前,我确实对单元测试和TDD有一定的了解,但在这种程度上,可能已经超出了我的理解范畴。
在进行了一些讨论之后,我得出结论:我可以对分页函数进行测试,因为它对程序中的大部分逻辑负责。
你将如何改进这个项目?
我发现这个问题有点令人困惑。当我完成笔试(几周前)时,我就被要求列出一份关于该项目的改进清单。假设面试官已经知道了这些,我就很难找到什么改进空间了。
我很快就明白了,面试官对我在电子邮件中提到的事情很感兴趣,于是我开始提到这些点——错误处理、移动优化、Ajax调用加载时的用户反馈、以及大型数据集的页面管理。
你知道BEM是什么吗?你在代码中使用的是BEM吗?
我回答说我知道BEM是什么。这是一个用于CSS项目的命名约定,代表Block、Element、Modifier。我还回答说,在我的CSS类命名中,我受到了BEM的启发,但它并不完全是BEM,因为它没有遵循所有的BEM规则。
你如何使这个网站更具移动友好度?
CSS媒体查询。这是最主要的一个。他们想知道,我是否知道如何利用媒体查询来让网站做出响应。
到目前为止。面试进展很好。我觉得我很完整地回答了这些问题,尽管我需要在了解面试官的具体想问什么之前,先讨论一下这些问题。
编程挑战
然后他们要求我扩展功能。我被要求实现一种排序机制,该机制将采用分页数据集,并根据名称和编号对它们进行重新排列。我有几分钟时间来思考这个问题。
我问了一些问题,比如我是否应该使用内置的JavaScript排序函数,或者构建自己的函数(稍后我们会看到,这是一个很大的错误)。分页数据以对象数组的形式存在,其中每个对象都有一个包含20个对象的数据数组,这些对象代表列表中的每一个项目。我提出了以下的算法:
1、将每个分页对象数据数组合并到一个新的数组中;
2、对新数组进行排序;
3、对排序后的数组进行分页,并将组件的状态设置为新近排序的数组。
这是一个很好的算法。我很快就知道该怎么做了。现在唯一的问题是实施它。这就是我犯错的地方。
首先,我花了很长时间来找出如何组合这些数组。我承认,这种情况给我带来了一些压力。因为我本可以用一个简单的reduce来解决它的时候,我做了各种奇怪的事情。公平地说,我当时并不像现在这样熟悉reduce。
//我应该做的
const pageData = pages.reduce((startingValue,page)=> startingValue.concat(page.data),[])
//我最终做的
const pages = this.state.pages; const pageData = [];pages.forEach(page => pageData = pageData.concat(page.data));
现在我有了一个包含所有数据的数组,我需要编写逻辑来对其进行排序。由于我在编程方面的经验,在很大程度上是基于我自己构建的项目,所以我花了很长时间来处理JavaScript排序函数。我必须要查资料,所以,我花了一些时间来检查MDN和stack overflow上的例子,以便我在实现它之前真正理解它。
我只是部分地完成了分拣工作,我被困在这里好一段时间。数组中的大多数名称都是正确排序的,但是在顶部有一些名称是无序的。在这个时候上,我试图保持冷静,但在我的脑海里,已经崩溃了。我想要知道为什么它没有正确排序。我被困在这里的时间比我想承认的要长。
经过面试官的讨论和督促。我最终想起来了字符串是按照它们的ASCII值排序的。大写字母的值是65-90,小写字母的值是97-122。没有正确排序的结果有一个大写的首字母,它具有先排序的效果,因为它们的ASCII值比小写字母要低。这是一个我永远不会再犯的错误。
当找到问题后,我立即用用.toLowerCase()解决了这个问题。
现在只剩下一件事了。
将已排序的数据传递到分页函数中。
在这里,我遇到了一个麻烦。
分页函数需要一个Ajax响应,并将每一项传递给一个formatData函数,该函数提取相关片段并返回一个新对象。然而,当我试图传递被排序的数据到这个函数的新数组时,它将不再具有原来的属性名,并且函数会给出一个错误。
我花了一些时间研究这个问题,然后我才发现我必须将formatData从分页函数中移出,并在数据传递给分页函数之前在响应数据上执行它。
完成了这些工作,并进行了一些更小的修改,代码终于可以工作了。虽然花了一些时间,但最终我解决了。
此时,技术面试的编程部分结束了。
我感到精疲力竭。
我们最后又聊了一会儿,然后结束了面试。在结束之前,我问题一些问题,他们告诉了我更多关于他们公司的事情。
然而,面试并没有就此止步。
我仔细复盘了这次面试,琢磨我做错了什么。
第二天,我花了三个小时来改进解决方案,然后我发了这封邮件:
嗨,面试X和面试官y。
我想感谢你们昨天同意和我交流。我已经思考了很多关于这个问题的解决方案,我决定今天就改进它。我提供了我们昨天工作的增强版本的代码。这是我所做的:
我扩展了排序功能,以便能够在第二次按下时逆转结果。
我将分类功能扩展到所有的titles。
我添加了一些图标来对titles进行排序。
我重构了分页函数,学习了单元测试的基础知识,并使用Jest来测试它的功能。
我增加了对分页的查询字符串支持,这样重载和链接就会在访问不同的页面时显示正确的数据。
我添加了媒体查询样式,使组件更具移动友好度。
在API调用发生时,我添加了一个加载器。
我添加了错误处理,让用户有机会重新启动API调用。
我在移动设备上改变了排序功能,并使用了一个选择框。
......
这可能有点矫枉过正,但我很受启发,我想要改进解决方案。
最好的问候,
Fredrik Strand Oseberg
这还不够。但至少我尽了最大的努力。过了一段时间,我收到了这封邮件:
嗨!
我们想感谢你的面试,但我们必须得出这样的结论:我们不能给你这个职位的offer,因为你在技术方面没有达到我们的期望。
我们喜欢你的背景,相信你能很好地融入我们的社区,所以我们在你的技术面试中给了你一份详细的反馈,希望你能在获得更多编程经验后再申请我们的职位。
我在哪里出错了?
幸运的是,我得到了一份详细的反馈报告。让我们来看看吧,我将和你们讨论其中的内容。
反馈1:“花费太多的时间来了解如何组合数组。首先在互联网上搜索,而不是检查JavaScript文档(例如:“js array doc”将提供w3schools或MDN,其中列出了函数),并错误地使用了这些示例(array.concat返回一个新数组)。没有人会记住API中的所有内容,所以能够很好地使用JS或库的文档是很重要的。”
要点:面试官希望你首先接触到MDN(或其他相关文档)。他们希望看到你能够找到并阅读文档,并根据发现的信息来实施它。
反馈2:“在排序分配中,候选人首先提出了一个奇怪的手动算法。幸运的是,他选择在JavaScript中使用内置的排序功能,但是不确定它是如何工作的,并且必须反复检查文档。”
要点:在交流中要绝对清楚。在这种情况下,我询问了面试官关于我是否应该使用内置的JavaScript排序功能,以搞清楚手头上任务的界限和限制。不幸的是,我认为这被误解为我建议自己使用的排序算法。
这最终产生了和我想要传达的相反的效果。确保你清楚地表达出你的问题想要澄清的东西。因为它们可能对你来说很有意义,但你的面试官可能会对此有所察觉。
反馈3:“当代码运行时,文本被排序为“区分大小写”。不幸的是,候选人花了很长时间才明白这个问题,但一旦被发现,就立即改了过来。”
要点:速度是最重要的。在编写程序时,总是会出现bug,但是要尽可能快地解决它们。找到问题的根源,如果你不知道,就迅速地去查文档。
反馈4:“花了一些时间来理解为什么要在重构的时候将formatData移出分页。”
再说一遍,速度是最重要的。
反馈5:“许多foreach循环,其中可以用数组.map或array.reduce来解决。了解更多函数式编程将是有益的。”
要点:学习数组.map、array.filter和array.reduce,并熟练地掌握它们。我一直在钻研函数式编程,这是一项艰巨的任务。但是你现在不需要完全精通这些知识,只要确保你能掌握了基础知识就行。
反馈6:“我希望候选人对单元测试有更多的了解。”
要点:这似乎是显而易见的,但重要的问题要多说几遍:测试很重要。测试很重要。测试很重要。学习它。使用它。
这份文件的其余部分都是赞扬。我不会说太多细节,因为它没那么重要。要点是:
他很好地使用了编辑器
他在Chrome中使用调试器(了解高级调试工具很重要)
在继续工作之前,他会检查这些东西是否正常工作(使用console.log)
他试图将代码分成更小的逻辑部分
他使用变量名而不是注释,这使得代码可读性更好
他很了解React
之前的项目令人印象深刻
拥有编程之外(设计/视觉)其他积极的品质
在准备过程中,我还能做些什么?
当你被拒绝的时候,你将不可避免地花费一些时间来思考你可以做些什么不同的事情。
更彻底地检查笔试代码
我花了太多时间研究我的JavaScript知识。我应该更了解我自己的代码。尽管我写了这篇文章,但在写作和面试之间的几周时间里,你需要回顾一下。我希望我在这上面花的时间比在模糊的JavaScript问题上更多。
做更多实操性的JavaScript任务
在面试前我做了很多理论工作。我现在希望我能够花更多的时间做更多的实际工作,或者至少是混合了一些实际的工作,或者构建一些常见的前端组件,比如排序列表、下拉菜单、分页等等。
面试结束
在第一次技术面试结束后我感觉如何?老实说,这是一次很棒的经历。我非常感谢面试官,他们给了我如此详细的反馈,让我能够在下一次面试前纠正我的错误。尽管我没有得到这份工作,但我离成为第一个前端开发者的工作又近了一步。
我也了解到面试是一件反复无常的事情。也许如果我在自己的项目中构建了一个排序机制,或者如果我得到的是一个与我之前完成的任务更接近的任务,面试结果将会有所不同。
我最大的优势是在过去的一年中我花了很多时间学习JavaScript,现在我能够很快地学习和采纳新的想法。不幸的是,我这次没有能力证明这一点。
通往成功之路
现在,我很容易对自己说:“我还不够好。我需要花3-4个月的时间来学习,然后再试一次。”
但我没有。
我决定在两周内尽可能多的申请工作。我向挪威最大的IT公司投递了简历。
两周后,我完成了几家公司的初步面试,然后我又接受了技术面试。
第二轮准备
我在第一次技术面试中学到的一件事是,准备工作很关键。它可以帮助你把技术面试变成是一场考试,并采取必要的步骤来确保你通过考试。
但将考试比作面试是错误的,因为它没有涵盖候选人的全部知识范围。那你能做什么呢?
扩大你的知识范围。
我使用了先进的记忆策略,在8个小时内记住了超过100个面试问题的答案。这些问题可以在这个数据库中找到。
此外,我还在在Code Wars和Hackerrank的实例上花了很多时间。并花了很多时间来构建一些事物。
第二次技术面试
我在上次失败的面试中吸取了很多教训,我做了很多的准备。
这次面试的重点是讨论前端概念。这是一次全面的面试,我觉得面试官想搞清楚我的知识范围,并弄清楚我的强项和弱项。
这次面试持续了大约两个小时。以下是我们所讨论的所有主题的列表:
JS,CSS和HTML概述
文档结构
项目结构
Git
性能
安全
可访问性
搜索引擎优化
响应式网页设计
编程挑战是基于vanilla Javascript的。我被要求用普通的Javascript将一个简单的类添加到一个div中。现在,如果你已经花时间用JS来使用主要的框架,你可能不熟悉classList API。幸运的是,我大部分时间都花在了所有的freeCodeCamp项目上。这就是它的样子:
const btn = document.querySelector('.btn');const menu = document.querySelector('.menu');function addClassNameToDiv() { if (!menu.classList.contains('new-class')) { menu.classList.add('new-class'); } else { menu.classList.remove('new-class'); }}btn.addEventListener('click', addClassNameToDiv)
或者,您可以使用classList.toggle('new class')将其转换为一行程序。如果你点击下拉菜单,我还被要求将它扩展至关闭菜单:
window.addEventListener('click', () => menu.classList.remove('new-class'));
从编程挑战中获得的信息是:
越短越好,只要它总是可读的
在性能方面,最好将查询选择器置于事件监听器回调函数之外(只调用一次,而不是每次都触发)
性能方面,getElementById和getElementByClassName比querySelector更好
第二天,我接到了经理的电话。我通过了面试,他们想给我一个机会。我本可以在这里停下来,不用参加其他的面试了。我可以说:“我已经拿到了一个offer,这已经足够了。”
但我做了相反的事情。
我打电话给所有我正在面试的公司,并告诉他们我已经收到了一个offer,并问他们是否可以加快进程,因为我现在有时间限制。
面试,尤其是技术面试,都是很艰难的心理考验。如果你一直在展示,面试官将期待你的表现能够超越预期。这很难。那么我为什么要这么做呢?
原因有四个。
1、我想向自己证明,这不是运气。
2、我想要尊重每一个给我面试机会的人,给他们一个公平的机会。
3、我想确保自己找到了适合自己的公司,让我成为一名开发人员。
4、为了你们,这个社区对我的帮助很大,我想从技术面试中获得尽可能多的信息,这样你们就可以从我的错误中吸取教训,并做出相应的准备。
我对我从freeCodeCamp获得的帮助和支持感到惭愧,我想要回报。
第三次技术面试
在与其他公司取得联系,并表明我获得了一家顶级公司的offer后,很多公司都迫不及待地想让我通过面试。在一周内,我完成了几次技术面试。
以下是第三次技术面试中的一些问题:
你是如何学习React的?你为什么要学习它?这有什么好处?
Redux是如何工作的?这个API由什么组成的?什么是不变性?不变性的好处是什么?
你将如何重新设计我们的网页?
你如何处理更深层次的应用程序?例如后端?
你自己做测试吗?什么是单元测试?
对你来说,什么是好的用户体验?
如何测试用户体验?
这次面试中的编程挑战是基于CSS的。
我收到了一张纸,上面有一些CSS规则,看起来是这样的:
// HTML Element// CSS Rules#menu { color: black;}.dropdown-menu { color: green;}div { color: blue;}
我的任务是解释我所看到的。我立即识别了了HTML Element并告诉面试官,element上的id和class可以在CSS中使用,以选择HTML Element。在这里,我解释说CSS是级联的,这意味着通常最后一条规则将适用。然而,在这种情况下,选择器有不同的权重。顺序如下所示:id> class>element。
这意味着,在上面的示例中,黑色将被应用到HTML Element中。
第四次技术面试
这是我进行的最后一次技术面试。虽然它仍然很伤脑筋,但现在我已经习惯了。下面是我们讨论的内容:
建立一个基本的网站。确定其中的组件。
你如何让它响应?
如何将文本垂直和水平居中?
什么是CSS框模型?内容框和边框之间的区别是什么?
React有什么好处?
array.forEach在for循环中的好处是什么?有没有可能需要使用for循环的情况?
编程挑战是建立一个不同程度难度的wordwrap函数。想象一下,你只能在屏幕上放20个字符,如果你超过它,你就得从一个新行开始。
我对这个问题的原始解决方案涉及拆分字符串,使用计数器和模数运算符来确定计数是否为20,然后在数组中插入一个换行符并加入字符串。
然后,任务难度增加了,只允许全部单词排成一行。也就是说,如果一个单词导致总数超过20,那么需要在单词前面插入一个换行符。
我在面试中并没有完全解决这个问题,但我的思路是正确的。在我不确定的时候,我使用了MDN,并且我取得了很好的进展。
这就足够了。
我不能把它写下来,如果你感兴趣的话,这里有一个解决的版本:
function wordWrap(str) { let totalCount = 0; const arr = str.split(' '), formattedStr = []; arr.forEach((word, index) => { totalCount += word.length; if (totalCount >= 20) { formattedStr.push('\n', word, ' '); totalCount = word.length; } else { formattedStr.push(word, ' '); } }); return formattedStr.join('');}
结论
如果看到了这里,恭喜你。这是一个漫长的过程。我尽可能提供更多的信息,希望它能帮助像你这样的人。
这样做的结果是,我陷入了一个我从未想过的境地。最后,我有5个offer可供选择。一家大公司甚至给我提供了一个“blind”offer,不管竞争对手给我多少钱,它都能更高。我最终选择了我第一次通过技术面试的公司,因为我相信这对我来说是最合适的。
技术面试可能是一场艰苦的精神折磨。你会受到挑战,你会被带出你的舒适区,这是一件好事。它能帮助你成长。它会让你变得更好。
如果你准备好了,你就能有所收获。
所以从我的经验来看,不要回避技术面试。不要因为你失败了就放弃。不要认为这是你作为开发者的终极衡量标准。它不是。它只是公司用来衡量你的生产力的最简单的工具。
申请工作。准备好。参加技术面试。从错误中学习。不断重复这一过程。
如果你这样做,我保证你会成功。