既然是要谈如何进入BAT,那么我们就从面试的角度来谈学习这件事,会谈谈一流互联网公司对于Java后端程序员的要求,相应的,也会谈谈如何达到这样的要求。 为了简单起见,这些要求分为三个层次,分别为基本要求、可选要求以及加分要求,接下来,我们就一个一个的来谈一谈。 1、基本要求 基本要求就是指,你必需要学会的知识,并且这里面大部份内容,在面试里出现的几率都是极高的。所以,这部份内容你没有选择,只能选择啃下它,你能够花一年,也能够花十年,或者带到棺材里学习也能够。 1)语言的基础部分: 基本要求的第一个,固然是语言的基础部分。基础部分其实就是语法以及一些关键字的做用,像一些if/else、for循环这类基础的语法,以及一些new、class、public这类的基础关键字,大部分状况下面试问的是比较少的,由于这部份内容,只要你写过几年Java,基本上都没有什么问题。 那么基础部分的重点,其实主要就是static、final、transient、volatile这一类的关键字,以及内部类、泛型这一类的高阶语法。 说到static,首先要记住的最重要的一点就是,类属性中被static所引用的变量,会被做为GC的root根节点。做为根节点就意味着,这一类变量是基本上不会被回收的。所以,static很容易引入内存泄漏的风险。 若是一个面试官让你解释static关键字,你告诉他static能够修饰属性、方法和内部类,以及修饰以后又有什么效果的话,那么面试官基本上不会记住你这个回答,整个印象就是平庸。 可是若是你说完之后,补充一下说道,你曾经遇到过一个内存泄漏的问题,就是由于static修饰的一个Map类型的变量致使的,最后排查了堆栈信息找到了问题的所在,而且解决了这个问题。那么,面试官这个时候心里中对你的印象,就会不天然的提高几分。 并且,对于static,更深刻的理解是,static会将所引用的属性、方法以及内部类,与类直接产生引用关系,而非与类的实例。这就是为何,你可使用类名.属性、类名.方法以及类名.内部类名,来直接引用一个被static所修饰的属性、方法或者内部类。 若是你没有用static修饰,那么你就必须使用实例才能引用这些方法、属性或者是内部类,最典型的就是内部类。相信不少同窗都好奇过,为何一个没有被static修饰的内部类,必需要这么声明。 由于你没有使用static修饰InnerClass,因此你必须new出来一个OutterClass的实例,才能在此基础上new出内部类的实例,由于内部类只能经过外部类的实例才能引用。若是你使用了static修饰,那么你就能够这样使用内部类。 这两种方式最大的区别就是,第一种方式,若是你想要得到InnerClass的实例,你必须有一个OutterClass的实例,全部其实这种方式你建立了两个实例,因此有两个new关键字。而第二种方式就好理解一些,静态内部类不依赖于外部类的实例存在,所以只须要直接建立内部类的实例就能够了,因此只有一个new关键字。 static说的有点多了,不过其实不光说了static关键字,也一块儿连同内部类的语法也大体都说了下。那么接下来,基础部分还有一个比较考验人的东西,就是volatile关键字。 这个关键字的重点就三个字,就是可见性。可是面试的时候,你说出可见性三个字,基本上满分100的话,最多只能获得20分。剩下的那80分,就要靠你用硬功夫去得到了。 所谓的硬功夫,其实就是要整明白,在并发当中,可见性究竟是什么意思。那么,为了弄明白可见性什么意思,就须要你了解什么叫主存和工做内存。 只有把这些概念都搞明白了,你才会知道volatile的真正做用究竟是什么。不过有一点要提醒你的是,volatile并不保证同步,这一点必定要记住。不光是应付面试官,在真正使用volatile的时候,也要注意这一点,不然很容易出现问题。 好了,基础部分就说这么多吧,挑了一些有表明性的说了下,归根结底,这一部分就是要你很是清晰的了解Java当中的关键字和语法,这里所谓的了解,是清晰的了解其实现原理,而非简单的会用而已。 2)Java运行时环境 Java运行时环境就是JRE的中文翻译,本质上其实就是指JVM。 首先对于JVM必需要知道的是,JVM与Hotspot的关系。JVM更多的是指JVM规范,而Hotspot是JVM的一种实现,也是咱们最经常使用的JVM实现。你能够把JVM规范当作接口,Hotspot当作实现类,这样去理解会比较简单一些。 此外,JVM最重要的三个部分必需要很是清楚,内存划分、class加载机制以及GC策略。搞清楚这三部分不只仅是为了面试,也是为了让你对于Java有更深入的理解,这对于你的Java生涯很是有帮助。 并且,关于内存划分,还有一点要注意,我们常说的划分方式,实际上是指的Hotspot的划分方式,而非JVM规范所规定的。 Hotspot的内存划分简单说分为三个部分,Young Generation(年轻代)、Old Generation(年老代)以及Perm Generation(永久代)。其中的Young Generation(年轻代),又分为Eden、From和To,其中From和To又统称为Survivor Spaces(幸存区)。 正常状况下,一个对象从建立到销毁,应该是从Eden,而后到Survivor Spaces(幸存区),再到Old Generation(年老代),最后在某次GC下消失。 固然,一个对象也可能直接在Eden里死掉,也可能一直在Old Generation(年老代)存活,这些都是有可能的。 关于内存划分,能够本身没事用内存分析工具看看,好比jmap、jvisualvm等等,观察一下各个区域的内存变化,结合实际去了解一下。 关于classloader机制的学习,能够结合tomcat去学习,了解清楚tomcat的classloader机制,看tomcat是如何保证各个APP之间的类隔离的。若是可能的话,看一下tomcat中classloader的源码,或者看一下开源项目niubi-job,当中也包含了与tomcat类加载机制类似的部分。 至于GC,须要清楚GC Roots都有哪些,以及如何判断一个对象能够被回收。此外,GC的算法和策略也要有大概的了解。 3)并发知识与concurrent包 要想进入一线互联网公司,这部份内容必需要会,不然的话,你始终都只能停留在比较low的段位。 关于并发知识,最重要的两个概念必定要搞清楚,那就是可见性和原子性。其中可见性与前面提到的volatile关键字是息息相关的,可见性只是并发领域里的一个概念,而volatile则是Java语言中,实实在在保证变量可见性的关键字。 前面说了,要弄清楚可见性,就须要搞清楚主存和工做内存。关于主存和工做内存,其实又属于JVM的知识范畴。因此从这里就能够看出来,知识都是有关联性的。 原子性其实相对于可见性来讲,反倒更好理解一些,相信那个万年不变的银行汇款的关于事务的例子,就足以大部分人理解原子性这个概念了,它其实就是一个或多个操做,被视做一个总体的意思。 有了并发的基础知识之后,你就须要研究一下concurrent包了。这里面的东西实际上是一个宝藏,一旦你须要写并发相关的功能,你会发现这里面的东西很是实用。 其中ConcurrentHashMap是面试最容易被问到的一个类,几乎全部的面试都会问你,ConcurrentHashMap和普通的同步HashMap有什么区别。 这个问题其实须要你知道两个知识就能够了,一个是HashMap的数据结构,一个是锁分段的技术,具体的这里就不解释了,你们本身下去找相关资料看吧。 此外,concurrent包里有一个很是重要的类,叫作AbstractQueuedSynchronizer,几乎全部的concurrent包内的并发工具类,都是基于这个抽象类扩展出来的。所以,把AbstractQueuedSynchronizer这个类研究透彻,很是有助于你理解concurrent包。 最后一点,面试的时候还常常会被问到的一个问题,就是ReentrantLock和synchronized关键字有什么区别。 这个问题不少人都答不上来,这只能说明一个问题,那就是大部分人在用synchronized和ReentrantLock的时候,并不会考虑这二者到底用哪一个好一些。 其实它们的区别很简单,简单的说,就是synchronized因为是底层JVM实现的互斥,所以效率会高一些。而ReentrantLock的功能则比synchronized更多,好比定时获取某个锁,多个等待条件等。 并发这一部分是一个程序员进阶的重要部分,但愿全部Java程序员均可以重视这一部分。 4)设计模式和反射 设计模式和反射这部份内容,我的以为是一个高阶程序员必须精通的部分。 用好了这部分知识,可让你在实际开发中少写N多代码,并且还可使得程序的结构更加良好。 关于反射,其实就是reflect包里的内容,这个包里的类其实并不难,主要是得多用,多看。好比Java领域里最经常使用的spring框架,里面其实大量充斥着设计模式和反射的真实使用场景,没事多研究一下,绝对让你受益不浅。 5)文件IO、NIO、网络IO以及网络协议 文件IO、NIO以及网络IO这一部分也是工做当中要常常用到的部分,所以也必需要掌握。 其中NIO更多的是了解其原理,此外,tomcat中有多种协议的实现,其中包括了BIO、NIO和APR,这三者必定很是清楚它们的区别,这个能够在connector的protocol属性配置。 至于网络IO部分,其实就是net包里的内容。这里面的内容是很是经常使用的东西,好比你调用HTTP-API,那么就须要使用这里面的类。在这个restful-API泛滥的时代,你少不了要使用HTTP协议调用API。 此外,在了解这部分的时候,网络协议也要适当的了解一下,最典型的TCP和HTTP协议是必定要了解的。 参加的面试中,基本上TCP协议是必定会问的,虽然这可能和LZ的简历写了TCP协议有关,但好比TCP协议的重试机制,三次握手的过程,TCP与UDP的区别这一类的知识,仍是要了解一下的。 至于HTTP协议,相对来讲就简单不少了,应用层的协议主要是知道其协议格式便可,好比都支持哪些header、每一个header都是什么含义等等。 6)小结 好了,到此为止,基本要求就差很少介绍完了。 没错,其实基本要求这部分,差很少就是要求你有扎实的Java基础。这也是全部一线互联网公司,基本都会写在招聘要求地前几条的要求。 所以,要想进入BAT,那么这一部分的内容必定要了解,并且这部分的内容对你实际开发也是很是有帮助的,并不只仅是为了应付面试。 2、可选要求 看到可选要求四个字,或许很多人会认为这部分不过重要。可是我能够很负责的告诉你,这部分每每才是决定公司要不要你的重要指标。 由于基本要求达标之后,公司主要挑选人才的标准其实就是可选要求这一部分,其实这一部分就是差别性的体现。 此外,要提早说明的是,这些可选要求,没有必须会和必须不会的内容,尽量多的了解,老是不会错的。 1)Spring、Mybatis框架 框架这部分其实不用多说了,spring和mybatis框架的原理和源码,若是你能够很是精通的话,那么这必定能成为你巨大的优点。 若是你是专门作WEB开发的Java后端猿,那么spring和mybatis框架基本上你是确定要用的。精通Spring和mybatis框架不只为了面试,对于你平常开发也有巨大的帮助,你能够作不少架构上的优化,为你的战友省去不少重复性的工做。 关于Spring框架,最核心的固然是IOC,其次即是AOP、MVC这两部分了。好好研究这三部分的源码,会让你从大部分程序员当中,脱颖而出。至于mybatis框架,主要仍是关注它如何实现动态SQL。 并且,待你研究透彻之后,你彻底能够本身尝试去造轮子,说不定能获得意想不到的收获。 2)Linux服务器 这一部分其实本来是运维应该精通的部分,可是做为一个Java后端猿,若是你能够精通linux服务器,那么对你排查线上问题,是有很大的帮助的。 大部分程序员都只知道一些经常使用的Linux命令,对于Linux系统自己的文件系统、网络以及IO等等,是彻底不了解的,这其实也包括LZ本身。可是,LZ见过身边有一些程序员,对于Linux玩的很是熟练,这不光光体如今多会几个命令,而是对整个Linux系统的了解。 能够预见的是,这些人在排查问题的时候,每每会更容易找到问题的根本。由于程序问题每每并非最难解决的,异常这东西见多了就都知道怎么回事了,大不了看看源码也总能找到缘由。最难解决的是环境问题,而环境问题无非就是操做系统层面的问题。 而显然大部分状况下,Java运行的操做系统都是Linux。 3)数据库优化 说完Linux,紧接着LZ要说的就是数据库了,这本来应该是DBA应该精通的部分,但做为一个Java后端猿,数据库基本上也是最常常打交道的了。 并且你们都知道,一个应用的性能瓶颈,每每都出如今数据库这一端,所以,一个Java后端猿若是能够精通数据库的话,那么对于你工做的实际帮助,也是很是大的。 相信很多人都碰到过SQL过慢的状况,这个时候,如何经过加索引、SQL分析和优化的手段,将SQL的执行时间优化到一个可接受的范围内,其实仍是比较考验人的。 反正,这玩意儿我是半斤八两的水平,基本的优化是没有问题的,但稍微复杂一些的就不行了。 因此,这一部分足够成为你的优点,体现出你的差别性。 4)消息服务 除了Linux和数据库之外,消息服务也是当今互联网公司里,必不可少的一个组件。 常见的消息组件好比rabbitMQ、activeMq,包括一些其它的开源消息组件,好比rocketMq。这里面任何一个,若是你能够精通其原理的话,也会成为你有力的竞争条件。 其实消息服务的重点,无非就是如何保证最终一致性、消息的顺序,包括消息事务等等这一类的问题。 虽然本人对此不是很了解,但LZ很肯定,这一部分若是你能够有本身独到的看法的话,必定会大大增长你的成功率。 5)缓存服务 说了消息服务之后,相信缓存服务你们也必定不陌生了。 常见的缓存好比memcached、redis这两个,若是你能搞清楚其中一个的话,也会给你加分许多。毕竟如今的互联网应用,缓存也是必不可少的了,所以若是你能彻底hold住缓存这一部分,那么你的差别性也就有了。 在缓存服务当中,有几个问题也是比较常见的,好比缓存满了怎么办,缓存的实时性如何处理,内存结构如何规划,分布式的状况下如何处理增删节点时缓存的命中问题等等。 6)负载均衡器 负载均衡器,这是最后一个可选要求了。 常见的负载均衡器就两种,一种是软负载均衡,好比nginx、Apache、lvs这一类的。另一种则是硬件负载均衡,常见的主要就是F5。 这两种方式各有优劣,其中硬件负载均衡如要用于简单应用、大访问量的场景,而软件复杂均衡则主要用于复杂应用,较小访问量的场景。固然了,二者还有一个不得不考虑的区别是,硬件复杂均衡通常都是很是贵的,而软负载均衡则基本上没有任何成本。 在负载均衡器方面,也有一些问题是比较常见的。好比如何保持会话,如何作流量控制,负载均衡策略都有哪几种,如何检查后端服务器的健康状态等等。 7)小结 好了,到这里,可选要求就说的差很少了。 这6个要求其实对应的就是Java后端开发中,最常接触到的一些东西。好比spring框架和数据库,这二者基本上是个Java程序员都接触过吧。 其他四个包括Linux服务器、消息服务、缓存服务以及负载均衡器,也是同样的,你们在实际工做当中,应该或多或少都接触过这几个东西。 可是真正能对其中一个很是了解精通的人,相信并不会太多。也正由于如此,若是你作到了,才能够体现出你的差别性,这或许会是你拿下offer的重要筹码之一。 可是,这里必需要再强调一下的是,这几样东西大多数人或多或少都会有一些了解,包括上面提到的问题,很多人也都知道答案。 然而,光知道答案是远远不够的,这并不足以成为你的优点,你须要对这些问题有着深入的了解,以及有着本身独特的看法,才足以让它成为你的优点。 3、加分要求 最后一个即是加分要求了,加分要求虽然不如基本要求和可选要求重要,但它也与可选要求相似,每每拿下offer的最终缘由,正是这些看似不是必要要求的部分。 要强调的是,这些加分要求中,在某些特殊状况下,可能会成为基本要求。 1)数据结构与算法 这一部份内容不用多说了,你们都懂的。精通数据结构与算法,绝对会成为你的一大亮点。 由于大部分程序员的这一部分基础都不太好,包括本人,面试的时候若是问到算法一类的问题,基本上就两个字:不会。 之前还看过Java集合框架的一些源码,对于一些经常使用的数据结构还有必定的了解。可是如今,已经基本上忘光了,就连最基本的冒泡排序,可能都不必定能写的对。 所以能够预见的是,数据结构与算法绝对是很是加分的一项。并且,在你面试一些与算法相关的职位时,这个加分要求还可能会成为基本要求。 2)计算机操做系统 计算机操做系统原理,是很是底层的内容。 这部份内容比较难,里面讲的都是一些最基本的底层原理,例如内存、指令、系统IO等等。以前也研读过一本关于操做系统的书,也写了一系列文章,地址是http://www.cnblogs.com/zuoxiaolong/category/518480.html。 不过看的仍是不够全面和深刻,若是你能够将操做系统研究透彻的话,那么在面试的时候,你彻底能够以此做为突破点,展现你的亮点。 3)计算机网络 其实网络这一部分,对于程序员来讲仍是比较重要的。 最近正在作的事情,就常常会碰到一些网络上的问题,虽然不少时候,这些问题其实能够找专门的网络人员去解决,但若是你本身对此不够了解的话,对于你的工做仍是会形成很大的障碍。 并且,要想精通TCP/IP协议,若是对计算机网络不了解的话,仍是很难真正理解的。 所以,计算机网络部分若是你能够精通的话,这也绝对会成为你的一个加分项。 4)熟练使用一种脚本语言 脚本语言在不少时候是很方便的,并且也很是实用。 最近就被迫正在使用Python作不少事情,其实用了之后你会发现,虽然Java也能够实现一样的目的,但确实选择合适的语言,会帮你节省大量的精力。 所以,若是你能够熟练使用一种脚本语言,好比Python、shell等等,这也一定会成为你的加分项。 5)你的github和博客 这点相信大部分人也都知道,若是你拥有本身的github和博客,而且里面有很多有价值的内容的话,那么必定会为你加很多分。 此外,就不说面试这回事,平时在github写写代码,在博客里写写文章,总结总结本身的技术和职场,也是很是有好处的。相信很多猿友都已经体会到了这其中的益处,也就再也不多说了。 毕竟说多了也无益,最主要的仍是本身要真真实实的去作,若是你但愿能够在这方面加分,那就从当前作起,而且坚持下去。 6)小结 到这里,加分的要求就说的差很少了。 其实可以加分的内容还有不少,只不过列出了比较常见的几种而已,好比你有其它一线互联网公司的背景,这也是能够加分的。只不过这种加分项比较难达到,并且,这里更多说的是草根大学生,所以在这里就没多说。 著做权归做者全部,转载需联系做者以得到受权,引用需注明出处。 做者ID:专业搬砖 连接:https://www.applysquare.com/topic-cn/Rmb5fNT17/