文章列表

如何设计一个亿级API网关?

作者:微信小助手

<section> <section class="V5" powered-by="xiumi.us"> <section> <section> <section> <section class="V5" powered-by="xiumi.us"> <section> <section> <section> <section class="V5" powered-by="xiumi.us"> <section> <section> <section> <section class="V5" powered-by="xiumi.us"> <section> <section> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="padding-top: 10px;padding-right: 10px;padding-left: 10px;box-sizing: border-box;background-color: rgb(239, 239, 239);"> <span style="display: inline-block;width: 5%;line-height: 0.8;font-weight: bolder;font-size: 48px;box-sizing: border-box;"> <section style="box-sizing: border-box;"> “ </section></span> <section style="display: inline-block;vertical-align: top;float: right;width: 90%;line-height: 1.5;font-size: 15px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><span style="letter-spacing: 1px;">API 网关可以看做系统与外界联通的入口,我们可以在网关处理一些非业务逻辑的逻辑,比如权限验证,监控,缓存,请求路由等等。</span></p> </section> </section> </section> </section> </section> <p style="line-height: 1.75em;"><br></p> </section> </section> </section> </section> </section> </section> </section> </section> </section> </section> </section> </section> </section> </section> </section> </section> <p style="text-align: center;margin-left: 8px;margin-right: 8px;margin-bottom: 5px;"><img class="" data-copyright="0" data-ratio="0.45864661654135336" data-s="300,640" src="/upload/e9274560ed8e78f0d990a5e8508eaadf.png" data-type="png" data-w="798" style=""></p> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="border-bottom-width: 1px;border-bottom-style: solid;border-bottom-color: black;margin-top: 0.5em;margin-bottom: 0.5em;line-height: 1.2;box-sizing: border-box;"> <section style="display: inline-block;border-bottom-width: 6px;border-bottom-style: solid;border-color: rgb(89, 89, 89);margin-bottom: -1px;font-size: 20px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;">为什么需要 API 网关</p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">为什么需要 API 网关?有如下几点原因:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">RPC 协议转成 HTTP。</span></strong><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">由于在内部开发中我们都是以 RPC 协议(thrift or dubbo)去做开发,暴露给内部服务,当外部服务需要使用这个接口的时候往往需要将 RPC 协议转换成 HTTP 协议。</span></p></li> <li><p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">请求路由。</span></strong><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">在我们的系统中由于同一个接口新老两套系统都在使用,我们需要根据请求上下文将请求路由到对应的接口。</span></p></li> <li><p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">统一鉴权。</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">因为</span><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">鉴权操作不涉及到业务逻辑,那么可以在网关层进行处理,不用下层到业务逻辑。</span></p></li> <li><p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">统一监控。</span></strong><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">由于网关是外部服务的入口,所以我们可以在这里监控我们想要的数据,比如入参出参,链路时间。</span></p></li> <li><p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">流量控制,熔断降级。</span></strong><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">对于流量控制,熔断降级非业务逻辑可以统一放到网关层。</span></p></li> </ul> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;"></span><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 1px;line-height: 1.75em;">有很多业务都会自己去实现一层网关层,用来接入自己的服务,但是对于整个公司来说这还不够。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> </strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>统一 API 网关</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">统一的 API 网关不仅有 API 网关的所有的特点,还有下面几个好处:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;margin-left: 8px;margin-right: 8px;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">统一技术组件升级。</span></strong><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">在公司中如果有某个技术组件需要升级,那么是需要和每个业务线沟通,通常几个月都搞不定。</span></p><p style="text-align: justify;line-height: 1.75em;"><span style="line-height: 1.6;font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">举个例子如果对于入口的安全鉴权有重大安全隐患需要升级,如果速度还是这么慢肯定是不行,那么有了统一的网关升级是很快的。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">统一服务接入。</span></strong><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">对于某个服务的接入也比较困难,比如公司已经研发出了比较稳定的服务组件,正在公司大力推广,这个周期肯定也特别漫长,由于有了统一网关,那么只需要统一网关统一接入。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">节约资源。</span></strong><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">不同业务不同部门如果按照我们上面的做法应该会都自己搞一个网关层,用来做这个事,可以想象如果一个公司有 100 个这种业务,每个业务配备 4 台机器,那么就需要 400 台机器。</span></p><p style="text-align: justify;line-height: 1.75em;"><span style="line-height: 1.6;font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">并且每个业务的开发 RD 都需要去开发这个网关层,随时去维护,增加人力。如果有了统一网关层,那么也许只需要 50 台机器就可以做这 100 个业务的网关层的事,并且业务 RD 不需要随时关注开发,上线的步骤。</span></p></li> </ul> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="border-bottom-width: 1px;border-bottom-style: solid;border-bottom-color: black;margin-top: 0.5em;margin-bottom: 0.5em;line-height: 1.2;box-sizing: border-box;"> <section style="display: inline-block;border-bottom-width: 6px;border-bottom-style: solid;border-color: rgb(89, 89, 89);margin-bottom: -1px;font-size: 20px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;">统一网关的设计</p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> </strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>异步化请求</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">对于我们自己实现的网关层,由于只有我们自己使用,对于吞吐量的要求并不高,所以我们一般同步请求调用即可。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">对于我们统一的网关层,如何用少量的机器接入更多的服务,这就需要我们的异步,用来提高更多的吞吐量。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">对于异步化一般有下面两种策略:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;margin-left: 8px;margin-right: 8px;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Tomcat/Jetty+NIO+servlet3。</span></strong><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">这种策略使用的比较普遍,京东,有赞,Zuul,都选取的是这个策略,这种策略比较适合 HTTP。在 Servlet3 中可以开启异步。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Netty+NIO。</span></strong><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">Netty 为高并发而生,目前唯品会的网关使用这个策略,在唯品会的技术文章中在相同的情况下 Netty 是每秒 30w+ 的吞吐量,Tomcat 是 13w+。</span></p><p style="text-align: justify;line-height: 1.75em;"><span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">可以看出它们是有一定的差距的,但是 Netty 需要自己处理 HTTP 协议,这一块比较麻烦。</span></p></li> </ul> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">对于网关是 HTTP 请求场景比较多的情况可以采用 Servlet,毕竟可以更加成熟的处理 HTTP 协议。如果更加重视吞吐量那么可以采用 Netty。</span></p> <p style="line-height: normal;"><br></p> <h4 style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 16px;"><strong><span style="color: rgb(89, 89, 89);letter-spacing: 1px;">全链路异步</span></strong></span></h4> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">对于来的请求我们已经使用异步了,为了达到全链路异步我们需要对去的请求也进行异步处理。对于去的请求我们可以利用 RPC 的异步支持进行异步请求。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;margin-bottom: 5px;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">所以基本可以达到下图:</span></p> <p style="text-align: center;margin-left: 8px;margin-right: 8px;margin-bottom: 5px;"><img class="" data-copyright="0" data-ratio="0.36923076923076925" data-s="300,640" src="/upload/cbc964d964361015223a50ce53ac2c28.png" data-type="png" data-w="1300" style=""></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">先在 Web 容器中开启 Servlet 异步,然后进入到网关的业务线程池中进行业务处理,然后进行 RPC 的异步调用并注册需要回调的业务,最后在回调线程池中进行回调处理。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> </strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>链式处理</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">在设计模式中有一个模式叫责任链模式,它的作用是避免请求发送者与接收者耦合在一起,让多个对象都有可能接收请求,将这些对象连接成一条链,并且沿着这条链传递请求,直到有对象处理它为止。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">通过这种模式将请求的发送者和请求的处理者解耦了。<span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 1.6;">在我们的各个框架中对此模式都有实现,比如 Servlet 里面的 Filter,SpringMVC 里面的 Interceptor。</span></span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;margin-bottom: 5px;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">在 Netflix Zuul 中也应用了这种模式,如下图所示:</span></p> <p style="text-align: center;margin-left: 8px;margin-right: 8px;margin-bottom: 5px;"><img class="" data-copyright="0" data-ratio="0.75" data-s="300,640" src="/upload/a734ab49c5242094e0c4b9b1f476f754.png" data-type="png" data-w="960" style=""></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">这种模式在网关的设计中我们可以借鉴到自己的网关设计:</span></p> <ul class=" list-paddingleft-2" style="margin-left: 8px;margin-right: 8px;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">preFilters:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">前置过滤器,用来处理一些公共的业务,比如统一鉴权,统一限流,熔断降级,缓存处理等,并且提供业务方扩展。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">routingFilters:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">用来处理一些泛化调用,主要是做协议的转换,请求的路由工作。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">postFilters:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">后置过滤器,主要用来做结果的处理,日志打点,记录时间等等。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">errorFilters:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">错误过滤器,用来处理调用异常的情况。</span></p></li> </ul> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">这种设计在有赞的网关也有应用。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> </strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>业务隔离</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">上面在全链路异步的情况下不同业务之间的影响很小,但是如果在提供的自定义 Filter 中进行了某些同步调用,一旦超时频繁那么就会对其他业务产生影响。所以我们需要采用隔离之术,降低业务之间的互相影响。</span></p> <p style="line-height: normal;"><br></p> <h4 style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 16px;"><strong><span style="color: rgb(89, 89, 89);letter-spacing: 1px;">信号量隔离</span></strong></span></h4> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">信号量隔离只是限制了总的并发数,服务还是主线程进行同步调用。这个隔离如果远程调用超时依然会影响主线程,从而会影响其他业务。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">因此,如果只是想限制某个服务的总并发调用量或者调用的服务不涉及远程调用的话,可以使用轻量级的信号量来实现。有赞的网关由于没有自定义 Filter,所以选取的是信号量隔离。</span></p> <p style="line-height: normal;"><br></p> <h4 style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 16px;"><strong><span style="color: rgb(89, 89, 89);letter-spacing: 1px;">线程池隔离</span></strong></span></h4> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">最简单的就是不同业务之间通过不同的线程池进行隔离,就算业务接口出现了问题由于线程池已经进行了隔离,那么也不会影响其他业务。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">在京东的网关实现之中就是采用的线程池隔离,比较重要的业务比如商品或者订单,都是单独的通过线程池去处理。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">但是由于是统一网关平台,如果业务线众多,大家都觉得自己的业务比较重要,则需要单独的线程池隔离。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">如果使用的是 Java 语言开发的话,那么在 Java 中线程是比较重的资源,比较受限,如果需要隔离的线程池过多不是很适用。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">如果使用一些其他语言比如 Golang 进行开发网关的话,线程是比较轻的资源,所以比较适合使用线程池隔离。</span></p> <p style="line-height: normal;"><br></p> <h4 style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 16px;"><strong><span style="color: rgb(89, 89, 89);letter-spacing: 1px;">集群隔离</span></strong></span></h4> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">如果有某些业务就需要使用隔离但是统一网关又没有线程池隔离,那么应该怎么办呢?</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">那么可以使用集群隔离,如果你的某些业务真的很重要那么可以为这一系列业务单独申请一个集群或者多个集群,通过机器之间进行隔离。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> </strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>请求限流</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">流量控制可以采用很多开源的实现,比如阿里最近开源的 Sentinel 和比较成熟的 Hystrix。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">一般限流分为集群限流和单机限流:</span></p> <ul class=" list-paddingleft-2" style="margin-left: 8px;margin-right: 8px;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">集群限流:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">利用统一存储保存当前流量的情况,一般可以采用 Redis,这个一般会有一些性能损耗。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">单机限流:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">限流每台机器我们可以直接利用 Guava 的令牌桶去做,由于没有远程调用性能消耗较小。</span></p></li> </ul> <h2 style="line-height: normal;"><br></h2> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>熔断降级</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">这一块也可以参照开源的实现 Sentinel 和 Hystrix,这里不是重点就不多提了。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> </strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>泛化调用</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">泛化调用指的是一些通信协议的转换,比如将 HTTP 转换成 Thrift。在一些开源的网关中比如 Zuul 是没有实现的,因为各个公司的内部服务通信协议都不同。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">比如在唯品会中支持 HTTP1<span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 25.6px;">、</span>HTTP2<span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 25.6px;">、</span>以及二进制的协议,然后转化成内部的协议。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">淘宝的支持 HTTPS<span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 25.6px;">、</span>HTTP1<span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 25.6px;">、</span>HTTP2 这些协议都可以转换成 HTTP、HSF<span style="letter-spacing: 1px;color: rgb(89, 89, 89);font-size: 15px;line-height: 25.6px;">、</span>Dubbo 等协议。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">如何去实现泛化调用呢?由于协议很难自动转换,那么其实每个协议对应的接口需要提供一种映射。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;margin-bottom: 5px;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">简单来说就是把两个协议都能转换成共同语言,从而互相转换,如下图:</span></p> <p style="text-align: center;margin-left: 8px;margin-right: 8px;margin-bottom: 5px;"><img class="" data-copyright="0" data-ratio="0.7197309417040358" data-s="300,640" src="/upload/8da8d8621abda343c432780a190c2138.png" data-type="png" data-w="1338" style=""></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">一般来说共同语言有三种方式指定:</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">json:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">json 数据格式比较简单,解析速度快,较轻量级。在 Dubbo 的生态中有一个 HTTP 转 Dubbo 的项目是用 JsonRpc 做的,将 HTTP 转化成 JsonRpc 再转化成 Dubbo。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;margin-bottom: 5px;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">比如可以将一个 www.baidu.com/id = 1 GET 映射为 json:</span></p> <p style="text-align: center;margin-left: 8px;margin-right: 8px;margin-bottom: 5px;"><img class="" data-copyright="0" data-ratio="0.19175911251980982" data-s="300,640" src="/upload/857e70ac7b6a4a329b34eb1a8130bd66.png" data-type="png" data-w="631" style=""></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">xml:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">xml 数据比较重,解析比较困难,这里不过多讨论。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">自定义描述语言:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">一般来说这个成本比较高,需要自己定义语言来进行描述并进行解析,但是其扩展性,自定义个性化性都是最高。例:Spring 自定义了一套自己的 SPEL 表达式语言。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">对于泛化调用如果要自己设计的话 json 基本可以满足,如果对于个性化的需要特别多的话倒是可以自己定义一套语言。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> </strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>管理平台</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">上面介绍的都是如何实现一个网关的技术关键。这里需要介绍网关的一个业务关键。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">有了网关之后,需要一个管理平台去对我们上面所描述的技术关键进行配置,包括但不限于下面这些配置:</span></p> <ul class=" list-paddingleft-2" style="margin-left: 8px;margin-right: 8px;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">限流</span></strong></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">熔断</span></strong></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">缓存</span></strong></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">日志</span></strong></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">自定义 Filter</span></strong></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">泛化调用</span></strong></p></li> </ul> <h3 style="line-height: normal;"><br></h3> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="border-bottom-width: 1px;border-bottom-style: solid;border-bottom-color: black;margin-top: 0.5em;margin-bottom: 0.5em;line-height: 1.2;box-sizing: border-box;"> <section style="display: inline-block;border-bottom-width: 6px;border-bottom-style: solid;border-color: rgb(89, 89, 89);margin-bottom: -1px;font-size: 20px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;">总结</p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;margin-bottom: 5px;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">最后一个合理的标准网关应该按照如下去实现:</span></p> <p style="text-align: center;margin-bottom: 5px;margin-left: 8px;margin-right: 8px;"><img class="" data-copyright="0" data-ratio="0.5002483854942872" data-s="300,640" src="/upload/8cf3b24eb70a56a7c1dafa43e4db7dac.png" data-type="png" data-w="2013" style=""></p> <p style="text-align: center;margin-left: 8px;margin-right: 8px;margin-bottom: 5px;"><img class="" data-copyright="0" data-ratio="0.5935550935550935" data-s="300,640" src="/upload/222578fda74a57ddc25ac4085cefe7dd.png" data-type="png" data-w="962" style=""></p> <h3 style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 14px;color: rgb(71, 193, 168);"><em><span style="font-size: 14px;letter-spacing: 1px;">参考资料:</span></em></span></h3> <ul class=" list-paddingleft-2" style="margin-left: 8px;margin-right: 8px;"> <li><p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 14px;"><em><span style="font-size: 14px;color: rgb(89, 89, 89);letter-spacing: 1px;">京东:http://www.yunweipai.com/archives/23653.html</span></em></span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 14px;"><em><span style="font-size: 14px;color: rgb(89, 89, 89);letter-spacing: 1px;">有赞网关:https://tech.youzan.com/api-gateway-in-practice/</span></em></span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 14px;"><em><span style="font-size: 14px;color: rgb(89, 89, 89);letter-spacing: 1px;">唯品会:https://mp.weixin.qq.com/s/gREMe-G7nqNJJLzbZ3ed3A</span></em></span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 14px;"><em><span style="font-size: 14px;color: rgb(89, 89, 89);letter-spacing: 1px;">Zuul:http://www.scienjus.com/api-gateway-and-netflix-zuul/</span></em></span></p></li> </ul> <p style="line-height: normal;"><br></p> <p style="white-space: normal;text-align: justify;line-height: 1.75em;"><span style="color: rgb(89, 89, 89);letter-spacing: 1px;"><em><span style="font-size: 14px;">作者:<span style="line-height: 25.6px;">李钊</span></span></em></span></p> <p style="white-space: normal;text-align: justify;line-height: 1.75em;"><span style="color: rgb(89, 89, 89);letter-spacing: 1px;"><em><span style="font-size: 14px;">编辑:陶家龙、孙淑娟</span></em></span></p> <p style="margin-bottom: 5px;text-align: justify;line-height: 1.75em;"><span style="font-size: 14px;color: rgb(89, 89, 89);letter-spacing: 1px;"><em>来源:转载自公众号咖啡拿铁(ID:close_3092860495)</em></span></p> <p style="text-align: center;"><img class="" data-copyright="0" data-ratio="0.3939393939393939" src="/upload/70dc65115d35e7d4c78cd84ded7bacac.gif" data-type="gif" data-w="660" style=""></p> <p style="text-align: center;"> <mp-miniprogram class="miniprogram_element" data-miniprogram-appid="wxc58e6e4759a08388" data-miniprogram-path="pages/index" data-miniprogram-nickname="51CTO播客" data-miniprogram-avatar="http://mmbiz.qpic.cn/mmbiz_png/M3DmwKCWDnP2VudjoA84HWJSg9Ige9Gfy7AKjdzNgmUkbYqIEIagHNoAGmSBibBxEMRNEjOJJArsoFchiakzmtxg/640?wx_fmt=png&amp;wxfrom=200" data-miniprogram-title="51CTO播客,随时随地,碎片化学习" data-miniprogram-imageurl="http://mmbiz.qpic.cn/mmbiz_jpg/MOwlO0INfQr9Q3mtOgKB1pDhwGk6wSq1rzef4CwViaBKDhCHlrlWxqm6gyJB64j3j1STM8vTicHzuQUzoDKGgTCg/0?wx_fmt=jpeg"></mp-miniprogram></p> <section style="box-sizing: border-box;"> <section class="V5" style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 0.5em;margin-bottom: 0.5em;box-sizing: border-box;"> <section style="font-size: 15px;border-style: solid;border-width: 0px 0px 1px;color: rgb(89, 89, 89);border-bottom-color: rgba(215, 215, 215, 0.960784);box-sizing: border-box;"> <p style="box-sizing: border-box;"><span style="letter-spacing: 1px;"><strong>精彩文章推荐:</strong></span></p> </section> </section> </section> </section> <p style="text-align: justify;line-height: 2em;"><a href="http://mp.weixin.qq.com/s?__biz=MjM5ODI5Njc2MA==&amp;mid=2655811899&amp;idx=1&amp;sn=fbeb7d99c3dd5cbb3c51a1163b0a6f10&amp;chksm=bd7434ec8a03bdfadf62cde157289ed1e7154734ed6f7e45c89f814fbcc7a659da536dac1096&amp;scene=21#wechat_redirect" target="_blank" style="font-size: 14px;color: rgb(89, 89, 89);letter-spacing: 1px;text-decoration: none;"><span style="font-size: 14px;color: rgb(89, 89, 89);letter-spacing: 1px;">京东10亿级调用量背后的高可用网关系统架构实践!</span></a><br></p> <p style="text-align: justify;line-height: 2em;"><a href="http://mp.weixin.qq.com/s?__biz=MjM5ODI5Njc2MA==&amp;mid=2655813593&amp;idx=1&amp;sn=579c6bed2f6944485a91463fb2b8abee&amp;chksm=bd74320e8a03bb1883a24fcb8eeecda0e718c299fe225e740d2f632c989ffa5f58181d8f44e2&amp;scene=21#wechat_redirect" target="_blank" style="font-size: 14px;color: rgb(89, 89, 89);letter-spacing: 1px;text-decoration: none;"><span style="font-size: 14px;color: rgb(89, 89, 89);letter-spacing: 1px;">饿了么高稳定、高性能、高可用、高容错API架构实践!</span></a><br></p> <p style="text-align: justify;line-height: 2em;"><a href="http://mp.weixin.qq.com/s?__biz=MjM5ODI5Njc2MA==&amp;mid=2655820821&amp;idx=1&amp;sn=45a9339150f0c815423845cf0fc5c4a7&amp;chksm=bd74d1c28a0358d47933d1d74c007c31053e1348afd6d167d4ac4e2d7b24c15a24baa86b034c&amp;scene=21#wechat_redirect" target="_blank" style="font-size: 14px;color: rgb(89, 89, 89);letter-spacing: 1px;text-decoration: none;"><span style="font-size: 14px;color: rgb(89, 89, 89);letter-spacing: 1px;">终于有人把“TCC分布式事务”实现原理讲明白了!</span></a><br></p>

数据库之架构:主备+分库?主从+读写分离?

作者:微信小助手

<p style="margin-bottom: 5px;max-width: 100%;min-height: 1em;font-variant-numeric: normal;font-variant-east-asian: normal;letter-spacing: 0.544px;widows: 1;font-size: 14px;white-space: pre-line;line-height: 30px;color: rgb(74, 74, 74);font-family: Avenir, -apple-system-font, 微软雅黑, sans-serif;background-color: rgb(255, 255, 255);box-sizing: border-box !important;overflow-wrap: break-word !important;"><span style="font-size: 14px;color: rgb(21, 153, 87);font-family: Menlo, Monaco, Consolas, Inconsolata, 微软雅黑, monospace;font-weight: bold;letter-spacing: 0.544px;">一、数据库架构原则<br></span></p> <ol class=" list-paddingleft-2" style=""> <li><p style="max-width: 100%;min-height: 1em;box-sizing: border-box !important;overflow-wrap: break-word !important;"><span style="max-width: 100%;color: rgb(74, 74, 74);line-height: 22px;font-size: 14px;box-sizing: border-box !important;overflow-wrap: break-word !important;">高可用</span></p></li> <li><p style="max-width: 100%;min-height: 1em;box-sizing: border-box !important;overflow-wrap: break-word !important;"><span style="max-width: 100%;color: rgb(74, 74, 74);line-height: 22px;font-size: 14px;box-sizing: border-box !important;overflow-wrap: break-word !important;">高性能</span></p></li> <li><p style="max-width: 100%;min-height: 1em;box-sizing: border-box !important;overflow-wrap: break-word !important;"><span style="max-width: 100%;color: rgb(74, 74, 74);line-height: 22px;font-size: 14px;box-sizing: border-box !important;overflow-wrap: break-word !important;">一致性</span></p></li> <li><p style="max-width: 100%;min-height: 1em;box-sizing: border-box !important;overflow-wrap: break-word !important;"><span style="max-width: 100%;color: rgb(74, 74, 74);line-height: 22px;font-size: 14px;box-sizing: border-box !important;overflow-wrap: break-word !important;">扩展性</span></p></li> </ol> <h1 style="margin-top: 1.5rem;margin-bottom: 1rem;font-weight: bold;font-size: 2px;max-width: 100%;font-variant-numeric: normal;font-variant-east-asian: normal;letter-spacing: 0.544px;white-space: normal;widows: 1;color: rgb(21, 153, 87);line-height: 1.35;font-family: Menlo, Monaco, Consolas, Inconsolata, 微软雅黑, monospace;background-color: rgb(255, 255, 255);box-sizing: border-box !important;overflow-wrap: break-word !important;"><span style="font-size: 14px;"><strong style="max-width: 100%;box-sizing: border-box !important;overflow-wrap: break-word !important;"><span style="font-size: 14px;max-width: 100%;box-sizing: border-box !important;overflow-wrap: break-word !important;">二、常见的架构方案</span></strong><strong style="max-width: 100%;box-sizing: border-box !important;overflow-wrap: break-word !important;"><span style="font-size: 14px;max-width: 100%;box-sizing: border-box !important;overflow-wrap: break-word !important;"></span></strong></span></h1> <h2 style="margin-top: 1.5rem;margin-bottom: 1rem;font-weight: bold;max-width: 100%;font-variant-numeric: normal;font-variant-east-asian: normal;letter-spacing: 0.544px;white-space: normal;widows: 1;color: rgb(21, 153, 87);line-height: 1.35;font-family: Menlo, Monaco, Consolas, Inconsolata, 微软雅黑, monospace;background-color: rgb(255, 255, 255);box-sizing: border-box !important;overflow-wrap: break-word !important;"><span style="font-size: 14px;color: rgb(0, 0, 0);">方案一:</span><span style="font-size: 14px;color: rgb(0, 0, 0);">主备架构,只有主库提供读写服务,备库冗余作故障转移用</span></h2> <p style="margin-top: 15px;margin-bottom: 15px;max-width: 100%;min-height: 1em;font-variant-numeric: normal;font-variant-east-asian: normal;letter-spacing: 0.544px;widows: 1;font-size: 14px;white-space: pre-line;line-height: 30px;color: rgb(74, 74, 74);font-family: Avenir, -apple-system-font, 微软雅黑, sans-serif;background-color: rgb(255, 255, 255);box-sizing: border-box !important;overflow-wrap: break-word !important;text-align: center;"><img class="" data-ratio="0.9288702928870293" data-type="png" data-w="239" src="/upload/4b8d81d96ad5ce77391e53fe28141268.png" style="border-width: 0px;border-style: none;border-color: initial;border-radius: 6px;box-sizing: border-box !important;overflow-wrap: break-word !important;width: 239px !important;visibility: visible !important;"></p> <pre class="" style="padding-top: 8px;padding-bottom: 6px;max-width: 100%;font-variant-numeric: normal;font-variant-east-asian: normal;letter-spacing: 0.544px;widows: 1;border-radius: 0px;overflow-y: auto;color: rgb(80, 97, 109);font-size: 10px;line-height: 12px;background: rgb(45, 45, 45);box-sizing: border-box !important;overflow-wrap: break-word !important;font-family: consolas, menlo, courier, monospace, &quot;Microsoft Yahei&quot;!important;border-width: 1px !important;border-style: solid !important;border-color: rgb(226, 226, 226) !important;"> <ol class=" list-paddingleft-2" style="list-style-type: none;"> <li><p style="max-widt

这 8 条职场「明规则」,大家藏在心里不说,但你必须知道

作者:微信小助手

<section class="xmteditor" style="display:none;" data-tools="新媒体管家" data-label="powered by xmt.cn"></section> <section data-role="outer" label="Powered by 135editor.com" style="font-size: 16px;font-family: 微软雅黑;"> <section data-role="outer" label="Powered by 135editor.com"> <section style="background-repeat: repeat;background-position: center center;padding: 10px;background-size: auto;background-image: url(&quot;https://mmbiz.qpic.cn/mmbiz_png/hFB4FUPIIlJF1ok7Sx3p3L9Ax808rCIyYEOLHglJiayLG1b8P4JJSziarETsYBImZVFibvZiciavOfQvWx2847bm71g/640?wx_fmt=png&quot;);box-sizing: border-box;"> <section> <p style="text-align: center;"><img class="" data-copyright="0" data-ratio="0.2671875" src="/upload/c164f8ab6c1beb7073d641edf10e8cfb.gif" data-type="gif" data-w="640" style=""></p> <p style="white-space: normal;"><br></p> <p style="text-align: center;"><img class="" data-copyright="0" data-ratio="0.66640625" data-s="300,640" src="/upload/418286dfa2ed7edc43e421444f26a696.jpg" data-type="jpeg" data-w="1280" style=""></p> <p style="white-space: normal;"><em><span style="color: rgb(127, 127, 127);font-size: 12px;caret-color: red;">题图:《实习生》</span></em></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p><span style="font-size: 18px;"><strong><span style="color: #FFFFFF;background-color: #0F88EB;">有哪些大家不说,但需要知道的社会规则?</span></strong></span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <section class="_135editor" data-tools="135编辑器" data-id="1" data-color="rgb(15, 136, 235)" style="border-width: 0px;border-style: none;border-color: initial;box-sizing: border-box;"> <section class="135brush" style="border-left: 5px solid #0f88eb;font-weight: bold;line-height: 32px;padding: 5px 10px;margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <p><span style="color: #3f3f3f;font-size: 15px;">知友:Sean Ye<span style="color: #7f7f7f;font-size: 14px;">(17000+ 赞同,人力资源话题优秀回答者,知乎编辑推荐)</span></span></p> </section> </section> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">整理几条职场明规则,这几条不但很少人提,更多时候有人希望用所谓「职场潜规则」来反向误导我们,希望大家提高警惕:</span></p> <p><br></p> <section class="_135editor" data-tools="135编辑器" data-id="93962" style="border-width: 0px;border-style: none;border-color: initial;box-sizing: border-box;"> <section style="text-align: center;padding-top: 10px;padding-bottom: 10px;box-sizing: border-box;"> <section style="display:flex;justify-content: center;align-items: center;"> <section style="height: 0px;width: 0px;border-left: 10px solid #0f88eb;border-bottom: 18px solid transparent;margin-bottom: -30px;margin-right: -10px;border-top-color: #0f88eb;border-right-color: #0f88eb;transform: rotate(25deg);-webkit-transform: rotate(25deg);-moz-transform: rotate(25deg);-o-transform: rotate(25deg);"></section> <section style="display: inline-block;background: rgb(15, 136, 235);padding-top: 4px;padding-bottom: 4px;color: rgb(255, 255, 255);box-sizing: border-box;"> <section style="font-size: 18px;padding-right: 12px;padding-left: 12px;box-sizing: border-box;" class="autonum" data-original-title="" title=""> 1 </section> </section> <section style="display: inline-block;margin-left: 4px;padding-top: 2px;padding-bottom: 2px;border-bottom: 3px solid rgb(15, 136, 235);border-top-color: rgb(15, 136, 235);border-left-color: rgb(15, 136, 235);border-right-color: rgb(15, 136, 235);box-sizing: border-box;"> <section class="135brush" data-brushtype="text" style="color: rgb(51, 51, 51);padding-right: 4px;padding-left: 4px;font-weight: bold;letter-spacing: 1px;box-sizing: border-box;"> <span style="font-size: 15px;">毕业能去大公司的,别去小公司</span> </section> </section> </section> </section> </section> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><strong><span style="color: rgb(89, 89, 89);font-size: 15px;">(这条肯定被喷,但还是得说)</span></strong></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">很多小公司招人都会说:与其去大公司做螺丝钉,只精通一块,不如到我们小公司,学得更多,成长更快。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">我呸。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">成长?<strong>你那是做杂务做得多,能学到啥精华。</strong>相反在成体系的大公司,专精一个领域之后,你完全有轮岗的机会,去其他领域继续专精。大公司预算多平台大,经历的事情天然就比小公司更多,小公司预算有限能做的也就那几样,放在大公司面前根本没法比。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">我一个 HR 朋友,从 GE 跳槽到了小公司,说自己很心虚。我问原因,她说:</span></p> <p><br></p> <blockquote> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 14px;">我原来在 GE 管上万人的薪资福利,现在来了小公司薪水翻上去了,但管的人就两三百。同事说忙死了,我天天下午 3 点就没事做了……</span></p> </blockquote> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">职业发展?肯定是大公司好啊,你看那些招人时口口声声我这里学得多的小公司,一到融资就会喊我这里有 BAT 背景员工,为什么?大公司的光环背书啊。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">钱?一般来说大公司更多,小公司画的饼就别想了,能分应届毕业生多少期权?行权价多优惠?小公司就算能多给一两千块,那咱也得看未来的钱不是吗?</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">我在<a class="weapp_text_link" style="font-size:15px;" data-miniprogram-appid="wxeb39b10e39bf6b54" data-miniprogram-path="zhihu/answer?id=147717382" data-miniprogram-nickname="知乎热榜" href="">你有什么道理后悔没有早点知道?</a>分享过自己的教训,和《How Google Works》里谷歌创始人的建议:</span></p> <p><br></p> <blockquote> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 14px;">在你职业生涯初期,能够获得的股权激励很有限,因此在正确的行业磨练技能要比在某家公司赌上自己的命运更加合算。在此之后,随着经验(以及年龄)的积累,挑选合适的企业变得越发重要。因为那时,股票在你的薪酬构成中所占比例大幅提升,因此你也应该相应的将对公司的挑选放在优先位置。</span></p> </blockquote> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <section class="_135editor" data-tools="135编辑器" data-id="93962" style="border-width: 0px;border-style: none;border-color: initial;box-sizing: border-box;"> <section style="text-align: center;padding-top: 10px;padding-bottom: 10px;box-sizing: border-box;"> <section style="display:flex;justify-content: center;align-items: center;"> <section style="height: 0px;width: 0px;border-left: 10px solid #0f88eb;border-bottom: 18px solid transparent;margin-bottom: -30px;margin-right: -10px;border-top-color: #0f88eb;border-right-color: #0f88eb;transform: rotate(25deg);-webkit-transform: rotate(25deg);-moz-transform: rotate(25deg);-o-transform: rotate(25deg);"></section> <section style="display: inline-block;background: rgb(15, 136, 235);padding-top: 4px;padding-bottom: 4px;color: rgb(255, 255, 255);box-sizing: border-box;"> <section style="font-size: 18px;padding-right: 12px;padding-left: 12px;box-sizing: border-box;" class="autonum" data-original-title="" title="" data-num="2"> 2 </section> </section> <section style="display: inline-block;margin-left: 4px;padding-top: 2px;padding-bottom: 2px;border-bottom: 3px solid rgb(15, 136, 235);border-top-color: rgb(15, 136, 235);border-left-color: rgb(15, 136, 235);border-right-color: rgb(15, 136, 235);box-sizing: border-box;"> <section class="135brush" data-brushtype="text" style="color: rgb(51, 51, 51);padding-right: 4px;padding-left: 4px;font-weight: bold;letter-spacing: 1px;box-sizing: border-box;"> <span style="font-size: 15px;">为自己的前途熬夜,不为公司熬夜</span> </section> </section> </section> </section> </section> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">年轻人说一点不加班,真的不太容易,我如今 35 岁朝上了,加班还是逃不掉的。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">只要身体还算好,偶尔加几次班是可以接受的。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">但我们需要弄清楚,我们的加班是不是有价值。方法很简单,问自己三个问题:</span></p> <p><br></p> <ul class=" list-paddingleft-2" style=""> <li><p style="line-height: 1.75em;letter-spacing: 0.5px;"><strong><span style="color: rgb(89, 89, 89);font-size: 15px;">通过加班,我的职场技能 / 专业知识能成长吗?</span></strong></p></li> </ul> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">如果这个项目有我一直很想学习的领域,或者能显著提升自己的能力,未来还能给自己的职业发展背书,那么何乐而不为呢?</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">当初张小龙的团队是在腾讯众多团队的竞争中用加班抢出微信这个产品,这个团队里的成员得到了实打实的实惠,就算是跳槽也有微信背书,这班加得值得。</span></p> <p><br></p> <ul class=" list-paddingleft-2" style=""> <li><p style="line-height: 1.75em;letter-spacing: 0.5px;"><strong><span style="color: rgb(89, 89, 89);font-size: 15px;">通过加班,能赢得更多的机会吗?</span></strong></p></li> </ul> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">有时候我们加班做的是一些杂务,但是我们知道我们的领导是靠谱的,这次我帮了领导,下回有好的培训机会,或者是轮岗机会,领导会优先想到我,考虑我,那这样的加班为什么不去呢?</span></p> <p><br></p> <ul class=" list-paddingleft-2" style=""> <li><p style="line-height: 1.75em;letter-spacing: 0.5px;"><strong><span style="color: rgb(89, 89, 89);font-size: 15px;">加班是常态性的吗?加了班能轮休吗?</span></strong></p></li> </ul> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">职场和生活一样,我帮你一把你帮我一把,如果老板急需的时候我加了班,那么我家里有事需要请假的时候,领导会毫不犹豫的批假吗?</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">&nbsp;<a class="weapp_text_link" style="font-size:15px;" data-miniprogram-appid="wxeb39b10e39bf6b54" data-miniprogram-path="zhihu/answer?id=125929932" data-miniprogram-nickname="知乎热榜" href="">员工不愿意加班,老板该怎么办?</a>里我提过自己的观点:所谓弹性工作制,是种互利模式。你希望员工可以为公司救急,企业也得给员工提供便利吧。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">996 常态化加班的公司,最为可怕:它以制度剥夺了员工的私人时间和个人成长的时间,而并不觉得有任何问题,这种加班对员工的伤害最重。</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <section class="_135editor" data-tools="135编辑器" data-id="93962" style="border-width: 0px;border-style: none;border-color: initial;box-sizing: border-box;"> <section style="text-align: center;padding-top: 10px;padding-bottom: 10px;box-sizing: border-box;"> <section style="display:flex;justify-content: center;align-items: center;"> <section style="height: 0px;width: 0px;border-left: 10px solid #0f88eb;border-bottom: 18px solid transparent;margin-bottom: -30px;margin-right: -10px;border-top-color: #0f88eb;border-right-color: #0f88eb;transform: rotate(25deg);-webkit-transform: rotate(25deg);-moz-transform: rotate(25deg);-o-transform: rotate(25deg);"></section> <section style="display: inline-block;background: rgb(15, 136, 235);padding-top: 4px;padding-bottom: 4px;color: rgb(255, 255, 255);box-sizing: border-box;"> <section style="font-size: 18px;padding-right: 12px;padding-left: 12px;box-sizing: border-box;" class="autonum" data-original-title="" title="" data-num="3"> 3 </section> </section> <section style="display: inline-block;margin-left: 4px;padding-top: 2px;padding-bottom: 2px;border-bottom: 3px solid rgb(15, 136, 235);border-top-color: rgb(15, 136, 235);border-left-color: rgb(15, 136, 235);border-right-color: rgb(15, 136, 235);box-sizing: border-box;"> <p style="color: rgb(51, 51, 51);padding-right: 4px;padding-left: 4px;font-weight: bold;letter-spacing: 1px;"><span style="font-size: 15px;">即使你不想跳槽</span></p> <p style="color: rgb(51, 51, 51);padding-right: 4px;padding-left: 4px;font-weight: bold;letter-spacing: 1px;"><span style="font-size: 15px;caret-color: red;">也要定期看看工作机会</span></p> </section> </section> </section> </section> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">哎,这贴写得真是得罪开公司的老总。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">定期看工作机的核心好处是:能跳出自己的圈子,看整个市场在做什么——保持市场敏感度。</span></p> <p><br></p> <p><strong><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 0.5px;caret-color: red;">(1)保持市场前沿科技和趋势的嗅觉:</span></strong><br></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">在公司里你看到的永远只是周边一小部分事情,你的项目越多事情越忙,就更腾不出手去看整个市场,但职场发展,无论走专家路线还是管理路线,你都要了解市场。否则很容易被市场淘汰。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><strong><span style="color: rgb(89, 89, 89);font-size: 15px;">(2)更清晰认识自己的不足和优势:</span></strong></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">出去面试,了解用人公司为什么想要你,这是你的核心优势;了解用人公司认为你还有哪些不足。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><strong><span style="color: rgb(89, 89, 89);font-size: 15px;">(3)了解自己的价格</span></strong></p> <p><br></p> <p><img class="" data-ratio="0.66" src="/upload/39de8220af3f1a961870b0e8772fc5f8.jpg" data-type="jpeg" data-w="600"></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">2016 年年终的调薪幅度,市场中位值是 6%,而且这个数据已经连续五年下滑。正常来说,我们如果五年没跳槽,内部也没有特殊调薪的话,我们的薪水老早被市场水平甩出 8 条长安街了。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">美世有薪酬调研报告,能精准到每个岗位每个城市不同级别员工的薪水的市场中位值。最便宜的报告也得 5-8 万,咱求职者又买不起,最简单的方法就是通过面试。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">当然,除此以外,还可以多交一些猎头朋友,详细的可以看这一篇:<a class="weapp_text_link" style="font-size:15px;" data-miniprogram-appid="wxeb39b10e39bf6b54" data-miniprogram-path="zhihu/article?id=21558028" data-miniprogram-nickname="知乎热榜" href="">工作五年,给名企投简历却石沉大海,怎么办?</a>其中将不同时期不同类型的猎头做了分类,对猎头好一点,交几个靠谱的猎头朋友,是职场人必须要做的,就算你不跳槽。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><strong><span style="color: rgb(89, 89, 89);font-size: 15px;">(4)提升面试技能和沟通技巧</span></strong></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">这个不多说了,面试技巧也要靠实操训练嘛。精细的面试准备,不但能让我们更好的应对面试,也能更清楚的了解(目标)行业内企业的现状。</span></p> <p><br></p> <section class="_135editor" data-tools="135编辑器" data-id="93962" style="border-width: 0px;border-style: none;border-color: initial;box-sizing: border-box;"> <section style="text-align: center;padding-top: 10px;padding-bottom: 10px;box-sizing: border-box;"> <section style="display:flex;justify-content: center;align-items: center;"> <section style="height: 0px;width: 0px;border-left: 10px solid #0f88eb;border-bottom: 18px solid transparent;margin-bottom: -30px;margin-right: -10px;border-top-color: #0f88eb;border-right-color: #0f88eb;transform: rotate(25deg);-webkit-transform: rotate(25deg);-moz-transform: rotate(25deg);-o-transform: rotate(25deg);"></section> <section style="display: inline-block;background: rgb(15, 136, 235);padding-top: 4px;padding-bottom: 4px;color: rgb(255, 255, 255);box-sizing: border-box;"> <section style="font-size: 18px;padding-right: 12px;padding-left: 12px;box-sizing: border-box;" class="autonum" data-original-title="" title="" data-num="4"> 4 </section> </section> <section style="display: inline-block;margin-left: 4px;padding-top: 2px;padding-bottom: 2px;border-bottom: 3px solid rgb(15, 136, 235);border-top-color: rgb(15, 136, 235);border-left-color: rgb(15, 136, 235);border-right-color: rgb(15, 136, 235);box-sizing: border-box;"> <section class="135brush" data-brushtype="text" style="color: rgb(51, 51, 51);padding-right: 4px;padding-left: 4px;font-weight: bold;letter-spacing: 1px;box-sizing: border-box;"> <span style="font-size: 15px;">不要一味看高薪跳槽</span> </section> </section> </section> </section> </section> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">这话得罪的是职场人,不是说好的不要搞毒鸡汤的吗?你怎么反而站在老板那边忽悠咱不要看薪水?</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">知乎有个问题:<a class="weapp_text_link" style="font-size:15px;" data-miniprogram-appid="wxeb39b10e39bf6b54" data-miniprogram-path="zhihu/answer?id=122271142" data-miniprogram-nickname="知乎热榜" href="">提离职被老板加薪挽留,该怎么办?</a></span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">我一直认为这是个伪命题,对于职场人来说,跳槽首先看的是职业发展,说俗一点就是未来的钱。</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">刚毕业的同学,几个公司比价,真能高个 2000-3000 的薪水,已经是雇主能给出的最优厚价码了;如果能选择一条正确的发展路径,到了 35-40 岁时,薪水差距能达到 50 万以上。</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">我有和我同岁的大学同学已经年薪百万了,我这厢还只是人一半……让我哭一会儿去。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">如果在一家企业,我们确定我们会有好的发展,那么就应该去,无论给的钱是不是多,即使不多,未来我们能把这钱捞回来。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">如果要细分,那么未来的钱,还要看行业的未来和具体岗位的细分,请各位移步:<a class="weapp_text_link" style="font-size:15px;" data-miniprogram-appid="wxeb39b10e39bf6b54" data-miniprogram-path="zhihu/answer?id=165065421" data-miniprogram-nickname="知乎热榜" href="">年轻时一味追求高工资会给自己带来什么?这样真的有利吗?</a></span></p> <p><br></p> <section class="_135editor" data-tools="135编辑器" data-id="93962" style="border-width: 0px;border-style: none;border-color: initial;box-sizing: border-box;"> <section style="text-align: center;padding-top: 10px;padding-bottom: 10px;box-sizing: border-box;"> <section style="display:flex;justify-content: center;align-items: center;"> <section style="height: 0px;width: 0px;border-left: 10px solid #0f88eb;border-bottom: 18px solid transparent;margin-bottom: -30px;margin-right: -10px;border-top-color: #0f88eb;border-right-color: #0f88eb;transform: rotate(25deg);-webkit-transform: rotate(25deg);-moz-transform: rotate(25deg);-o-transform: rotate(25deg);"></section> <section style="display: inline-block;background: rgb(15, 136, 235);padding-top: 4px;padding-bottom: 4px;color: rgb(255, 255, 255);box-sizing: border-box;"> <section style="font-size: 18px;padding-right: 12px;padding-left: 12px;box-sizing: border-box;" class="autonum" data-original-title="" title="" data-num="5"> 5 </section> </section> <section style="display: inline-block;margin-left: 4px;padding-top: 2px;padding-bottom: 2px;border-bottom: 3px solid rgb(15, 136, 235);border-top-color: rgb(15, 136, 235);border-left-color: rgb(15, 136, 235);border-right-color: rgb(15, 136, 235);box-sizing: border-box;"> <section class="135brush" data-brushtype="text" style="color: rgb(51, 51, 51);padding-right: 4px;padding-left: 4px;font-weight: bold;letter-spacing: 1px;box-sizing: border-box;"> <span style="font-size: 15px;">别太早想着斜杠青年</span> </section> </section> </section> </section> </section> <p><br></p> <p><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 0.5px;caret-color: red;">南都周刊今年找我约稿写</span><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 0.5px;caret-color: red;">「</span><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 0.5px;caret-color: red;">斜杠青年</span><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 0.5px;caret-color: red;">」</span><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 0.5px;caret-color: red;">,而我的态度是年轻人的目标应该是成为斜杠中年。</span><br></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">《双重职业》这本书的</span><span style="color: rgb(89, 89, 89);font-size: 15px;caret-color: red;">作者麦瑞克·阿尔伯最初用的词是 Slash——斜杠可以,青年完全是译者的发挥啊。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">去互联网上搜索一下知名的职场「斜杠青年」,但凡做出些成绩的大多是 30+ 的中年人:</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">企业家/登山爱好者/滑翔伞玩家的王石,美赞臣高管同时写出 10 万 + 微信文章的薇安;</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">PPT 专家/大学生导师/社团运营专家的秋叶大叔,创业者 / 生涯规划师 / 畅销书作者的古典;</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">就算是天时地利人和一帆风顺的李叫兽(自媒体/营销咨询顾问/百度副总裁)也要将近 30 岁了。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">哪有刚毕业就杠上开花的?</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">先把专业做精细了。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">所谓斜杠,应当是百尺竿头再进一步的飞跃,而非焦头烂额慌乱之下的逃窜。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">有些自诩为「斜杠青年」的小伙伴们,可能从头到尾就没有读过麦瑞克·阿尔伯的《双重职业》这本书,仅仅凭听到的道听途说和自己的猜测,树立了一个自己眼里所谓的「斜杠青年」。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">作者表示这个锅她不背。麦瑞克对于想做斜杠青年的小伙伴们提出过忠告:</span></p> <p><br></p> <blockquote> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 14px;">你可以斜杠,但别一拥而上。不要指望同时以三个身份开展三个职业。<strong>大多数斜杠,需要经过时间的洗礼和岁月的磨砺,从基础做起逐渐成就卓越。只有在坚实的基础上,才能进一步添砖加瓦,斜出特色,杠上开花。</strong></span></p> </blockquote> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <section class="_135editor" data-tools="135编辑器" data-id="93962" style="border-width: 0px;border-style: none;border-color: initial;box-sizing: border-box;"> <section style="text-align: center;padding-top: 10px;padding-bottom: 10px;box-sizing: border-box;"> <section style="display:flex;justify-content: center;align-items: center;"> <section style="height: 0px;width: 0px;border-left: 10px solid #0f88eb;border-bottom: 18px solid transparent;margin-bottom: -30px;margin-right: -10px;border-top-color: #0f88eb;border-right-color: #0f88eb;transform: rotate(25deg);-webkit-transform: rotate(25deg);-moz-transform: rotate(25deg);-o-transform: rotate(25deg);"></section> <section style="display: inline-block;background: rgb(15, 136, 235);padding-top: 4px;padding-bottom: 4px;color: rgb(255, 255, 255);box-sizing: border-box;"> <section style="font-size: 18px;padding-right: 12px;padding-left: 12px;box-sizing: border-box;" class="autonum" data-original-title="" title="" data-num="6"> 6 </section> </section> <section style="display: inline-block;margin-left: 4px;padding-top: 2px;padding-bottom: 2px;border-bottom: 3px solid rgb(15, 136, 235);border-top-color: rgb(15, 136, 235);border-left-color: rgb(15, 136, 235);border-right-color: rgb(15, 136, 235);box-sizing: border-box;"> <section class="135brush" data-brushtype="text" style="color: rgb(51, 51, 51);padding-right: 4px;padding-left: 4px;font-weight: bold;letter-spacing: 1px;box-sizing: border-box;"> <span style="font-size: 15px;">打磨自己的沟通技能</span> </section> </section> </section> </section> </section> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">我打过交道的不少年轻小伙伴都认为:善于沟通 = 性格外向,能聊天能侃大山。认为自己很能聊的小伙伴们当然不吝惜将出色的沟通能力放在自己的简历上就来投递咨询公司的岗位了。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">沟通能力并不仅仅说你口吐莲花,拽上客户吧啦吧啦侃上小半天,唬得客户一愣一愣滴。</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">在我看来,<strong>沟通能力首先在于听,</strong>听懂客户的需求,更能听懂弦外之音——客户真正想要的是什么。</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">我参加过一个培训,隐约记得考官出过一题:如果客户做出错误决策时,应该如何说服他?</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">有人说讲述过去的成功案例;有人说用数据分析的结果说服客户;而正确答案是继续听,了解客户为什么会这么说,而非急于说服客户。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><strong><span style="color: rgb(89, 89, 89);font-size: 15px;">提问也是重要的能力:</span></strong><span style="color: rgb(89, 89, 89);font-size: 15px;">客户说准备来年上一套弹性福利体系,身为咨询顾问可不能点头回家做方案去了。为什么客户会有这种想法?是全球总部的要求?还是员工调研的反馈?抑或是竞争对手上了这样的福利?</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">出色的顾问能够提出有效的问题(ask effective question 是顾问培训中的必修课),迅速了解客户决策背后的原因。只有了解了客户需求背后的动因才能提出靠谱的解决方案。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">刚刚进入职场的小伙伴们,很容易想当然,又怕问多了露怯(我当年也这样),于是乎,干活就靠猜:领导应该是这个意思 / 客户估计要的是那个东西。闷头苦干,然后发现完全不是对方要的东西,遇上脾气暴的主,还会发火训自己。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">小伙伴不免怀疑人生,我这么努力,为什么没人看到?</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">我教各位从一个小技巧入手,每次别人在说的时候,立即<strong>拿笔记下来</strong>;别人说完,你<strong>确认一遍</strong>:「领导您希望我们做的是 ABC 三点,对不对?」,等到领导 / 客户确认后,再接着<strong>邮件补刀</strong>:我们刚刚开会达成三点内容,分别是 ABC。接下来,保持<strong>定期更新</strong>:A 完成了 B 还在做 C 还没开始启动。让对方心里有底。</span></p> <p><br></p> <section class="_135editor" data-tools="135编辑器" data-id="93962" style="border-width: 0px;border-style: none;border-color: initial;box-sizing: border-box;"> <section style="text-align: center;padding-top: 10px;padding-bottom: 10px;box-sizing: border-box;"> <section style="display:flex;justify-content: center;align-items: center;"> <section style="height: 0px;width: 0px;border-left: 10px solid #0f88eb;border-bottom: 18px solid transparent;margin-bottom: -30px;margin-right: -10px;border-top-color: #0f88eb;border-right-color: #0f88eb;transform: rotate(25deg);-webkit-transform: rotate(25deg);-moz-transform: rotate(25deg);-o-transform: rotate(25deg);"></section> <section style="display: inline-block;background: rgb(15, 136, 235);padding-top: 4px;padding-bottom: 4px;color: rgb(255, 255, 255);box-sizing: border-box;"> <section style="font-size: 18px;padding-right: 12px;padding-left: 12px;box-sizing: border-box;" class="autonum" data-original-title="" title="" data-num="7"> 7 </section> </section> <section style="display: inline-block;margin-left: 4px;padding-top: 2px;padding-bottom: 2px;border-bottom: 3px solid rgb(15, 136, 235);border-top-color: rgb(15, 136, 235);border-left-color: rgb(15, 136, 235);border-right-color: rgb(15, 136, 235);box-sizing: border-box;"> <section class="135brush" data-brushtype="text" style="color: rgb(51, 51, 51);padding-right: 4px;padding-left: 4px;font-weight: bold;letter-spacing: 1px;box-sizing: border-box;"> <span style="font-size: 15px;">不要盲目崇拜别人开的挂</span> </section> </section> </section> </section> </section> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">之前有一阵子开始吹开挂的人生,知乎有一个问题:<a class="weapp_text_link" style="font-size:15px;" data-miniprogram-appid="wxeb39b10e39bf6b54" data-miniprogram-path="zhihu/question?id=37106162" data-miniprogram-nickname="知乎热榜" href="">为什么有人的人生跟开了挂一样?</a></span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">的确,这个世界上有不少开挂的主,但和我们有半毛钱关系吗?</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><strong><span style="color: rgb(89, 89, 89);font-size: 15px;">那些开了挂的人,不会告诉你为什么她开挂背后的各种因素,甚至会故意隐藏部分关键信息;更不愿意让吃瓜群众知道,即使再怎么努力,也很难复制他的开挂人生。</span></strong><span style="color: rgb(89, 89, 89);font-size: 15px;">更可怕的是,单纯迷恋别人的开挂,很可能被人收智商税。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">举个例子:</span></p> <p><br></p> <blockquote> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 14px;">罗振宇在多个场合说明拒绝被存量绑架的观点时,最常举的例子是,2013 年,自己卖掉了房子和车。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 14px;">但在今年年初,罗振宇在一场演讲中又说,当时自己确实卖掉了房子,「<strong>但后半句我没有说,我到香港买了腾讯的股票,实际上,比北京房价涨得好,然后,在这一轮房价上涨之前,我又把北京的房子买回来了。</strong>」</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 14px;">由此得出结论:「对创业者来说,最重要一条:甭管听谁忽悠你的,责任你自己负,因为,<strong>别人不会把他选择的所有维度参数都告诉你。」</strong></span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 14px;">提及此事,罗振宇满脸的「我就喜欢你看不惯我又干不掉我的样子」:「我当时一直在说(买股票),只不过没有公开说而已……我就算没有这些,我先卖了,然后我改主意了,买了,不行吗,这不是公民的权利吗?」</span></p> </blockquote> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">还有那位知乎上被热议的 Molly 妹子,一年 300 多天在旅游,成绩从来不耽误。都因为人家一边旅游一边学习,不用睡觉啊,你服不服?</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">你能不睡觉吗?我反正不行,我不睡觉就是一坨行走的五花肉,毫无生气可言。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">我建议各位刚进入社会的小伙伴们:<strong>不要太花精力迷恋别人怎么开的挂,你学不到什么有价值的东西。不如多看失败者的经历,从中吸取教训。</strong></span></p> <p><br></p> <section class="_135editor" data-tools="135编辑器" data-id="93962" style="border-width: 0px;border-style: none;border-color: initial;box-sizing: border-box;"> <section style="text-align: center;padding-top: 10px;padding-bottom: 10px;box-sizing: border-box;"> <section style="display:flex;justify-content: center;align-items: center;"> <section style="height: 0px;width: 0px;border-left: 10px solid #0f88eb;border-bottom: 18px solid transparent;margin-bottom: -30px;margin-right: -10px;border-top-color: #0f88eb;border-right-color: #0f88eb;transform: rotate(25deg);-webkit-transform: rotate(25deg);-moz-transform: rotate(25deg);-o-transform: rotate(25deg);"></section> <section style="display: inline-block;background: rgb(15, 136, 235);padding-top: 4px;padding-bottom: 4px;color: rgb(255, 255, 255);box-sizing: border-box;"> <section style="font-size: 18px;padding-right: 12px;padding-left: 12px;box-sizing: border-box;" class="autonum" data-original-title="" title="" data-num="8"> 8 </section> </section> <section style="display: inline-block;margin-left: 4px;padding-top: 2px;padding-bottom: 2px;border-bottom: 3px solid rgb(15, 136, 235);border-top-color: rgb(15, 136, 235);border-left-color: rgb(15, 136, 235);border-right-color: rgb(15, 136, 235);box-sizing: border-box;"> <section class="135brush" data-brushtype="text" style="color: rgb(51, 51, 51);padding-right: 4px;padding-left: 4px;font-weight: bold;letter-spacing: 1px;box-sizing: border-box;"> <span style="font-size: 15px;">职场上不要全靠自己摸索,要找师傅</span> </section> </section> </section> </section> </section> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">职场上很流行一句毒鸡汤,在问题:<a class="weapp_text_link" style="font-size:15px;" data-miniprogram-appid="wxeb39b10e39bf6b54" data-miniprogram-path="zhihu/answer?id=134724986" data-miniprogram-nickname="知乎热榜" href="">有哪些典型的「学生思维」?</a>问题里我也亲眼目睹过:</span></p> <p><br></p> <blockquote> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 14px;">学生时代有老师,到职场上就要靠自己,别指望有人教你。</span></p> </blockquote> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">咪蒙在《职场不相信眼泪,要哭回家哭》里也说过类似的话:没有任何人有义务来指导你。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">胡说八道。职场上同样需要老师,需要有人教我们,公司有义务指导我们。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">没有老师,我们也许也能摸索出成功的路,但那必然更加艰辛。实际上职场导师,不但对新人有价值,对公司的发展也很有价值:新人迅速融入,企业当然受益。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">企业也没有意识到培训新人,让新人迅速上手的重要性,而有些老员工担心新人抢了饭碗,故意不教新人,反而编出毒鸡汤:职场都别指望有人教你。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">对于这种人和这种企业,职场新人能躲就躲——这是不欢迎新人的文化。你在这里即使能通过百倍努力获得成功,但这种程度的努力可能已经让你在一个友好环境下,取得 3-5 倍的成就了。</span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">新人进入职场后,要以谦虚的态度抓紧时间找自己的导师,<strong>如果找到,别人给了你帮助和提点,记在心里。未来有机会,记得感恩。</strong></span></p> <p><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><span style="color: rgb(89, 89, 89);font-size: 15px;">职场在不断发展,公司和员工的关系也在调整,多年来坚持要求员工着正装上班的 IBM 如今也主动建议员工日常只需休闲装办公。更不要说那些为老板挡酒才能晋升的胡说八道。</span></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><br></p> <p style="line-height: 1.75em;letter-spacing: 0.5px;"><strong><span style="color: rgb(89, 89, 89);font-size: 15px;">戳小程序查看剩下的 2,518 个回答:<a class="weapp_text_link" style="font-size:15px;" data-miniprogram-appid="wxeb39b10e39bf6b54" data-miniprogram-path="zhihu/question?id=64351318" data-miniprogram-nickname="知乎热榜" href="">有哪些大家不说,但需要知道的社会规则?</a></span></strong></p> <p><br></p> <p style="text-align: center;"><img class="" data-copyright="0" data-ratio="0.22476190476190477" data-s="300,640" src="/upload/9ce34a7619143b03522caa3314779f12.png" data-type="png" data-w="525"></p> <p><br></p> <section class="_135editor" data-tools="135编辑器" data-id="93662" style="font-size: medium;white-space: normal;border-width: 0px;border-style: none;border-color: initial;box-sizing: border-box;"> <section style="margin: 5px 5px 15px;"> <section style="margin-bottom: -20px;display: inline-block;background-color: rgba(151, 228, 249, 0.7);width: 30px;height: 30px;transform: rotate(45deg);-webkit-transform: rotate(45deg);-moz-transform: rotate(45deg);-o-transform: rotate(45deg);"></section> <section style="margin-bottom: -20px;margin-left: -3px;display: inline-block;background-color: rgba(175, 178, 249, 0.7);width: 30px;height: 30px;transform: rotate(45deg);-webkit-transform: rotate(45deg);-moz-transform: rotate(45deg);-o-transform: rotate(45deg);"></section> <section data-width="100%" style="padding: 10px;width: 100%;vertical-align: top;border-width: 1px;border-style: solid;border-radius: 5px;border-color: rgb(13, 10, 10);overflow: hidden;box-sizing: border-box;"> <section style="padding: 1em 0.8em;text-align: justify;letter-spacing: 1.5px;color: rgb(63, 63, 63);font-size: 14px;line-height: 1.75em;box-sizing: border-box;"> <section class="135brush"> <p style="font-size: 16px;letter-spacing: 0.5px;line-height: 1.75em;"><span style="font-size: 14px;">我们精选了 16 个和人类恋爱、情感有关的故事。</span></p> <p style="font-size: 16px;letter-spacing: 0.5px;line-height: 1.75em;"><span style="font-size: 14px;">它们不能教你恋爱,也不是所有故事都有美满的结局;</span></p> <p style="font-size: 16px;letter-spacing: 0.5px;line-height: 1.75em;"><span style="font-size: 14px;">看完你会感受到——「啧,感情啊,真是复杂的东西」。</span></p> <p style="font-size: 16px;"><br></p> <p style="font-size: 16px;letter-spacing: 0.5px;line-height: 1.75em;"><span style="font-size: 14px;">不要期待通过它们来理解爱情。</span></p> <p style="font-size: 16px;letter-spacing: 0.5px;line-height: 1.75em;"><span style="font-size: 14px;">恋爱难,开头难、中间难、结尾难。</span></p> <p style="font-size: 16px;letter-spacing: 0.5px;line-height: 1.75em;"><span style="font-size: 14px;">在这里,让我们当一个人间情爱的围观者。</span></p> <p style="letter-spacing: 0.5px;line-height: 1.75em;text-align: center;"><br></p> <p style="letter-spacing: 0.5px;line-height: 1.75em;">识别图中二维码,回复<span style="font-size: 16px;"><strong>「感情」</strong></span>获取</p> </section> </section> </section> <section style="margin-right: 10px;display: flex;justify-content: flex-end;"> <section style="margin-top: -20px;margin-bottom: 10px;line-height: 1;transform: translate3d(10px, 0px, 0px);-webkit-transform: translate3d(10px, 0px, 0px);-moz-transform: translate3d(10px, 0px, 0px);-o-transform: translate3d(10px, 0px, 0px);"> <section style="margin-bottom: -20px;display: inline-block;background-color: rgba(151, 228, 249, 0.7);width: 30px;height: 30px;transform: rotate(45deg);-webkit-transform: rotate(45deg);-moz-transform: rotate(45deg);-o-transform: rotate(45deg);"></section> <section style="margin-bottom: -20px;margin-left: -3px;display: inline-block;background-color: rgba(175, 178, 249, 0.7);width: 30px;height: 30px;transform: rotate(45deg);-webkit-transform: rotate(45deg);-moz-transform: rotate(45deg);-o-transform: rotate(45deg);"></section> </section> </section> </section> </section> <p style="font-size: medium;letter-spacing: 0.5px;line-height: 1.75em;text-align: center;"><br></p> <p style="text-align: center;"><img class="" data-copyright="0" data-ratio="1.509375" data-s="300,640" src="/upload/7d404aba6e7e91cca82706fa62f6a12.jpg" data-type="jpeg" data-w="1280"></p> </section> </section> </section> </section>

一个牛逼的创业公司后台技术栈搭建方案

作者:微信小助手

<section style="box-sizing: border-box;font-size: 16px;"> <section style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="padding-top: 10px;padding-right: 10px;padding-left: 10px;box-sizing: border-box;background-color: rgb(239, 239, 239);"> <span style="display: inline-block;width: 5%;line-height: 0.8;font-weight: bolder;font-size: 48px;box-sizing: border-box;"> <section style="box-sizing: border-box;"> “ </section></span> <section style="display: inline-block;vertical-align: top;float: right;width: 90%;line-height: 1.5;font-size: 15px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><span style="letter-spacing: 1px;">在大多数创业公司,因为没有大公司那些完善的基础设施,需要从开源界的一个个系统和组件做选型,最终形成整个的后台技术栈。</span></p> </section> <section style="clear: both;box-sizing: border-box;"></section> </section> </section> </section> </section> <p style="line-height: 1.75em;"><br></p> <p style="text-align: center;margin-left: 8px;margin-right: 8px;margin-bottom: 5px;"><img class="rich_pages" data-copyright="0" data-ratio="0.5622254758418741" data-s="300,640" src="/upload/ca4c80875b0da40196a44bf0682b0d6.png" data-type="png" data-w="683" style=""></p> <section class="" mpa-from-tpl="t"> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;margin-bottom: 5px;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">说到后台技术栈,脑海中是不是浮现的下面这样一幅图?</span></p> <section mpa-from-tpl="t" mpa-is-content="t"> <p style="text-align: center;margin-left: 8px;margin-right: 8px;line-height: 1.75em;margin-bottom: 5px;"><img class="" data-ratio="0.504424778761062" data-type="png" data-w="904" src="/upload/c9f991515978673c0cf31276bafd8fdb.png" style="top: 0px;left: 0px;right: 0px;bottom: 0px;box-sizing: border-box !important;word-wrap: break-word !important;width: 677px !important;visibility: visible !important;"></p> </section> <p style="text-align: center;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 14px;"><em><span style="color: rgb(89, 89, 89);letter-spacing: 1px;">图 1</span></em></span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">有点眼晕,以下只是我们会用到的一些语言的合集,而且只是语言层面的一部分,就整个后台技术栈来说,这只是一个开始,从语言开始,还有很多很多的内容。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">今天要说的后台是大后台的概念,放在服务器上的东西都属于后台的东西,比如使用的框架,语言,数据库,服务,操作系统等等。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">整个后台技术栈,我的理解包括四个层面的内容:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">语言:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">用了哪些开发语言,如:C++/Java/Go/PHP/Python/Ruby 等等。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">组件:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">用了哪些组件,如:MQ 组件,数据库组件等等。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">流程:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">怎样的流程和规范,如:开发流程,项目流程,发布流程,监控告警流程,代码规范等等。</span></p></li> <li> <section class="" mpa-from-tpl="t"> <p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">系统:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">系统化建设,上面的流程需要有系统来保证,如:规范发布流程的发布系统,代码管理系统等等。</span></p> </section></li> </ul> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;margin-bottom: 5px;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">结合以上的的 4 个层面的内容,整个后台技术栈的结构如图 2 所示:</span></p> <p style="text-align: center;line-height: 1.75em;margin-left: 8px;margin-right: 8px;margin-bottom: 5px;"><img class="" data-ratio="0.5490196078431373" data-type="png" data-w="918" src="/upload/6bf197bb5651536a18f51f48a2aea96.png" style="max-width: 625px;border: 0px;vertical-align: middle;top: 0px;left: 0px;right: 0px;bottom: 0px;box-sizing: border-box !important;word-wrap: break-word !important;width: 677px !important;visibility: visible !important;"></p> <p style="text-align: center;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 14px;"><em><span style="color: rgb(89, 89, 89);letter-spacing: 1px;">图 2:后台技术栈结构</span></em></span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">以上的这些内容都需要我们从零开始搭建,在创业公司,没有大公司那些完善的基础设施,需要我们从开源界,从云服务商甚至有些需要自己去组合,去拼装,去开发一个适合自己的组件或系统以达成我们的目标。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">咱们一个个系统和组件的做选型,最终形成我们的后台技术栈。</span></p> </section> <p style="line-height: normal;"><br></p> <section class="" mpa-from-tpl="t"> <section style="box-sizing: border-box;font-size: 16px;"> <section style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="border-bottom-width: 1px;border-bottom-style: solid;border-bottom-color: black;margin-top: 0.5em;margin-bottom: 0.5em;line-height: 1.2;box-sizing: border-box;"> <section style="display: inline-block;border-bottom-width: 6px;border-bottom-style: solid;border-color: rgb(89, 89, 89);margin-bottom: -1px;font-size: 20px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;">各系统组件选型</p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> </section> <section class="" mpa-from-tpl="t"> <section style="box-sizing: border-box;font-size: 16px;"> <section style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span><span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span></strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>项目管理/Bug 管理/问题管理</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">项目管理软件是整个业务的需求,问题,流程等等的集中地,大家的跨部门沟通协同大多依赖于项目管理工具。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">有一些 SaaS 的项目管理服务可以使用,但是很多时间不满足需求,此时我们可以选择一些开源的项目,这些项目本身有一定的定制能力,有丰富的插件可以使用。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">一般的创业公司需求基本上都能得到满足,常用的项目如下:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Redmine:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">用 Ruby 开发的,有较多的插件可以使用,能自定义字段,集成了项目管理,Bug 问题跟踪,WiKi&nbsp;等功能,不过好多插件 N 年没有更新了。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Phabricator:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">用 PHP 开发的,Facebook 之前的内部工具,开发这工具的哥们离职后自己搞了一个公司专门做这个软件,集成了代码托管, Code Review,任务管理,文档管理,问题跟踪等功能,强烈推荐较敏捷的团队使用。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Jira:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">用 Java 开发的,有用户故事,Task 拆分,燃尽图等等,可以做项目管理,也可以应用于跨部门沟通场景,较强大。</span></p></li> <li> <section class="" mpa-from-tpl="t"> <p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">悟空 CRM :</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">这个不是项目管理,这个是客户管理,之所以在这里提出来,是因为在 To B 的创业公司里面,往往是以客户为核心来做事情的,可以将项目管理和问题跟进的在悟空 CRM 上面来做。</span></p> </section><p><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 1px;line-height: 1.75em;">它的开源版本已经基本实现了 CRM 的核心功能,还带有一个任务管理功能,用于问题跟进,不过用这个的话,还是需要另一个项目管理的软件协助,顺便说一嘴,这个系统的代码写得很难维护,只能适用于客户规模小(1 万以内)时。</span><br></p></li> </ul> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;font-size: 16px;"> <section style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span><span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span></strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>DNS</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">DNS 是一个很通用的服务,创业公司基本上选择一个合适的云厂商就行了,</span><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">国内主要是两家:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">阿里万网:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">阿里 2014 年收购了万网,整合了其域名服务,最终形成了现在的阿里万网,其中就包含 DNS 这块的服务。</span></p></li> <li> <section class="" mpa-from-tpl="t"> <p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">腾讯 DNSPod:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">腾讯 2012 年以 4000 万收购 DNSPod 100% 股份,主要提供域名解析和一些防护功能。</span></p> </section></li> </ul> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">如果你的</span><span style="font-size: 15px;letter-spacing: 1px;color: rgb(89, 89, 89);">业务是在国内</span><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">,主要就是这两家,选 一个就好,像今日头条这样的企业用的也是 DNSPod 的服务,除非一些特殊的原因才需要自建,比如一些 CDN 厂商,或者对区域有特殊限制的。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">要实惠一点用阿里最便宜的基础版就好了,要成功率高一些,还是用 DNSPod 的贵的那种。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">在国外还是选择亚马逊吧,阿里的 DNS 服务只有在日本和美国有节点,东南亚最近才开始部点,DNSPod 也只有美国和日本,像一些出海的企业,其选择的云服务基本都是亚马逊。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">如果是线上产品,DNS 强烈建议用付费版,阿里的那几十块钱的付费版基本可以满足需求。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">如果还需要一些按省份或按区域调试的逻辑,则需要加钱,一年也就几百块,省钱省力。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">如果是国外,优先选择亚马逊,如果需要国内外互通并且有自己的 App</span><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 1px;line-height: 1.75em;">&nbsp;的话,建议还是自己实现一些容灾逻辑或者智能调度。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 1px;line-height: 1.75em;">因为没有一个现成的 DNS 服务能同时较好的满足国内外场景,或者用多个域名,不同的域名走不同的 DNS 。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;font-size: 16px;"> <section style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span><span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span></strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>LB(负载均衡)</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">LB(负载均衡)是一个通用服务,</span><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">一般云厂商的 LB 服务基本都有如下功能:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">支持四层协议请求(包括 TCP、UDP 协议)</span></strong></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">支持七层协议请求(包括 HTTP、HTTPS 协议)</span></strong></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">集中化的证书管理系统支持 HTTPS 协议</span></strong></p></li> <li> <section class="" mpa-from-tpl="t"> <p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">健康检查</span></strong></p> </section></li> </ul> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">如果你线上的服务机器都是用的云服务,并且是在同一个云服务商的话,可以直接使用云服务商提供的 LB 服务,如阿里云的 SLB,腾讯云的 CLB,亚马逊的 ELB 等等。如果是自建机房基本都是 LVS + Nginx。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;font-size: 16px;"> <section style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span><span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span></strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>CDN</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">CDN 现在已经是一个很红很红的市场,基本上只能挣一些辛苦钱,都是贴着成本在卖。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">国内以网宿为龙头,他们家占据整个国内市场份额的 40% 以上,后面就是腾讯,阿里。网宿有很大一部分是因为直播的兴起而崛起。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">国外,Amazon 和 Akamai 合起来占比大概在 50%,曾经的国际市场老大 Akamai 拥有全球超一半的份额,在 Amazon CDN入局后,份额跌去了将近 20%,众多中小企业都转向后者,Akamai 也是无能为力。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">国内出海的 CDN 厂商,更多的是为国内的出海企业服务,三家大一点的 CDN 服务商里面也就网宿的节点多一些,但是也多不了多少。阿里和腾讯还处于前期阶段,仅少部分国家有节点。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">就创业公司来说,CDN 用腾讯云或阿里云即可,其相关系统较完善,能轻松接入,网宿在系统支持层面相对较弱一些,而且还贵一些。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">并且,当流量上来后,CDN 不能只用一家,需要用多家,不同的 CDN 在全国的节点覆盖不一样。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">而且针对不同的客户云厂商内部有些区分客户集群,并不是全节点覆盖(但有些云厂商说自己是全网节点),除了节点覆盖的问题,多 CDN 也在一定程度上起到容灾的作用。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;font-size: 16px;"> <section style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span><span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span></strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>RPC 框架</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">维基百科对 RPC 的定义是:远程过程调用(Remote Procedure Call,RPC)是一个计算机通信协议。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">该协议允许运行于一台计算机的程序调用另一台计算机的子程序,而程序员无需额外地为这个交互作用编程。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">通俗来讲,一个完整的 RPC 调用过程,就是 Server 端实现了一个函数,客户端使用 RPC 框架提供的接口,调用这个函数的实现,并获取返回值的过程。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">业界 RPC 框架大致分为两大流派,一种侧重跨语言调用,另一种是偏重服务治理。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">跨语言调用型的 RPC 框架有 Thrift、gRPC、Hessian、Hprose 等。这类 RPC 框架侧重于服务的跨语言调用,能够支持大部分的语言进行语言无关的调用,非常适合多语言调用场景。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">但这类框架没有服务发现相关机制,实际使用时需要代理层进行请求转发和负载均衡策略控制。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">其中,gRPC 是 Google 开发的高性能、通用的开源 RPC 框架,其由 Google 主要面向移动应用开发并基于 HTTP/2 协议标准而设计,基于 ProtoBuf(Protocol Buffers)序列化协议开发,且支持众多开发语言。本身它不是分布式的,所以要实现框架的功能需要进一步的开发。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Hprose(High Performance Remote Object Service Engine)是一个 MIT 开源许可的新型轻量级跨语言跨平台的面向对象的高性能远程动态通讯中间件。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">服务治理型的 RPC 框架的特点是功能丰富,提供高性能的远程调用、服务发现及服务治理能力,适用于大型服务的服务解耦及服务治理,对于特定语言(Java)的项目可以实现透明化接入。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">缺点是语言耦合度较高,跨语言支持难度较大。</span><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">国内常见的冶理型 RPC 框架如下:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Dubbo:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Dubbo 是阿里巴巴公司开源的一个 Java 高性能优秀的服务框架,使得应用可通过高性能的 RPC 实现服务的输出和输入功能,可以和 Spring 框架无缝集成。</span></p><p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">当年在淘宝内部,Dubbo 由于跟淘宝另一个类似的框架 HSF 有竞争关系,导致 Dubbo 团队解散,最近又活过来了,有专职同学投入。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">DubboX:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">DubboX 是由当当在基于 Dubbo 框架扩展的一个 RPC 框架,支持 REST 风格的远程调用、Kryo/FST 序列化,增加了一些新的 feature。</span></p><p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Motan:Motan 是新浪微博开源的一个 Java 框架。它诞生的比较晚,起于 2013 年,2016 年 5 月开源。Motan 在微博平台中已经广泛应用,每天为数百个服务完成近千亿次的调用。</span></p></li> <li> <section class="" mpa-from-tpl="t"> <p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">RPCX:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">RPCX 是一个类似阿里巴巴 Dubbo 和微博 Motan 的分布式的 RPC 服务框架,基于 Golang NET/RPC 实现。</span></p> </section><p><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 1px;line-height: 1.75em;">但是 <span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 1px;line-height: 29.75px;">RPCX</span> 基本只有一个人在维护,没有完善的社区,使用前要慎重,之前做 Golang 的 RPC 选型时也有考虑这个,最终还是放弃了,选择了 gRPC,如果想自己自研一个 RPC 框架,可以参考学习一下。</span><br></p></li> </ul> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;font-size: 16px;"> <section style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span></strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>名字发现/服务发现</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">名字发现和服务发现分为两种模式,一个是客户端发现模式,一种是服务端发现模式。</span><span style="line-height: 1.75em;font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">框架中常用的服务发现是客户端发现模式。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="color: rgb(89, 89, 89);font-size: 15px;letter-spacing: 1px;line-height: 1.75em;">所谓服务端发现模式是指客户端通过一个负载均衡器向服务发送请求,负载均衡器查询服务注册表并把请求路由到一台可用的服务实例上。现在常用的负载均衡器都是此类模式,常用于微服务中。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">所有的名字发现和服务发现都要依赖于一个可用性非常高的服务注册表,</span><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">业界常用的服务注册表有如下三个:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Etcd:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">一个高可用、分布式、一致性、Key-Value 方式的存储,被用在分享配置和服务发现中。两个著名的项目使用了它:Kubernetes 和 Cloud Foundry。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Consul:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">一个发现和配置服务的工具,为客户端注册和发现服务提供了API,Consul 还可以通过执行健康检查决定服务的可用性。</span></p></li> <li> <section class="" mpa-from-tpl="t"> <p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Apache ZooKeeper:</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">一个广泛使用、高性能的针对分布式应用的协调服务。Apache ZooKeeper 本来是 Hadoop 的子工程,现在已经是顶级工程了。</span></p> </section></li> </ul> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">除此之外也可以自己实现服务实现,或者用 Redis 也行,只是需要自己实现高可用性。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;font-size: 16px;"> <section style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span></strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>关系数据库</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">关系数据库分为两种,一种是传统关系数据库,如 Oracle,MySQL,Maria,DB2,PostgreSQL 等等。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="color: rgb(71, 193, 168);font-size: 15px;letter-spacing: 1px;">另一种是 NewSQL,即至少要满足以下五点的新型关系数据库:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">完整地支持 SQL,支持 JOIN / GROUP BY /子查询等复杂 SQL 查询。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">支持传统数据标配的 ACID 事务,支持强隔离级别。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">具有弹性伸缩的能力,扩容缩容对于业务层完全透明。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">真正的高可用,异地多活、故障恢复的过程不需要人为的接入,系统能够自动地容灾和进行强一致的数据恢复。</span></p></li> <li> <section class="" mpa-from-tpl="t"> <p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">具备一定的大数据分析能力。</span></p> </section></li> </ul> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">传统关系数据库用得最多的是 MySQL,因为成熟,稳定,一些基本的需求都能满足,在一定数据量级之前基本单机传统数据库都可以搞定。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">而且现在较多的开源系统都是基于 MySQL,开箱即用,再加上主从同步和前端缓存,百万 PV 的应用都可以搞定了。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">不过 CentOS 7 已经放弃了 MySQL,而改使用 MariaDB。MariaDB 数据库管理系统是 MySQL 的一个分支,主要由开源社区在维护,采用 GPL 授权许可。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">开发这个分支的原因之一是:甲骨文公司收购了 MySQL 后,有将 MySQL 闭源的潜在风险,因此社区采用分支的方式来避开这个风险。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">在 Google 发布了 F1: A Distributed SQL Database That Scales 和 Spanner: Google’s Globally-Distributed Databasa 之后,业界开始流行起 NewSQL。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">于是有了 CockroachDB,然后有了奇叔公司的 TiDB。国内已经有比较多的公司使用 TiDB,之前在创业公司时在大数据分析时已经开始应用 TiDB,当时应用的主要原因是 MySQL 要使用分库分表,逻辑开发比较复杂,扩展性不够。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;font-size: 16px;"> <section style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span><span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span></strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>NoSQL</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">NoSQL 顾名思义就是 Not-Only SQL,也有人说是 No–SQL,个人偏向于 Not-Only SQL,它并不是用来替代关系库,而是作为关系型数据库的补充而存在。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">常见 NoSQL 有四个类型:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">键值,</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">适用于内容缓存,适合混合工作负载并发高扩展要求大的数据集,其优点是简单,查询速度快,缺点是缺少结构化数据,常见的有 Redis,Memcache,BerkeleyDB 和 Voldemort 等等。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">列式,</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">以列簇式存储,将同一列数据存在一起,常见于分布式的文件系统,其中以 Hbase,Cassandra 为代表。</span></p><p style="text-align: justify;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">Cassandra 多用于写多读少的场景,国内用得比较多的有 360,大概 1500 台机器的集群,国外大规模使用的公司比较多,如 eBay,Instagram,Apple 和沃尔玛等等。</span></p></li> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">文档,</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">数据存储方案非常适用承载大量不相关且结构差别很大的复杂信息。性能介于 KV 和关系数据库之间,它的灵感来自&nbsp;Lotus Notes,常见的有 MongoDB,CouchDB 等等。</span></p></li> <li> <section class="" mpa-from-tpl="t"> <p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">图形,</span></strong><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">图形数据库擅长处理任何涉及关系的状况,比如社交网络,推荐系统等。专注于构建关系图谱,需要对整个图做计算才能得出结果,不容易做分布式的集群方案,常见的有 Neo4J,InfoGrid 等。</span></p> </section></li> </ul> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">除了以上 4 种类型,还有一些特种的数据库,如对象数据库,XML 数据库,这些都有针对性对某些存储类型做了优化的数据库。</span></p> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">在实际应用场景中,何时使用关系数据库,何时使用 NoSQL,使用哪种类型的数据库,这是我们在做架构选型时一个非常重要的考量,甚至会影响整个架构的方案。</span></p> <p style="line-height: normal;"><br></p> <section style="box-sizing: border-box;font-size: 16px;"> <section style="box-sizing: border-box;" powered-by="xiumi.us"> <section style="margin-top: 10px;margin-bottom: 10px;box-sizing: border-box;"> <section style="width: 0.6em;display: inline-block;vertical-align: middle;box-sizing: border-box;"> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.2;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <span style="width: 0.6em;height: 0.6em;display: block;opacity: 0.6;margin-top: 2px;margin-bottom: 2px;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span> <strong><span style="width: 0.6em;height: 0.6em;display: block;opacity: 1;box-sizing: border-box;background-color: rgb(89, 89, 89);"></span></strong> </section> <section style="display: inline-block;vertical-align: middle;font-size: 18px;padding-left: 5px;color: rgb(89, 89, 89);box-sizing: border-box;"> <p style="box-sizing: border-box;"><strong>消息中间件</strong></p> </section> </section> </section> </section> <p style="line-height: normal;"><br></p> <p style="text-align: justify;margin-left: 8px;margin-right: 8px;line-height: 1.75em;"><span style="font-size: 15px;color: rgb(89, 89, 89);letter-spacing: 1px;">消息中间件在后台系统中是必不可少的一个组件,</span><span style="font-size: 15px;letter-spacing: 1px;color: rgb(71, 193, 168);">一般我们会在以下场景中使用消息中间件:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p style="text-align: justify;line-height: 1.75em;"><strong><span style="font-size: 15px;color: rgb(89, 89,

Google MapReduce架构设计

作者:微信小助手

<p><strong><span style="font-size: 14px;letter-spacing: 1px;">前情回顾</span></strong></p> <p><span style="font-size: 14px;letter-spacing: 1px;"><a href="http://mp.weixin.qq.com/s?__biz=MjM5ODYxMDA5OQ==&amp;mid=2651961826&amp;idx=1&amp;sn=98f6ec5271c7a4153c3e50c7b8be4f33&amp;chksm=bd2d0c3e8a5a8528177b34db26876e510486c78f331bd5e51082a22eda9d65b9bfc35abe1c53&amp;scene=21#wechat_redirect" target="_blank">Google MapReduce到底解决什么问题?</a></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">Google MapReduce是Google产出的一个编程模型,同时Google也给出架构实现,它能够解决“能用</span><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">分治法</span><span style="font-size: 14px;letter-spacing: 1px;">解决的问题”。 </span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;"><a href="http://mp.weixin.qq.com/s?__biz=MjM5ODYxMDA5OQ==&amp;mid=2651961851&amp;idx=1&amp;sn=56e85dc9bd82d48577e487f42876748d&amp;chksm=bd2d0c278a5a85313431f6ad7ba47f1721f6d5823e4c072d77cdc9fe18bac53dc1a1e8a6fc3a&amp;scene=21#wechat_redirect" target="_blank">Google MapReduce有啥巧妙优化?</a></span></p> <p style="text-align: center;"><img class="" data-copyright="0" data-ratio="0.37316176470588236" data-s="300,640" src="/upload/451f94289123b0e721ccc4669309dfbd.png" data-type="png" data-w="544" style=""></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><strong><span style="font-size: 14px;letter-spacing: 1px;">分区函数</span></strong><span style="font-size: 14px;letter-spacing: 1px;">:保证不同map输出的相同key,落到同一个reduce里</span></p></li> <li><p><strong><span style="font-size: 14px;letter-spacing: 1px;">合并函数</span></strong><span style="font-size: 14px;letter-spacing: 1px;">:在map结束时,对相同key的多个输出做本地合并,节省总体资源</span></p></li> <li><p><strong><span style="font-size: 14px;letter-spacing: 1px;">输入文件到map如何切分</span></strong><span style="font-size: 14px;letter-spacing: 1px;">:随意,切分均匀就行</span></p></li> </ul> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">画外音:看懂了这个流程,对工程架构的理解,会容易很多。</span></em></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><strong><span style="font-size: 14px;letter-spacing: 1px;">上述执行流程,Google MapReduce通过怎样的工程架构实现的呢?</span></strong></p> <p style="text-align: center;"><img class="" data-copyright="0" data-ratio="0.7485604606525912" data-s="300,640" src="/upload/73ce0d1630c8a47434e4aad53fac82f3.png" data-type="png" data-w="521" style=""></p> <p><span style="font-size: 14px;letter-spacing: 1px;">先看下总体架构图,有个直观的印象。</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><strong><span style="font-size: 14px;letter-spacing: 1px;">用户使用GoogleMR系统,必须输入的是什么?</span></strong></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">输入数据</span><span style="font-size: 14px;letter-spacing: 1px;">,必选</span></p></li> </ul> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">画外音:否则系统处理啥。</span></em></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="color: rgb(255, 76, 0);font-size: 14px;letter-spacing: 1px;">map函数,必选</span></p></li> <li><p><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">reduce函数</span><span style="font-size: 14px;letter-spacing: 1px;">,必选</span></p></li> </ul> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">画外音:分治法,分与合的业务逻辑。</span></em></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">分区函数</span><span style="font-size: 14px;letter-spacing: 1px;">,必选</span></p></li> </ul> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">画外音:保证同一个key,在合并阶段,必须落到同一个reduce上,系统<span style="font-size: 14px;letter-spacing: 1px;">提供默认hash(key)法。</span></span></em></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">合并函数</span><span style="font-size: 14px;letter-spacing: 1px;">,可选</span></p></li> </ul> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">画外音:看用户是否需要在map结束阶段进行优化。</span></em></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><strong><span style="font-size: 14px;letter-spacing: 1px;">用户提供各个输入后,GoogleMR的执行流程是什么?</span></strong></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">画外音:</span></em></span></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">不妨假设,用户设置了M个map节点,R个reduce节点;</span></em></span><em style="color: rgb(0, 82, 255);"><span style="font-size: 14px;letter-spacing: 1px;">例如:M=500,R=200</span></em></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">(1) 在集群中创建大量可执行实例副本(fork);</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;"><br></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">(2) 这些副本中</span><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">有一个master</span><span style="font-size: 14px;letter-spacing: 1px;">,</span><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">其他均为worker</span><span style="font-size: 14px;letter-spacing: 1px;">,任务的分配由master完成, M个map实例和R个reduce实例由worker完成;</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;"><br></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">(3) 将输入数据分成M份,然后被分配到map任务的worker,从其中一份读取输入数据,执行用户的map函数处理,并在<strong>本地内存</strong>生成临时数据;</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;"><br></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">(4) 本地内存临时数据,通过分区函数,被分成R份,周期性的写到<strong>本地磁盘</strong>,由master调度,传给被分配到reduce任务的worker;</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;"><br></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">(5) 负责reduce任务的worker,从远程读取多个map输出的数据,执行用户的reduce函数处理,处理结果写入输出文件;</span></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">画外音:可能对key要进行外部排序。</span></em></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;"><br></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">(6) 所有map和reduce的worker都结束工作后,master唤醒用户程序,MapReduce调用返回,结果被输出到了R个文件中。</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><strong><span style="font-size: 14px;letter-spacing: 1px;">GoogleMR系统里的master和worker是啥?</span></strong></p> <p><span style="font-size: 14px;letter-spacing: 1px;">(1) <strong>master</strong>:单点master会存储一些元数据,监控所有map与reduce的状态,记录哪个数据要给哪个map,哪个数据要给哪个reduce,掌控全局视野,做中控;</span></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">画外音:是不是和GFS的master非常像?</span></em></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">(2) <strong>worker</strong>:多个worker进行业务逻辑处理,具体一个worker是用来执行map还是reduce,是由master调度的;</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;color: rgb(0, 82, 255);">画外音:是不是和工作线程池非常像?这里的worker是分布在多台机器上的而已。</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><strong><span style="font-size: 14px;letter-spacing: 1px;">master的高可用是如何保证的?</span></strong></p> <p><span style="font-size: 14px;letter-spacing: 1px;">一个简单的方法是,将元数据固化到磁盘上,用一个shadow-master来做高可用。</span></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">画外音:GFS不就是这么干的么?</span></em></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">然而现实情况是:</span><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">没有将元数据固化到磁盘上</span><span style="font-size: 14px;letter-spacing: 1px;">,元数据被存放在master的内存里用以提高工作效率,</span><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">当master挂掉后,通知用户“任务执行失败”</span><span style="font-size: 14px;letter-spacing: 1px;">,让其选择重新执行。</span></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">画外音:</span></em></span></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="color: rgb(0, 82, 255);font-size: 14px;letter-spacing: 1px;">(1) 单点master,掌控全局视野,能让系统的复杂性降低非常多;</span></em></span></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="color: rgb(0, 82, 255);font-size: 14px;letter-spacing: 1px;">(2) master挂掉的概率很小;</span></em></span></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="color: rgb(0, 82, 255);font-size: 14px;letter-spacing: 1px;">(3) 不做高可用,能让系统的复杂性降低非常多;</span></em></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><strong><span style="font-size: 14px;letter-spacing: 1px;">worker的高可用是如何保证的?</span></strong></p> <p><span style="font-size: 14px;letter-spacing: 1px;">master会周期性的ping每个worker,如果超时未返回,master会把对应的worker置为无效,把这个worker的工作任务重新执行:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;letter-spacing: 1px;">如果重新执行的是reduce任务,不需要有额外的通知</span></p></li> <li><p><span style="font-size: 14px;letter-spacing: 1px;">如果重新执行的是map任务,需要通知执行reduce的worker节点,输入数据换了一个worker</span></p></li> </ul> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><strong><span style="font-size: 14px;letter-spacing: 1px;">随时都可能有map或者reduce挂掉,任务完成前重新被执行,会不会影响MR的最终结果?</span></strong></p> <p><span style="font-size: 14px;letter-spacing: 1px;">在用户输入不变的情况下,MR的输出一定是不变的,这就要求MR系统必须具备</span><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">幂等性</span><span style="font-size: 14px;letter-spacing: 1px;">:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;letter-spacing: 1px;">对相同的输入,不管哪个</span><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">负责map的worker执行的结果,一定是不变的</span><span style="font-size: 14px;letter-spacing: 1px;">,产出的R个本地输出文件内容也一定是不变的</span></p></li> <li><p><span style="font-size: 14px;letter-spacing: 1px;">对于M个map,每个map输出的R个本地文件,只要这些输入不变,对应</span><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">接收这些数据的reduce的worker执行结果,一定是不变的</span><span style="font-size: 14px;letter-spacing: 1px;">,输出文件内容也也一定是不变的</span></p></li> </ul> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><strong><span style="font-size: 14px;letter-spacing: 1px;">长尾效应怎么解决?</span></strong></p> <p><span style="font-size: 14px;letter-spacing: 1px;">一个MR执行时间的最大短板,往往是“长尾worker”。</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">导致“长尾worker”的原因有很多:</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">(1) 用户的分区函数设计得不合理,导致某些reduce负载不均,要处理大量的数据;</span></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="font-size: 14px;letter-spacing: 1px;">画外音:</span></em></span></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="color: rgb(0, 82, 255);font-size: 14px;letter-spacing: 1px;">最坏的情况,所有数据最终都落到一个reduce上,分布式并行处理,转变为了单机串行处理;</span></em></span></p> <p><span style="color: rgb(0, 82, 255);"><em><span style="color: rgb(0, 82, 255);font-size: 14px;letter-spacing: 1px;">所以,分区函数的负载均衡性,是用户需要考虑的。</span></em></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">(2) 因为系统的原因,worker所在的机器磁盘坏了,CPU有问题,也可能导致任务执行很慢;</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">GoogleMR有一个“</span><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">备用worker</span><span style="font-size: 14px;letter-spacing: 1px;">”的机制,当某些worker的执行时间超出预期时,会启动另一个worker执行相同的任务,以尝试解决长尾效应。</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;"><br></span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">总结</span></p> <p><span style="font-size: 14px;letter-spacing: 1px;">Google MapReduce架构,提现了很多经典架构实践:</span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">单点master</span><span style="font-size: 14px;letter-spacing: 1px;">简化系统复杂度</span></p></li> <li><p><span style="font-size: 14px;letter-spacing: 1px;">单点</span><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">master不高可用</span><span style="font-size: 14px;letter-spacing: 1px;">,简化系统复杂度</span></p></li> <li><p><span style="font-size: 14px;letter-spacing: 1px;">master对worker的监控以及重启,保证</span><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">worker高可用</span></p></li> <li><p><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">幂等性</span><span style="font-size: 14px;letter-spacing: 1px;">,保证结果的正确性</span></p></li> <li><p><span style="font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">多个worker执行同一个任务</span><span style="font-size: 14px;letter-spacing: 1px;">优化长尾问题</span></p></li> </ul> <p><span style="font-size: 14px;letter-spacing: 1px;"><br></span></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 17px;letter-spacing: 0.544px;white-space: normal;background-color: rgb(255, 255, 255);text-align: center;line-height: normal;box-sizing: border-box !important;word-wrap: break-word !important;"><img width="auto" data-ratio="1" data-w="250" data-type="jpeg" data-s="300,640" class="" src="/upload/7ddc9700032e2c5cee163f1f1a37b46c.jpg" style="max-width: 677px;box-sizing: border-box;word-wrap: break-word;visibility: visible !important;width: 130px !important;"></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 17px;letter-spacing: 0.544px;white-space: normal;background-color: rgb(255, 255, 255);text-align: center;line-height: normal;box-sizing: border-box !important;word-wrap: break-word !important;"><span style="max-width: 100%;letter-spacing: 1px;box-sizing: border-box !important;word-wrap: break-word !important;"><span style="max-width: 100%;font-size: 12px;box-sizing: border-box !important;word-wrap: break-word !important;">架构师之路-分享</span><span style="max-width: 100%;font-size: 12px;color: rgb(255, 76, 0);box-sizing: border-box !important;word-wrap: break-word !important;">可落地</span><span style="max-width: 100%;font-size: 12px;box-sizing: border-box !important;word-wrap: break-word !important;">的技术文章</span></span></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 17px;letter-spacing: 0.544px;text-align: justify;white-space: normal;background-color: rgb(255, 255, 255);box-sizing: border-box !important;word-wrap: break-word !important;"><span style="max-width: 100%;font-size: 14px;letter-spacing: 1px;box-sizing: border-box !important;word-wrap: break-word !important;"><br></span></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 17px;letter-spacing: 0.544px;text-align: justify;white-space: normal;background-color: rgb(255, 255, 255);box-sizing: border-box !important;word-wrap: break-word !important;"><span style="max-width: 100%;font-size: 14px;letter-spacing: 1px;box-sizing: border-box !important;word-wrap: break-word !important;">相关推荐:</span><br style="max-width: 100%;box-sizing: border-box !important;word-wrap: break-word !important;"></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 17px;letter-spacing: 0.544px;text-align: justify;white-space: normal;background-color: rgb(255, 255, 255);box-sizing: border-box !important;word-wrap: break-word !important;"><span style="max-width: 100%;font-size: 14px;letter-spacing: 1px;box-sizing: border-box !important;word-wrap: break-word !important;">《<a href="http://mp.weixin.qq.com/s?__biz=MjM5ODYxMDA5OQ==&amp;mid=2651961812&amp;idx=1&amp;sn=592e3cc722bdfc4201d07cb1b087ed06&amp;chksm=bd2d0c088a5a851eec1e01cea1d68a365edab46bc6985c469adb4b0df17f0e6c2f07bda8da47&amp;scene=21#wechat_redirect" target="_blank" style="color: rgb(87, 107, 149);-webkit-tap-highlight-color: rgba(0, 0, 0, 0);max-width: 100%;box-sizing: border-box !important;word-wrap: break-word !important;">GFS架构启示</a>》</span></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 17px;letter-spacing: 0.544px;text-align: justify;white-space: normal;background-color: rgb(255, 255, 255);box-sizing: border-box !important;word-wrap: break-word !important;"><span style="max-width: 100%;font-size: 14px;letter-spacing: 1px;box-sizing: border-box !important;word-wrap: break-word !important;">《<a href="http://mp.weixin.qq.com/s?__biz=MjM5ODYxMDA5OQ==&amp;mid=2651961826&amp;idx=1&amp;sn=98f6ec5271c7a4153c3e50c7b8be4f33&amp;chksm=bd2d0c3e8a5a8528177b34db26876e510486c78f331bd5e51082a22eda9d65b9bfc35abe1c53&amp;scene=21#wechat_redirect" target="_blank" style="color: rgb(87, 107, 149);-webkit-tap-highlight-color: rgba(0, 0, 0, 0);max-width: 100%;box-sizing: border-box !important;word-wrap: break-word !important;">Google MapReduce解决什么问题?</a>》</span></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 17px;letter-spacing: 0.544px;text-align: justify;white-space: normal;background-color: rgb(255, 255, 255);box-sizing: border-box !important;word-wrap: break-word !important;"><span style="max-width: 100%;font-size: 14px;letter-spacing: 1px;box-sizing: border-box !important;word-wrap: break-word !important;">《<a href="http://mp.weixin.qq.com/s?__biz=MjM5ODYxMDA5OQ==&amp;mid=2651961851&amp;idx=1&amp;sn=56e85dc9bd82d48577e487f42876748d&amp;chksm=bd2d0c278a5a85313431f6ad7ba47f1721f6d5823e4c072d77cdc9fe18bac53dc1a1e8a6fc3a&amp;scene=21#wechat_redirect" target="_blank" style="color: rgb(87, 107, 149);-webkit-tap-highlight-color: rgba(0, 0, 0, 0);max-width: 100%;box-sizing: border-box !important;word-wrap: break-word !important;">Google MapReduce巧妙优化思路?</a>》</span></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-size: 17px;letter-spacing: 0.544px;text-align: justify;white-space: normal;background-color: rgb(255, 255, 255);box-sizing: border-box !important;word-wrap: break-word !important;"><span style="font-size: 14px;letter-spacing: 1px;"><span style="font-family:-apple-system-font, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Hiragino Sans GB, Microsoft YaHei UI, Microsoft YaHei, Arial, sans-serif;">《<a href="http://mp.weixin.qq.com/s?__biz=MjM5ODYxMDA5OQ==&amp;mid=2651961863&amp;idx=2&amp;sn=5060b9971b4d1a702747a01ed553c674&amp;chksm=bd2d0fdb8a5a86cde2e29f9ed8ad6b689974ad6f48f900e96aae9b5887dfe97a47f356abf784&amp;scene=21#wechat_redirect" target="_blank">过载保护+异构服务器的负载均衡</a>》</span></span><span style="font-family: &quot;Helvetica Neue&quot;, Helvetica, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;">&nbsp;</span></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-size: 17px;letter-spacing: 0.544px;text-align: justify;white-space: normal;background-color: rgb(255, 255, 255);box-sizing: border-box !important;word-wrap: break-word !important;"><span style="font-family: &quot;Helvetica Neue&quot;, Helvetica, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;"><span style="color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;text-align: justify;background-color: rgb(255, 255, 255);">《</span><a href="http://mp.weixin.qq.com/s?__biz=MjM5ODYxMDA5OQ==&amp;mid=2651961856&amp;idx=2&amp;sn=fc7fc8cbe7b76029dadcc0f717c65298&amp;chksm=bd2d0fdc8a5a86cae199e80018cbe847afd14985dddaa172b6aca9da4de026789f8162aa2964&amp;scene=21#wechat_redirect" target="_blank" style="color: rgb(87, 107, 149);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;text-align: justify;white-space: normal;background-color: rgb(255, 255, 255);-webkit-tap-highlight-color: rgba(0, 0, 0, 0);max-width: 100%;box-sizing: border-box !important;word-wrap: break-word !important;">程序员养女儿的四大要点!</a><span style="color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;text-align: justify;background-color: rgb(255, 255, 255);">》</span></span></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-size: 17px;letter-spacing: 0.544px;text-align: justify;white-space: normal;background-color: rgb(255, 255, 255);box-sizing: border-box !important;word-wrap: break-word !important;"><span style="font-family: &quot;Helvetica Neue&quot;, Helvetica, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;"><span style="color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;text-align: justify;background-color: rgb(255, 255, 255);"><br></span></span></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-size: 17px;letter-spacing: 0.544px;text-align: justify;white-space: normal;background-color: rgb(255, 255, 255);box-sizing: border-box !important;word-wrap: break-word !important;"><strong><span style="font-family: &quot;Helvetica Neue&quot;, Helvetica, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;"><span style="color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;text-align: justify;background-color: rgb(255, 255, 255);">作业</span></span></strong><span style="font-family: &quot;Helvetica Neue&quot;, Helvetica, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;"><span style="color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;text-align: justify;background-color: rgb(255, 255, 255);">:</span></span></p> <p style="max-width: 100%;min-height: 1em;color: rgb(51, 51, 51);font-size: 17px;letter-spacing: 0.544px;text-align: justify;white-space: normal;background-color: rgb(255, 255, 255);box-sizing: border-box !important;word-wrap: break-word !important;"><span style="color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;text-align: justify;background-color: rgb(255, 255, 255);">MR中有</span><span style="text-align: justify;background-color: rgb(255, 255, 255);font-family: &quot;Helvetica Neue&quot;, Helvetica, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;color: rgb(255, 76, 0);">大量的数据进行传输</span><span style="color: rgb(51, 51, 51);font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;font-size: 14px;letter-spacing: 1px;text-align: justify;background-color: rgb(255, 255, 255);">,如何进行优化呢?</span></p>

Java设计模式(转)——9.外观模式

作者:じ☆ve宝贝

9.外观模式(Facade) 外观模式是为了解决类与类之间的依赖关系的,像spring一样,可以将类和类之间的关系配置到配置文件中,而外观模式就是将他们的关系放在一个Facade类中,降低了类类之间的耦合度,该模式中没有涉及到接口,看下类图:(我们以一个计算机的启动过程为例) ![外观设计模式](/upload/content171.png "外观设计模式") 我们先看下实现类: ``` public class CPU { public void startup(){ System.out.println("cpu startup!"); } public void shutdown(){ System.out.println("cpu shutdown!"); } } ``` ``` public class Memory { public void startup(){ System.out.println("memory startup!"); } public void shutdown(){ System.out.println("memory shutdown!"); } } ``` ``` public class Disk { public void startup(){ System.out.println("disk startup!"); } public void shutdown(){ System.out.println("disk shutdown!"); } } ``` ``` public class Computer { private CPU cpu; private Memory memory; private Disk disk; public Computer(){ cpu = new CPU(); memory = new Memory(); disk = new Disk(); } public void startup(){ System.out.println("start the computer!"); cpu.startup(); memory.startup(); disk.startup(); System.out.println("start computer finished!"); } public void shutdown(){ System.out.println("begin to close the computer!"); cpu.shutdown(); memory.shutdown(); disk.shutdown(); System.out.println("computer closed!"); } } ``` User类如下: ``` public class User { public static void main(String[] args) { Computer computer = new Computer(); computer.startup(); computer.shutdown(); } } ``` 输出: start the computer! cpu startup! memory startup! disk startup! start computer finished! begin to close the computer! cpu shutdown! memory shutdown! disk shutdown! computer closed! 如果我们没有Computer类,那么,CPU、Memory、Disk他们之间将会相互持有实例,产生关系,这样会造成严重的依赖,修改一个类,可能会带来其他类的修改,这不是我们想要看到的,有了Computer类,他们之间的关系被放在了Computer类里,这样就起到了解耦的作用,这就是外观模式!

Hadoop+Hbase 分布式集群架构“完全篇”

作者:微信小助手

<p><strong style="color: rgb(61, 167, 66);font-size: 16px;"><span style="font-size: 14px;">1. 认识 Hadoop 和 Hbase</span></strong><br></p> <h2><span style="font-size: 14px;"></span></h2> <h3><span style="font-size: 14px;"><br></span></h3> <h3><strong><span style="font-size: 14px;">1.1 Hadoop 简单介绍</span><span style="font-size: 14px;"></span></strong></h3> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">Hadoop&nbsp;是一个使用&nbsp;Java&nbsp;编写的&nbsp;Apache&nbsp;开放源代码框架,它允许使用简单的编程模型跨大型计算机的大型数据集进行分布式处理。</span><span style="font-size: 14px;">Hadoop&nbsp;框架工作的应用程序可以在跨计算机群集提供分布式存储和计算的环境中工作。</span><span style="font-size: 14px;">Hadoop&nbsp;旨在从单一服务器扩展到数千台机器,每台机器都提供本地计算和存储。</span></p> <p><br></p> <h3><strong><span style="font-size: 14px;">1.2 Hadoop 架构</span></strong><span style="font-size: 14px;"></span></h3> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">Hadoop&nbsp;框架包括以下四个模块:</span></p> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;"></span><span style="font-size: 14px;">Hadoop&nbsp;Common:</span><span style="font-size: 14px;">这些是其他&nbsp;Hadoop&nbsp;模块所需的&nbsp;Java&nbsp;库和实用程序。</span><span style="font-size: 14px;">这些库提供文件系统和操作系统级抽象,并包含启动&nbsp;Hadoop&nbsp;所需的必要&nbsp;Java&nbsp;文件和脚本。</span></p></li> <li><p><span style="font-size: 14px;">Hadoop&nbsp;YARN:</span><span style="font-size: 14px;">这是作业调度和集群资源管理的框架。</span></p></li> <li><p><span style="font-size: 14px;">Hadoop&nbsp;分布式文件系统(HDFS):</span><span style="font-size: 14px;">提供对应用程序数据的高吞吐量访问的分布式文件系统。</span></p></li> <li><p><span style="font-size: 14px;"></span><span style="font-size: 14px;">Hadoop&nbsp;MapReduce:</span><span style="font-size: 14px;">这是基于&nbsp;YARN&nbsp;的大型数据集并行处理系统。</span></p></li> </ul> <p><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">我们可以使用下图来描述&nbsp;Hadoop&nbsp;框架中可用的这四个组件。</span></p> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: center;"><img class="rich_pages" data-copyright="0" data-ratio="1.18" data-s="300,640" src="/upload/b6210c97fcec5682b0d69c3b97ca89ea.png" data-type="png" data-w="400" style=""></p> <p><br><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;">自&nbsp;2012&nbsp;年以来,术语“Hadoop”通常不仅指向上述基本模块,而且还指向可以安装在&nbsp;Hadoop&nbsp;之上或之外的其他软件包,例如&nbsp;Apache&nbsp;Pig、Apache&nbsp;Hive、Apache&nbsp;HBase、Apache&nbsp;火花等。</span><br></p> <p><span style="font-size: 14px;">&nbsp;</span></p> <h3><strong><span style="font-size: 14px;">1.3&nbsp;Hadoop&nbsp;如何工作?</span><span style="font-size: 14px;"></span></strong></h3> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">(1)阶段1</span></p> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">用户/应用程序可以通过指定以下项目向&nbsp;Hadoop(Hadoop&nbsp;作业客户端)提交所需的进程:</span></p> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;"></span><span style="font-size: 14px;">分布式文件系统中输入和输出文件的位置。</span></p></li> <li><p><span style="font-size: 14px;">Java类以 J</span><span style="font-size: 14px;">AR</span><span style="font-size: 14px;"> </span><span style="font-size: 14px;">文件的形式包含了 </span><span style="font-size: 14px;">M</span><span style="font-size: 14px;">ap 和 </span><span style="font-size: 14px;">R</span><span style="font-size: 14px;">educe </span><span style="font-size: 14px;">功能的实现。</span></p></li> <li><p><span style="font-size: 14px;">通过设置作业特定的不同参数来进行作业配置。</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">(2)阶段2</span></p> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">然后,Hadoop&nbsp;作业客户端将作业(JAR/可执行文件等)和配置提交给&nbsp;JobTracker,JobTracker&nbsp;负责将软件/配置分发到从站,调度任务和监视它们,向作业客户端提供状态和诊断信息。</span></p> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">(3)阶段3</span></p> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">不同节点上的&nbsp;TaskTrackers&nbsp;根据&nbsp;MapReduce&nbsp;实现执行任务,并将&nbsp;reduce&nbsp;函数的输出存储到文件系统的输出文件中。</span></p> <p><span style="font-size: 14px;">&nbsp;</span></p> <h3><strong><span style="font-size: 14px;">1.4 Hadoop 的优点</span><span style="font-size: 14px;"></span></strong></h3> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">Hadoop&nbsp;框架允许用户快速编写和测试分布式系统。</span><span style="font-size: 14px;">它是高效的,它自动分配数据并在机器上工作,反过来利用&nbsp;CPU&nbsp;核心的底层并行性。</span></p></li> </ul> <ul style="list-style-type: none;" class=" list-paddingleft-2"> <li><p><br></p></li> </ul> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">Hadoop&nbsp;不依赖硬件提供容错和高可用性(FTHA),而是&nbsp;Hadoop&nbsp;库本身被设计为检测和处理应用层的故障。</span></p></li> </ul> <ul style="list-style-type: none;" class=" list-paddingleft-2"> <li><p><br></p></li> </ul> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">服务器可以动态添加或从集群中删除,Hadoop&nbsp;继续运行而不会中断。</span></p></li> </ul> <ul style="list-style-type: none;" class=" list-paddingleft-2"> <li><p><br></p></li> </ul> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">Hadoop&nbsp;的另一大优点是,除了是开放源码,它是所有平台兼容的,因为它是基于&nbsp;Java&nbsp;的。</span></p></li> </ul> <p><span style="font-size: 14px;">&nbsp;</span></p> <h3><strong><span style="font-size: 14px;">1.5 HBase 介绍</span></strong><span style="font-size: 14px;"></span></h3> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">Hbase&nbsp;全称为&nbsp;Hadoop&nbsp;Database,即&nbsp;Hbase&nbsp;是&nbsp;Hadoop&nbsp;的数据库,是一个分布式的存储系统。</span><span style="font-size: 14px;">Hbase&nbsp;利用&nbsp;Hadoop&nbsp;的&nbsp;HDFS&nbsp;作为其文件存储系统,利用&nbsp;Hadoop&nbsp;的&nbsp;MapReduce&nbsp;来处理&nbsp;Hbase&nbsp;中的海量数据。</span><span style="font-size: 14px;">利用&nbsp;ZooKeeper&nbsp;作为其协调工具。</span><span style="font-size: 14px;">&nbsp;</span></p> <p><span style="font-size: 14px;">&nbsp;</span></p> <h3><strong><span style="font-size: 14px;">1.6 HBase 体系架构</span></strong><span style="font-size: 14px;"></span></h3> <p><br></p> <p style="text-align: center;"><img class="rich_pages" data-copyright="0" data-ratio="0.5397536394176932" data-s="300,640" src="/upload/9777ac47a74d4e3fa5d5e1ad05f02f82.png" data-type="png" data-w="893" style=""></p> <p><br><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;text-decoration: underline;">Client</span></p> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">包含访问 HBase 的接口并维护 Cache 来加快对 HBase 的访问</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;text-decoration: underline;">ZooKeeper</span><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">保证任何时候,集群中只有一个 Master</span></p></li> <li><p><span style="font-size: 14px;">存贮所有&nbsp;Region&nbsp;的寻址入口。</span></p></li> <li><p><span style="font-size: 14px;"></span><span style="font-size: 14px;">实时监控&nbsp;Region&nbsp;Server&nbsp;的上线和下线信息。</span><span style="font-size: 14px;">并实时通知 Master</span></p></li> <li><p><span style="font-size: 14px;">存储 HBase 的 Schema 和 table 元数据</span><br></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;text-decoration: underline;">Master</span><span style="font-size: 14px;"></span></p> <p><br></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">为 Region Server 分配 region</span></p></li> <li><p><span style="font-size: 14px;">负责 Region Server 的负载均衡</span></p></li> <li><p><span style="font-size: 14px;">发现失效的 Region Server 并重新分配其上的 region</span></p></li> <li><p><span style="font-size: 14px;">管理用户对 table 的增删改操作</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;text-decoration: underline;">RegionServer</span><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">Region Server 维护 region,处理对这些 region 的 IO 请求</span></p></li> <li><p><span style="font-size: 14px;">Region Server 负责切分在运行过程中变得过大的 region </span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;text-decoration: underline;">HLog(WAL log)</span><span style="font-size: 14px;"></span></p> <p><br></p> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">HLog&nbsp;文件就是一个普通的&nbsp;Hadoop&nbsp;Sequence&nbsp;File,Sequence&nbsp;File&nbsp;&nbsp;的&nbsp;Key&nbsp;是&nbsp;&nbsp;HLogKey&nbsp;对象,HLogKey&nbsp;中记录了写入数据的归属信息,除了&nbsp;table&nbsp;和&nbsp;region&nbsp;名字外,同时还包括&nbsp;Sequence&nbsp;Number&nbsp;和&nbsp;Timestamp,Timestamp&nbsp;是”&nbsp;写入时间”,Sequence&nbsp;Number&nbsp;的起始值为&nbsp;0,或者是最近一次存入文件系统中&nbsp;Sequence&nbsp;Number。</span></p></li> </ul> <p><br></p> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">HLog SequeceFile 的 Value 是 HBase 的 KeyValue 对象,即对应 HFile 中的&nbsp; KeyValue</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;text-decoration: underline;">Region</span><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">HBase 自动把表水平划分成多个区域(Rregion<span style="font-size: 14px;">)</span>,每个&nbsp;Region&nbsp;会保存一个表里面某段连续的数据;</span><span style="font-size: 14px;">每个表一开始只有一个&nbsp;Region,随着数据不断插&nbsp;入表,region&nbsp;不断增大,当增大到一个阀值的时候,Region&nbsp;就会等分会&nbsp;两个新的&nbsp;Region(裂变);</span></p></li> </ul> <ul style="list-style-type: none;" class=" list-paddingleft-2"> <li><p><span style="font-size: 14px;">&nbsp;</span></p></li> </ul> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">当&nbsp;table&nbsp;中的行不断增多,就会有越来越多的&nbsp;Region。</span><span style="font-size: 14px;">这样一张完整的表被保存在多个&nbsp;Regionserver&nbsp;上。</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;text-decoration: underline;">Memstore&nbsp;与 StoreFile</span><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">一个 Region 由多个 Store 组成,一个 Store 对应一个 CF(列族)</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">store 包括位于内存中的 Memstore 和位于磁盘的 StoreFile 写操作先写入 Memstore,当 MemStore 中的数据达到某个阈值,HRegionServer 会启动&nbsp; FlashCache 进程写入 StoreFile,每次写入形成单独的一个 StoreFile</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">当&nbsp;StoreFile&nbsp;文件的数量增长到一定阈值后,系统会进行合并(minor、&nbsp;major&nbsp;compaction),在合并过程中会进行版本合并和删除工作&nbsp;,形成更大的&nbsp;&nbsp;StoreFile。</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">当一个&nbsp;Region&nbsp;所有&nbsp;StoreFile&nbsp;的大小和超过一定阈值后,会把当前的&nbsp;region&nbsp;分割为两个,并由&nbsp;HMaster&nbsp;分配到相应的&nbsp;RegionServer&nbsp;服务器,实现负载均衡。</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">客户端检索数据,先在&nbsp;MemStore&nbsp;找,找不到再找&nbsp;StoreFile。</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">HRegion&nbsp;是&nbsp;HBase&nbsp;中分布式存储和负载均衡的最小单元。</span><span style="font-size: 14px;">最小单元就表示不同的&nbsp;HRegion&nbsp;可以分布在不同的&nbsp;HRegion&nbsp;Server上。</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">HRegion&nbsp;由一个或者多个&nbsp;Store&nbsp;组成,每个&nbsp;Store&nbsp;保存一个&nbsp;Columns&nbsp;Family。</span></p></li> </ul> <p><span style="font-size: 14px;"><br></span></p> <ul class=" list-paddingleft-2" style="list-style-type: disc;"> <li><p><span style="font-size: 14px;">每个&nbsp;Strore&nbsp;又由一个&nbsp;MemStore&nbsp;和&nbsp;0&nbsp;至多个&nbsp;StoreFile&nbsp;组成。</span></p></li> </ul> <p><br></p> <h2><span style="color: rgb(61, 167, 66);"><strong><span style="font-size: 14px;">2. 安装搭建 Hadoop</span></strong></span><span style="font-size: 14px;"></span></h2> <h3><span style="font-size: 14px;"><br></span></h3> <h3><strong><span style="font-size: 14px;">2.1&nbsp;配置说明</span></strong><span style="font-size: 14px;"></span></h3> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">本次集群搭建共三台机器,具体说明下:</span></p> <p><span style="font-size: 14px;"><br></span></p> <table width="682" style="width: 556px;"> <tbody style="border-collapse: collapse;border-width: 1px;border-style: solid;border-color: silver;"> <tr style="border-collapse: collapse;border-width: 1px;border-style: solid;border-color: silver;"> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><strong><span style="font-size: 14px;">主机名</span></strong></td> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><strong><span style="font-size: 14px;">IP</span></strong></td> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><strong><span style="font-size: 14px;">说明</span></strong></td> </tr> <tr style="border-collapse: collapse;border-width: 1px;border-style: solid;border-color: silver;"> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><span style="font-size: 14px;">hadoop01</span></td> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><span style="font-size: 14px;">192.168.10.101</span></td> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><span style="font-size: 14px;">DataNode、NodeManager、ResourceManager、NameNode</span></td> </tr> <tr style="border-collapse: collapse;border-width: 1px;border-style: solid;border-color: silver;"> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><span style="font-size: 14px;">hadoop02</span></td> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><span style="font-size: 14px;">192.168.10.102</span></td> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><span style="font-size: 14px;">DataNode、NodeManager、SecondaryNameNode</span></td> </tr> <tr style="border-collapse: collapse;border-width: 1px;border-style: solid;border-color: silver;"> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><span style="font-size: 14px;">hadoop03</span></td> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><span style="font-size: 14px;">192.168.10.106</span></td> <td style="border-collapse: collapse;border-color: silver;padding: 8px 14px;"><span style="font-size: 14px;">DataNode、NodeManager</span></td> </tr> </tbody> </table> <p><span style="font-size: 14px;">&nbsp;</span></p> <h3><strong><span style="font-size: 14px;">2.2&nbsp;安装前准备</span></strong><span style="font-size: 14px;"></span></h3> <h4><span style="font-size: 14px;"><br></span></h4> <h4><span style="font-size: 14px;">2.2.1&nbsp;机器配置说明</span></h4> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: left;"><img class="rich_pages" data-ratio="0.2982456140350877" data-s="300,640" src="/upload/9ec79fc17e690e773791385b6e10ee34.png" data-type="png" data-w="456" style="width: 291px;height: 87px;"></p> <p><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;color: rgb(136, 136, 136);">注:本集群内所有进程均由&nbsp;CLSN&nbsp;用户启动;要在集群所有服务器都进行操作。</span><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;">&nbsp;</span></p> <h4><span style="font-size: 14px;">2.2.2&nbsp;关闭 SELinux、防火墙</span></h4> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: left;"><img class="rich_pages" data-ratio="0.25822532402791626" data-s="300,640" src="/upload/30abef6ae9278921492f2e5af82375ef.png" data-type="png" data-w="1003" style=""></p> <p><br><span style="font-size: 14px;"></span></p> <h4><span style="font-size: 14px;">2.2.3&nbsp;准备用户</span></h4> <h4><br></h4> <p style="text-align: left;"><img class="rich_pages" data-ratio="0.1265597147950089" data-s="300,640" src="/upload/b91d0dc115a295265035756f72ab6f92.png" data-type="png" data-w="561" style="width: 336px;height: 43px;"></p> <h4>  </h4> <h4><span style="font-size: 14px;">2.2.4&nbsp;修改 Hosts 文件,域名解析</span></h4> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: left;"><img class="rich_pages" data-ratio="0.27448609431680776" data-s="300,640" src="/upload/e319de030cd726560c5b0849810f3025.png" data-type="png" data-w="827" style=""></p> <p style="text-align: left;"><span style="font-size: 14px;text-align: justify;">  </span><br></p> <p><span style="font-size: 14px;"></span></p> <h4><span style="font-size: 14px;">2.2.5&nbsp;同步时间</span></h4> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: left;"><img class="rich_pages" data-ratio="0.18414322250639387" data-s="300,640" src="/upload/d20b1e950fb5baa9d5caf2b533c4fd81.png" data-type="png" data-w="391" style="width: 223px;height: 41px;"></p> <p><span style="font-size: 14px;">  </span></p> <h4><span style="font-size: 14px;">2.2.6 SSH 互信配置</span></h4> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">(1)生成密钥对,一直回车即可</span></p> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: left;"><img class="rich_pages" data-ratio="0.1008174386920981" data-s="300,640" src="/upload/6eee7ea9e00263b7a0034744fcec29f1.png" data-type="png" data-w="367" style="width: 221px;height: 22px;"></p> <p><br><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;">(2)保证每台服务器各自都有对方的公钥</span></p> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: left;"><img class="rich_pages" data-ratio="0.47953216374269003" data-s="300,640" src="/upload/1a6313aae7f1be9c48b51450a9e47c62.png" data-type="png" data-w="684" style="width: 431px;height: 207px;"></p> <p><span style="font-size: 14px;color: rgb(136, 136, 136);">注:要在集群所有服务器都进行操作</span><span style="font-size: 14px;"></span></p> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">(3)验证无秘钥认证登录</span></p> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: left;"><img class="rich_pages" data-ratio="0.22666666666666666" data-s="300,640" src="/upload/699601018ff3fffd52572c32efea485b.png" data-type="png" data-w="450" style="width: 287px;height: 65px;"></p> <p><br><span style="font-size: 14px;"></span></p> <h3><strong><span style="font-size: 14px;">2.3&nbsp;配置 JDK</span></strong><span style="font-size: 14px;"></span></h3> <p><span style="font-size: 14px;"><br></span></p> <p><span style="font-size: 14px;">在三台机器上都需要操作</span></p> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: left;"><img class="rich_pages" data-ratio="0.4778523489932886" data-s="300,640" src="/upload/4478bb07e43039a45c3c12a4842ceff3.png" data-type="png" data-w="745" style="width: 443px;height: 212px;"></p> <h3><span style="font-size: 14px;"><br></span></h3> <h3><strong><span style="font-size: 14px;">2.4&nbsp;安装 Hadoop</span></strong></h3> <section class="code-snippet__fix code-snippet__js"> <ul class="code-snippet__line-index code-snippet__js"> <li></li> <li></li> <li></li> <li></li> </ul> <pre class="code-snippet__js" data-lang="cs"><code><span class="code-snippet_outer">[<span class="code-snippet__meta">root@hadoop01 ~</span>]<span class="code-snippet__meta">#&nbsp;wget&nbsp;https://mirrors.tuna.tsinghua.edu.cn/apache/hadoop/common/hadoop-3.2.0/hadoop-3.2.0.tar.gz</span></span></code><code><span class="code-snippet_outer">[<span class="code-snippet__meta">root@hadoop01 ~</span>]<span class="code-snippet__meta"># tar -xvf hadoop-3.2.0.tar.gz -C /usr/local/</span></span></code><code><span class="code-snippet_outer">[<span class="code-snippet__meta">root@hadoop01 ~</span>]<span class="code-snippet__meta"># chown along.along -R /usr/local/hadoop-3.2.0/</span></span></code><code><span class="code-snippet_outer">[<span class="code-snippet__meta">root@hadoop01 ~</span>]<span class="code-snippet__meta"># ln -s /usr/local/hadoop-3.2.0/ /usr/local/hadoop</span></span></code></pre> </section> <p><span style="font-size: 14px;"></span><br></p> <h2><span style="color: rgb(61, 167, 66);"><strong><span style="font-size: 14px;">3. 配置启动 Hadoop</span></strong></span><span style="font-size: 14px;"></span></h2> <h3><span style="font-size: 14px;"><br></span></h3> <h3><strong><span style="font-size: 14px;">3.1 &nbsp;hadoop-env.sh&nbsp;配置 Hadoop 环境变量</span><span style="font-size: 14px;"></span></strong></h3> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: left;"><img class="rich_pages" data-ratio="0.27986348122866894" data-s="300,640" src="/upload/dde21c06961c499db42fa60490d2840.png" data-type="png" data-w="586" style="width: 353px;height: 99px;"></p> <p><br><span style="font-size: 14px;"></span></p> <h3><strong><span style="font-size: 14px;">3.2 core-site.xml&nbsp;配置 HDFS</span><span style="font-size: 14px;"></span></strong></h3> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: left;"><img class="rich_pages" data-ratio="0.8201834862385321" data-s="300,640" src="/upload/fcba04f6687f5c4254589feb33072234.png" data-type="png" data-w="545" style="width: 369px;height: 303px;"></p> <p><span style="font-size: 14px;"></span><br></p> <p><span style="font-size: 14px;"></span></p> <h3><strong><span style="font-size: 14px;">3.3 hdfs-site.xml&nbsp;配置 NameNode</span></strong><span style="font-size: 14px;"></span></h3> <p><span style="font-size: 14px;"><br></span></p> <section class="code-snippet__fix code-snippet__js"> <ul class="code-snippet__line-index code-snippet__js"> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> </ul> <pre class="code-snippet__js" data-lang="xml"><code><span class="code-snippet_outer">[along@hadoop01 hadoop]$ vim hdfs-site.xml</span></code><code><span class="code-snippet_outer"><span class="code-snippet__tag">&lt;<span class="code-snippet__name">configuration</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__comment">&lt;!--&nbsp;设置namenode的http通讯地址&nbsp;--&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">name</span>&gt;</span>dfs.namenode.http-address<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">name</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">value</span>&gt;</span>hadoop01:50070<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">value</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;/<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> </span></code><code><span class="code-snippet_outer"> <span class="code-snippet__comment">&lt;!--&nbsp;设置secondarynamenode的http通讯地址&nbsp;--&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">name</span>&gt;</span>dfs.namenode.secondary.http-address<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">name</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">value</span>&gt;</span>hadoop02:50090<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">value</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;/<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> </span></code><code><span class="code-snippet_outer"> <span class="code-snippet__comment">&lt;!--&nbsp;设置namenode存放的路径&nbsp;--&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">name</span>&gt;</span>dfs.namenode.name.dir<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">name</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">value</span>&gt;</span>/data/hadoop/name<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">value</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;/<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> </span></code><code><span class="code-snippet_outer"> <span class="code-snippet__comment">&lt;!--&nbsp;设置hdfs副本数量&nbsp;--&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">name</span>&gt;</span>dfs.replication<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">name</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">value</span>&gt;</span>2<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">value</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;/<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__comment">&lt;!--&nbsp;设置datanode存放的路径&nbsp;--&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">name</span>&gt;</span>dfs.datanode.data.dir<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">name</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">value</span>&gt;</span>/data/hadoop/datanode<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">value</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;/<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> </span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">name</span>&gt;</span>dfs.permissions<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">name</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">value</span>&gt;</span>false<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">value</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;/<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"><span class="code-snippet__tag">&lt;/<span class="code-snippet__name">configuration</span>&gt;</span></span></code><code><span class="code-snippet_outer">[root@hadoop01 ~]# mkdir /data/hadoop/name -p</span></code><code><span class="code-snippet_outer">[root@hadoop01 ~]# mkdir /data/hadoop/datanode -p</span></code></pre> </section> <p><span style="font-size: 14px;"> </span><br><span style="font-size: 14px;"></span></p> <h3><strong><span style="font-size: 14px;">3.4 mapred-site.xml&nbsp;配置框架</span></strong><span style="font-size: 14px;"></span></h3> <p><span style="font-size: 14px;"><br></span></p> <p style="text-align: left;"><img class="rich_pages" data-ratio="1.1615008156606852" data-s="300,640" src="/upload/3d6c2cb1c8ddff7d24abb8d383736a55.png" data-type="png" data-w="613" style="width: 438px;height: 509px;"></p> <p><br><span style="font-size: 14px;"></span></p> <h3><strong><span style="font-size: 14px;">3.5 yarn-site.xml&nbsp;配置 ResourceManager</span></strong><span style="font-size: 14px;"></span></h3> <p><strong><span style="font-size: 14px;"><br></span></strong></p> <section class="code-snippet__fix code-snippet__js"> <ul class="code-snippet__line-index code-snippet__js"> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> </ul> <pre class="code-snippet__js" data-lang="xml"><code><span class="code-snippet_outer">[along@hadoop01 hadoop]$ vim yarn-site.xml</span></code><code><span class="code-snippet_outer"><span class="code-snippet__tag">&lt;<span class="code-snippet__name">configuration</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">name</span>&gt;</span>yarn.resourcemanager.hostname<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">name</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">value</span>&gt;</span>hadoop01<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">value</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;/<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> </span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">description</span>&gt;</span>The http address of the RM web application.<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">description</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">name</span>&gt;</span>yarn.resourcemanager.webapp.address<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">name</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">value</span>&gt;</span>${yarn.resourcemanager.hostname}:8088<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">value</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;/<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> </span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">description</span>&gt;</span>The address of the applications manager interface in the RM.<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">description</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">name</span>&gt;</span>yarn.resourcemanager.address<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">name</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">value</span>&gt;</span>${yarn.resourcemanager.hostname}:8032<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">value</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;/<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> </span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">property</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">description</span>&gt;</span>The address of the scheduler interface.<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">description</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__name">name</span>&gt;</span>yarn.resourcemanager.scheduler.address<span class="code-snippet__tag">&lt;/<span class="code-snippet__name">name</span>&gt;</span></span></code><code><span class="code-snippet_outer"> <span class="code-snippet__tag">&lt;<span class="code-snippet__nam

jfinal el表达式不生效

作者:じ☆ve宝贝

> 近期用jfinal写一个小页面做一下数据展示,打算集成jsp,没想到el表达式不生效,对比了之前的一个项目后发现原来问题出在web.xml中 ### 修改前: ![](/upload/9dc49578e97e4b4e9716955ff205938b.png) ### 修改后 ![](/upload/5a25fd37815949f09df8103278d469fc.png) ``` <?xml version="1.0" encoding="UTF-8"?> <web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" version="2.5" metadata-complete="true" > ``` **搞定**

大厂面试必问的volatile关键字,这一篇文章搞定!

作者:微信小助手

<p style="font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;letter-spacing: 0.544px;white-space: normal;background-color: rgb(255, 255, 255);text-size-adjust: auto;font-size: 15px;word-spacing: 2px;text-align: center;" data-mpa-powered-by="yiban.io"><span style="color: rgb(136, 136, 136);font-size: 14px;">点击蓝色“</span><span style="font-size: 14px;color: rgb(0, 128, 255);">架构文摘</span><span style="color: rgb(136, 136, 136);font-size: 14px;">”关注我哟</span></p> <p style="margin-bottom: 10px;font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;letter-spacing: 0.544px;white-space: normal;background-color: rgb(255, 255, 255);color: rgb(62, 62, 62);text-size-adjust: auto;font-size: 15px;word-spacing: 2px;text-align: center;"><span style="font-size: 14px;color: rgb(136, 136, 136);">加个“</span><span style="color: rgb(0, 128, 255);font-size: 14px;">星标</span><span style="font-size: 14px;color: rgb(136, 136, 136);">”,每天上午 09:25,干货推送!</span></p> <p style="margin-bottom: 10px;font-family: -apple-system-font, BlinkMacSystemFont, &quot;Helvetica Neue&quot;, &quot;PingFang SC&quot;, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei UI&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;letter-spacing: 0.544px;white-space: normal;background-color: rgb(255, 255, 255);color: rgb(62, 62, 62);text-size-adjust: auto;font-size: 15px;word-spacing: 2px;text-align: center;"><span style="font-size: 14px;color: rgb(136, 136, 136);"><br></span></p> <section class="output_wrapper" style="font-size: 15px;color: rgb(62, 62, 62);line-height: 1.6;letter-spacing: 0px;font-family: &quot;Helvetica Neue&quot;, Helvetica, &quot;Hiragino Sans GB&quot;, &quot;Microsoft YaHei&quot;, Arial, sans-serif;"> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">在Java相关的岗位面试中,很多面试官都喜欢考察面试者对Java并发的了解程度,而以volatile关键字作为一个小的切入点,往往可以一问到底,把Java内存模型(JMM),Java并发编程的一些特性都牵扯出来,深入地话还可以考察JVM底层实现以及操作系统的相关知识。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">下面我们以一次假想的面试过程,来深入了解下volitile关键字吧!</p> <h3 style="color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;font-weight: bold;font-size: 1.3em;border-bottom: 2px solid rgb(0, 172, 193);"><span style="font-size: inherit;line-height: inherit;display: inline-block;font-weight: normal;background: rgb(0, 172, 193);color: rgb(255, 255, 255);padding-top: 3px;padding-right: 10px;padding-left: 10px;border-top-right-radius: 3px;border-top-left-radius: 4px;margin-right: 2px;">面试官: Java并发这块了解的怎么样?说说你对volatile关键字的理解</span></h3> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">就我理解的而言,被volatile修饰的共享变量,就具有了以下两点特性:</p> <ol style="" class=" list-paddingleft-2"> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">保证了不同线程对该变量操作的内存可见性;</span></p></li> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">禁止指令重排序</span></p></li> </ol> <h3 style="color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;font-weight: bold;font-size: 1.3em;border-bottom: 2px solid rgb(0, 172, 193);"><span style="font-size: inherit;line-height: inherit;display: inline-block;font-weight: normal;background: rgb(0, 172, 193);color: rgb(255, 255, 255);padding-top: 3px;padding-right: 10px;padding-left: 10px;border-top-right-radius: 3px;border-top-left-radius: 4px;margin-right: 2px;">面试官: 能不能详细说下什么是内存可见性,什么又是重排序呢?</span></h3> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">这个聊起来可就多了,我还是从Java内存模型说起吧。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">Java虚拟机规范试图定义一种Java内存模型(JMM),来屏蔽掉各种硬件和操作系统的内存访问差异,让Java程序在各种平台上都能达到一致的内存访问效果。简单来说,由于CPU执行指令的速度是很快的,但是内存访问的速度就慢了很多,相差的不是一个数量级,所以搞处理器的那群大佬们又在CPU里加了好几层高速缓存。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">在Java内存模型里,对上述的优化又进行了一波抽象。JMM规定所有变量都是存在主存中的,类似于上面提到的普通内存,每个线程又包含自己的工作内存,方便理解就可以看成CPU上的寄存器或者高速缓存。所以线程的操作都是以工作内存为主,它们只能访问自己的工作内存,且工作前后都要把值在同步回主内存。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">这么说得我自己都有些不清楚了,拿张纸画一下:</p> <figure style="font-size: inherit;color: inherit;line-height: inherit;"> <img class="" data-ratio="0.5882352941176471" src="/upload/927fe219f31f1afd81ae6065b617765.png" data-type="png" data-w="816" style="font-size: inherit;color: inherit;line-height: inherit;display: block;margin-right: auto;margin-left: auto;height: auto !important;" title=""> </figure> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">在线程执行时,首先会从主存中read变量值,再load到工作内存中的副本中,然后再传给处理器执行,执行完毕后再给工作内存中的副本赋值,随后工作内存再把值传回给主存,主存中的值才更新。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">使用工作内存和主存,虽然加快的速度,但是也带来了一些问题。比如看下面一个例子:</p> <pre style="font-size: inherit;color: inherit;line-height: inherit;"><code class="hljs ini" style="margin-right: 2px;margin-left: 2px;line-height: 18px;font-size: 14px;letter-spacing: 0px;font-family: Consolas, Inconsolata, Courier, monospace;border-radius: 0px;color: rgb(169, 183, 198);background: rgb(40, 43, 46);padding: 0.5em;overflow-wrap: normal !important;word-break: normal !important;overflow: auto !important;display: -webkit-box !important;"><span class="hljs-attr" style="font-size: inherit;line-height: inherit;color: rgb(165, 218, 45);overflow-wrap: inherit !important;word-break: inherit !important;">i</span>&nbsp;=&nbsp;i&nbsp;+&nbsp;<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">1</span>;<br></code></pre> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">假设i初值为0,当只有一个线程执行它时,结果肯定得到1,当两个线程执行时,会得到结果2吗?这倒不一定了。可能存在这种情况:</p> <pre style="font-size: inherit;color: inherit;line-height: inherit;"><code class="hljs sql" style="margin-right: 2px;margin-left: 2px;line-height: 18px;font-size: 14px;letter-spacing: 0px;font-family: Consolas, Inconsolata, Courier, monospace;border-radius: 0px;color: rgb(169, 183, 198);background: rgb(40, 43, 46);padding: 0.5em;overflow-wrap: normal !important;word-break: normal !important;overflow: auto !important;display: -webkit-box !important;">线程1:&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">load</span>&nbsp;i&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">from</span>&nbsp;主存&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;i&nbsp;=&nbsp;<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">0</span><br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;i&nbsp;+&nbsp;<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">1</span>&nbsp;&nbsp;//&nbsp;i&nbsp;=&nbsp;<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">1</span><br>线程<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">2</span>:&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">load</span>&nbsp;i&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">from</span>主存&nbsp;&nbsp;//&nbsp;因为线程<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">1</span>还没将i的值写回主存,所以i还是<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">0</span><br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;i&nbsp;+&nbsp;&nbsp;<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">1</span>&nbsp;//i&nbsp;=&nbsp;<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">1</span><br>线程<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">1</span>:&nbsp;&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">save</span>&nbsp;i&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">to</span>&nbsp;主存<br>线程<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">2</span>:&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">save</span>&nbsp;i&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">to</span>&nbsp;主存<br></code></pre> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">如果两个线程按照上面的执行流程,那么i最后的值居然是1了。如果最后的写回生效的慢,你再读取i的值,都可能是0,这就是缓存不一致问题。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">下面就要提到你刚才问到的问题了,JMM主要就是围绕着如何在并发过程中如何处理<code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(233, 105, 0);background: rgb(248, 248, 248);">原子性</code>、<code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(233, 105, 0);background: rgb(248, 248, 248);">可见性</code>和<code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(233, 105, 0);background: rgb(248, 248, 248);">有序性</code>这3个特征来建立的,通过解决这三个问题,可以解除缓存不一致的问题。而volatile跟可见性和有序性都有关。</p> <h3 style="color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;font-weight: bold;font-size: 1.3em;border-bottom: 2px solid rgb(0, 172, 193);"><span style="font-size: inherit;line-height: inherit;display: inline-block;font-weight: normal;background: rgb(0, 172, 193);color: rgb(255, 255, 255);padding-top: 3px;padding-right: 10px;padding-left: 10px;border-top-right-radius: 3px;border-top-left-radius: 4px;margin-right: 2px;">面试官:那你具体说说这三个特性呢?</span></h3> <h4 style="line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;font-weight: bold;font-size: 1.2em;color: rgb(0, 172, 193);"><span style="font-size: inherit;color: inherit;line-height: inherit;">原子性(Atomicity)</span></h4> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.5em;margin-bottom: 1.5em;">Java中,对基本数据类型的读取和赋值操作是原子性操作,所谓原子性操作就是指这些操作是不可中断的,要做一定做完,要么就没有执行。比如:</p> <pre st

看动画轻松理解「递归」与「动态规划」

作者:微信小助手

<section class="xmteditor" style="display:none;" data-tools="新媒体管家" data-label="powered by xmt.cn"></section> <section class="output_wrapper" style="font-size: 14px;color: rgb(62, 62, 62);line-height: 1.8;word-spacing: 2px;letter-spacing: 2px;font-family: 'Helvetica Neue', Helvetica, 'Hiragino Sans GB', 'Microsoft YaHei', Arial, sans-serif;background-image: linear-gradient(90deg, rgba(50, 0, 0, 0.05) 3%, rgba(0, 0, 0, 0) 3%), linear-gradient(360deg, rgba(50, 0, 0, 0.05) 3%, rgba(0, 0, 0, 0) 3%);background-size: 20px 20px;background-position: center center;"> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">在学习「数据结构和算法」的过程中,因为人习惯了平铺直叙的思维方式,所以「递归」与「动态规划」这种带循环概念(绕来绕去)的往往是相对比较难以理解的两个抽象知识点。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">程序员小吴打算使用动画的形式来帮助理解「递归」,然后通过「递归」的概念延伸至理解「动态规划」算法思想。</p> <h3 style="color: inherit;line-height: inherit;margin-top: 1.6em;margin-bottom: 1.6em;font-weight: bold;border-bottom: 2px solid rgb(239, 112, 96);font-size: 1.3em;"><span style="font-size: inherit;line-height: inherit;display: inline-block;font-weight: normal;background: rgb(239, 112, 96);color: rgb(255, 255, 255);padding: 3px 10px 1px;border-top-right-radius: 3px;border-top-left-radius: 3px;margin-right: 3px;">什么是递归</span><span style="display: inline-block;vertical-align: bottom;border-bottom: 36px solid rgb(239, 235, 233);border-right: 20px solid transparent;"> </span></h3> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">先下定义:<strong style="font-size: inherit;line-height: inherit;color: rgb(233, 105, 0);">递归算法是一种直接或者间接调用自身函数或者方法的算法。</strong></p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">通俗来说,递归算法的实质是把问题分解成规模缩小的同类问题的子问题,然后递归调用方法来表示问题的解。它有如下特点:</p> <ul style="" class=" list-paddingleft-2"> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">1. 一个问题的解可以分解为几个子问题的解</span></p></li> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">2. 这个问题与分解之后的子问题,除了数据规模不同,求解思路完全一样</span></p></li> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">3. 存在递归终止条件,即必须有一个明确的递归结束条件,称之为递归出口</span></p></li> </ul> <figure style="font-size: inherit;color: inherit;line-height: inherit;"> <img class="" data-copyright="0" data-cropselx1="0" data-cropselx2="556" data-cropsely1="0" data-cropsely2="313" data-ratio="0.5634837355718783" src="/upload/68c252abc3d7b1644f69000a60427546.gif" data-type="gif" data-w="953" style="font-size: inherit;color: inherit;line-height: inherit;display: block;margin-right: auto;margin-left: auto;width: 556px;height: 313px;" title="递归动画"> <figcaption style="line-height: inherit;margin-top: 10px;text-align: center;color: rgb(153, 153, 153);font-size: 0.7em;"> 递归动画 </figcaption> </figure> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">通过动画一个一个特点来进行分析。</p> <h4 style="color: inherit;line-height: inherit;margin-top: 1.6em;margin-bottom: 1.6em;font-weight: bold;font-size: 1.2em;"><span style="font-size: inherit;color: inherit;line-height: inherit;">1.一个问题的解可以分解为几个子问题的解</span></h4> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">子问题就是相对与其前面的问题数据规模更小的问题。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">在动图中①号问题(一块大区域)划分为②号问题,②号问题由两个子问题(两块中区域)组成。</p> <h4 style="color: inherit;line-height: inherit;margin-top: 1.6em;margin-bottom: 1.6em;font-weight: bold;font-size: 1.2em;"><span style="font-size: inherit;color: inherit;line-height: inherit;">2. 这个问题与分解之后的子问题,除了数据规模不同,求解思路完全一样</span></h4> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">「①号划分为②号」与「②号划分为③号」的逻辑是一致的,求解思路是一样的。</p> <h4 style="color: inherit;line-height: inherit;margin-top: 1.6em;margin-bottom: 1.6em;font-weight: bold;font-size: 1.2em;"><span style="font-size: inherit;color: inherit;line-height: inherit;">3. 存在递归终止条件,即存在递归出口</span></h4> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">把问题分解为子问题,把子问题再分解为子子问题,一层一层分解下去,不能存在无限循环,这就需要有终止条件。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">①号划分为②号,②号划分为③号,③号划分为④号,划分到④号的时候每个区域只有一个不能划分的问题,这就表明存在递归终止条件。</p> <h3 style="color: inherit;line-height: inherit;margin-top: 1.6em;margin-bottom: 1.6em;font-weight: bold;border-bottom: 2px solid rgb(239, 112, 96);font-size: 1.3em;"><span style="font-size: inherit;line-height: inherit;display: inline-block;font-weight: normal;background: rgb(239, 112, 96);color: rgb(255, 255, 255);padding: 3px 10px 1px;border-top-right-radius: 3px;border-top-left-radius: 3px;margin-right: 3px;">从递归的经典示例开始</span><span style="display: inline-block;vertical-align: bottom;border-bottom: 36px solid rgb(239, 235, 233);border-right: 20px solid transparent;"> </span></h3> <h4 style="color: inherit;line-height: inherit;margin-top: 1.6em;margin-bottom: 1.6em;font-weight: bold;font-size: 1.2em;"><span style="font-size: inherit;color: inherit;line-height: inherit;">一.数组求和</span></h4> <figure style="font-size: inherit;color: inherit;line-height: inherit;"> <img class="" data-copyright="0" data-cropselx1="0" data-cropselx2="556" data-cropsely1="0" data-cropsely2="313" data-ratio="0.5634837355718783" src="/upload/a344655759e884f6a34c2d00f076c016.gif" data-type="gif" data-w="953" style="font-size: inherit;color: inherit;line-height: inherit;display: block;margin-right: auto;margin-left: auto;width: 556px;height: 313px;" title="数组求和"> <figcaption style="line-height: inherit;margin-top: 10px;text-align: center;color: rgb(153, 153, 153);font-size: 0.7em;"> 数组求和 </figcaption> </figure> <pre style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 0px;margin-bottom: 0px;padding: 0px;"><code class="hljs" style="margin-right: 2px;margin-left: 2px;line-height: 18px;word-spacing: 0px;letter-spacing: 0px;font-family: Consolas, Inconsolata, Courier, monospace;border-radius: 0px;color: rgb(169, 183, 198);background: rgb(40, 43, 46);padding: 0.5em;display: block !important;overflow-wrap: normal !important;word-break: normal !important;overflow: auto !important;"><span class="linenum hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);padding-right: 20px;overflow-wrap: inherit !important;word-break: inherit !important;">1</span>Sum(arr[0...n-1])&nbsp;=&nbsp;arr[0]&nbsp;+&nbsp;Sum(arr[1...n-1])<br></code></pre> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">后面的 Sum 函数要解决的就是比前一个 Sum 更小的同一问题。</p> <pre style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 0px;margin-bottom: 0px;padding: 0px;"><code class="hljs" style="margin-right: 2px;margin-left: 2px;line-height: 18px;word-spacing: 0px;letter-spacing: 0px;font-family: Consolas, Inconsolata, Courier, monospace;border-radius: 0px;color: rgb(169, 183, 198);background: rgb(40, 43, 46);padding: 0.5em;display: block !important;overflow-wrap: normal !important;word-break: normal !important;overflow: auto !important;"><span class="linenum hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);padding-right: 20px;overflow-wrap: inherit !important;word-break: inherit !important;">1</span>Sum(arr[1...n-1])&nbsp;=&nbsp;arr[1]&nbsp;+&nbsp;Sum(arr[2...n-1])<br></code></pre> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">以此类推,直到对一个空数组求和,空数组和为 0 ,此时变成了最基本的问题。</p> <pre style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 0px;margin-bottom: 0px;padding: 0px;"><code class="hljs" style="margin-right: 2px;margin-left: 2px;line-height: 18px;word-spacing: 0px;letter-spacing: 0px;font-family: Consolas, Inconsolata, Courier, monospace;border-radius: 0px;color: rgb(169, 183, 198);background: rgb(40, 43, 46);padding: 0.5em;display: block !important;overflow-wrap: normal !important;word-break: normal !important;overflow: auto !important;"><span class="linenum hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);padding-right: 20px;overflow-wrap: inherit !important;word-break: inherit !important;">1</span>Sum(arr[n-1...n-1]&nbsp;)&nbsp;=&nbsp;arr[n-1]&nbsp;+&nbsp;Sum([])<br></code></pre> <h4 style="color: inherit;line-height: inherit;margin-top: 1.6em;margin-bottom: 1.6em;font-weight: bold;font-size: 1.2em;"><span style="font-size: inherit;color: inherit;line-height: inherit;">二.汉诺塔问题</span></h4> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">汉诺塔(Hanoi Tower)问题也是一个经典的递归问题,该问题描述如下:</p> <blockquote style="line-height: inherit;padding: 15px 15px 15px 1rem;font-size: 0.9em;margin-top: 1em;margin-bottom: 1em;color: rgb(0, 0, 0);border-left-width: 5px;border-left-color: rgb(239, 112, 96);background: rgb(239, 235, 233);overflow: auto;overflow-wrap: normal;word-break: normal;"> <p style="font-size: inherit;color: inherit;line-height: inherit;">汉诺塔问题:古代有一个梵塔,塔内有三个座A、B、C,A座上有64个盘子,盘子大小不等,大的在下,小的在上。有一个和尚想把这个盘子从A座移到B座,但每次只能允许移动一个盘子,并且在移动过程中,3个座上的盘子始终保持大盘在下,小盘在上。</p> </blockquote> <figure style="font-size: inherit;color: inherit;line-height: inherit;"> <img class="" data-copyright="0" data-cropselx1="0" data-cropselx2="556" data-cropsely1="0" data-cropsely2="313" data-ratio="0.5634837355718783" src="/upload/d9d7966ec40632bb21c81d5363fe5995.gif" data-type="gif" data-w="953" style="font-size: inherit;color: inherit;line-height: inherit;display: block;margin-right: auto;margin-left: auto;width: 556px;height: 313px;" title="两个盘子"> <figcaption style="line-height: inherit;margin-top: 10px;text-align: center;color: rgb(153, 153, 153);font-size: 0.7em;"> 两个盘子 </figcaption> </figure> <figure style="font-size: inherit;color: inherit;line-height: inherit;"> <img class="" data-copyright="0" data-cropselx1="0" data-cropselx2="556" data-cropsely1="0" data-cropsely2="313" data-ratio="0.5634837355718783" src="/upload/f291c4156d0ee6b614b5b101c40aa609.gif" data-type="gif" data-w="953" style="font-size: inherit;color: inherit;line-height: inherit;display: block;margin-right: auto;margin-left: auto;width: 556px;height: 313px;" title="三个盘子"> <figcaption style="line-height: inherit;margin-top: 10px;text-align: center;color: rgb(153, 153, 153);font-size: 0.7em;"> 三个盘子 </figcaption> </figure> <ul style="" class=" list-paddingleft-2"> <li><p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">① &nbsp;如果只有 1 个盘子,则不需要利用 B 塔,直接将盘子从 A 移动到 C 。</p></li> <li><p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">② 如果有 2 个盘子,可以先将盘子 2 上的盘子 1 移动到 B ;将盘子 2 移动到 C ;将盘子 1 移动到 C 。这说明了:可以借助 B 将 2 个盘子从 A 移动到 C ,当然,也可以借助 C 将 2 个盘子从 A 移动到 B 。</p></li> <li><p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">③ 如果有 3 个盘子,那么根据 2 个盘子的结论,可以借助 C 将盘子 3 上的两个盘子从 A 移动到 B ;将盘子 3 从 A 移动到 C ,A 变成空座;借助 A 座,将 B 上的两个盘子移动到 C 。<br></p></li> <li><p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">④ 以此类推,上述的思路可以一直扩展到 n 个盘子的情况,将将较小的 n-1个盘子看做一个整体,也就是我们要求的子问题,以借助 B 塔为例,可以借助空塔 B 将盘子A上面的 n-1 个盘子从 A 移动到 B ;将A 最大的盘子移动到 C , A 变成空塔;借助空塔 A ,将 B 塔上的 n-2 个盘子移动到 A,将 B 最大的盘子移动到 C, B 变成空塔。。。</p></li> </ul> <h4 style="color: inherit;line-height: inherit;margin-top: 1.6em;margin-bottom: 1.6em;font-weight: bold;font-size: 1.2em;"><span style="font-size: inherit;color: inherit;line-height: inherit;">三.爬台阶问题</span></h4> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">问题描述:</p> <blockquote style="line-height: inherit;padding: 15px 15px 15px 1rem;font-size: 0.9em;margin-top: 1em;margin-bottom: 1em;color: rgb(0, 0, 0);border-left-width: 5px;border-left-color: rgb(239, 112, 96);background: rgb(239, 235, 233);overflow: auto;overflow-wrap: normal;word-break: normal;"> <p style="font-size: inherit;color: inherit;line-height: inherit;">一个人爬楼梯,每次只能爬1个或2个台阶,假设有n个台阶,那么这个人有多少种不同的爬楼梯方法?</p> </blockquote> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">先从简单的开始,以 4 个台阶为例,可以通过每次爬 1 个台阶爬完楼梯:</p> <figure style="font-size: inherit;color: inherit;line-height: inherit;"> <img class="" data-copyright="0" data-cropselx1="0" data-cropselx2="556" data-cropsely1="0" data-cropsely2="313" data-ratio="0.5634837355718783" src="/upload/4dc21036c9830afad7340a34bc744aa2.gif" data-type="gif" data-w="953" style="font-size: inherit;color: inherit;line-height: inherit;display: block;margin-right: auto;margin-left: auto;width: 556px;height: 313px;" title="每次爬 1 个台阶"> <figcaption style="line-height: inherit;margin-top: 10px;text-align: center;color: rgb(153, 153, 153);font-size: 0.7em;"> 每次爬 1 个台阶 </figcaption> </figure> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">可以通过先爬 2 个台阶,剩下的每次爬 1 个台阶爬完楼梯</p> <figure style="font-size: inherit;color: inherit;line-height: inherit;"> <img class="" data-copyright="0" data-cropselx1="0" data-cropselx2="556" data-cropsely1="0" data-cropsely2="313" data-ratio="0.5634837355718783" src="/upload/78772af73e22895c9854d1fc0b81da6e.gif" data-type="gif" data-w="953" style="font-size: inherit;color: inherit;line-height: inherit;display: block;margin-right: auto;margin-left: auto;width: 556px;height: 313px;" title="先爬 2 个台阶"> <figcaption style="line-height: inherit;margin-top: 10px;text-align: center;color: rgb(153, 153, 153);font-size: 0.7em;"> 先爬 2 个台阶 </figcaption> </figure> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">在这里,可以思考一下:可以根据第一步的走法把所有走法分为两类:</p> <ul style="" class=" list-paddingleft-2"> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">① 第一类是第一步走了 1 个台阶</span></p></li> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">② 第二类是第一步走了 2 个台阶</span></p></li> </ul> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">所以 n 个台阶的走法就等于先走 1 阶后,n-1 个台阶的走法 ,然后加上先走 2 阶后,n-2 个台阶的走法。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">用公式表示就是:</p> <blockquote style="line-height: inherit;padding: 15px 15px 15px 1rem;font-size: 0.9em;margin-top: 1em;margin-bottom: 1em;color: rgb(0, 0, 0);border-left-width: 5px;border-left-color: rgb(239, 112, 96);background: rgb(239, 235, 233);overflow: auto;overflow-wrap: normal;word-break: normal;"> <p style="font-size: inherit;color: inherit;line-height: inherit;"><code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(248, 35, 117);background: rgb(248, 248, 248);">f(n) = f(n-1)+f(n-2)</code></p> </blockquote> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">有了递推公式,递归代码基本上就完成了一半。那么接下来考虑递归终止条件。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">当有一个台阶时,我们不需要再继续递归,就只有一种走法。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">所以 <code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(248, 35, 117);background: rgb(248, 248, 248);">f(1)=1</code>。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">通过用 <code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(248, 35, 117);background: rgb(248, 248, 248);">n = 2</code>,<code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(248, 35, 117);background: rgb(248, 248, 248);">n = 3</code> 这样比较小的数试验一下后发现这个递归终止条件还不足够。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;"><code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(248, 35, 117);background: rgb(248, 248, 248);">n = 2</code> 时,<code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(248, 35, 117);background: rgb(248, 248, 248);">f(2) = f(1) + f(0)</code>。如果递归终止条件只有一个<code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(248, 35, 117);background: rgb(248, 248, 248);">f(1) = 1</code>,那 <code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(248, 35, 117);background: rgb(248, 248, 248);">f(2)</code> 就无法求解,递归无法结束。 &nbsp;<br>所以除了 <code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(248, 35, 117);background: rgb(248, 248, 248);">f(1) = 1</code> 这一个递归终止条件外,还要有 <code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(248, 35, 117);background: rgb(248, 248, 248);">f(0) = 1</code>,表示走 0 个台阶有一种走法,从思维上以及动图上来看,这显得的有点不符合逻辑。所以为了便于理解,把 <code style="font-size: inherit;line-height: inherit;overflow-wrap: break-word;padding: 2px 4px;border-radius: 4px;margin-right: 2px;margin-left: 2px;color: rgb(248, 35, 117);background: rgb(248, 248, 248);">f(2) = 2</code> 作为一种终止条件,表示走 2 个台阶,有两种走法,一步走完或者分两步来走。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">总结如下:</p> <ul style="" class=" list-paddingleft-2"> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">① 假设只有一个台阶,那么只有一种走法,那就是爬 1 个台阶</span></p></li> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">② 假设有两个个台阶,那么有两种走法,一步走完或者分两步来走</span></p></li> </ul> <figure style="font-size: inherit;color: inherit;line-height: inherit;"> <img class="" data-copyright="0" data-cropselx1="0" data-cropselx2="556" data-cropsely1="0" data-cropsely2="313" data-ratio="0.5634837355718783" src="/upload/b25826c0c178da5a00265f2fcd7105b4.gif" data-type="gif" data-w="953" style="font-size: inherit;color: inherit;line-height: inherit;display: block;margin-right: auto;margin-left: auto;width: 556px;height: 313px;" title="递归终止条件"> <figcaption style="line-height: inherit;margin-top: 10px;text-align: center;color: rgb(153, 153, 153);font-size: 0.7em;"> 递归终止条件 </figcaption> </figure> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">通过递归条件:</p> <pre style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 0px;margin-bottom: 0px;padding: 0px;"><code class="hljs" style="margin-right: 2px;margin-left: 2px;line-height: 18px;word-spacing: 0px;letter-spacing: 0px;font-family: Consolas, Inconsolata, Courier, monospace;border-radius: 0px;color: rgb(169, 183, 198);background: rgb(40, 43, 46);padding: 0.5em;display: block !important;overflow-wrap: normal !important;word-break: normal !important;overflow: auto !important;"><span class="linenum hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);padding-right: 20px;overflow-wrap: inherit !important;word-break: inherit !important;">1</span>f(1)&nbsp;=&nbsp;1;<br><span class="linenum hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);padding-right: 20px;overflow-wrap: inherit !important;word-break: inherit !important;">2</span>f(2)&nbsp;=&nbsp;2;<br><span class="linenum hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);padding-right: 20px;overflow-wrap: inherit !important;word-break: inherit !important;">3</span>f(n)&nbsp;=&nbsp;f(n-1)+f(n-2)<br></code></pre> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">很容易推导出递归代码:</p> <pre style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 0px;margin-bottom: 0px;padding: 0px;"><code class="hljs cpp" style="margin-right: 2px;margin-left: 2px;line-height: 18px;word-spacing: 0px;letter-spacing: 0px;font-family: Consolas, Inconsolata, Courier, monospace;border-radius: 0px;color: rgb(169, 183, 198);background: rgb(40, 43, 46);padding: 0.5em;display: block !important;overflow-wrap: normal !important;word-break: normal !important;overflow: auto !important;"><span class="linenum hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);padding-right: 20px;overflow-wrap: inherit !important;word-break: inherit !important;">1</span><span class="hljs-function" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;"><span class="hljs-keyword" style="font-size: inherit;line-height: inherit;overflow-wrap: inherit !important;word-break: inherit !important;">int</span>&nbsp;<span class="hljs-title" style="font-size: inherit;line-height: inherit;color: rgb(165, 218, 45);overflow-wrap: inherit !important;word-break: inherit !important;">f</span><span class="hljs-params" style="font-size: inherit;line-height: inherit;color: rgb(255, 152, 35);overflow-wrap: inherit !important;word-break: inherit !important;">(<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">int</span>&nbsp;n)</span>&nbsp;</span>{<br><span class="linenum hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);padding-right: 20px;overflow-wrap: inherit !important;word-break: inherit !important;">2</span>&nbsp;&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">if</span>&nbsp;(n&nbsp;==&nbsp;<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">1</span>)&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">return</span>&nbsp;<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">1</span>;<br><span class="linenum hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);padding-right: 20px;overflow-wrap: inherit !important;word-break: inherit !important;">3</span>&nbsp;&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">if</span>&nbsp;(n&nbsp;==&nbsp;<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">2</span>)&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">return</span>&nbsp;<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">2</span>;<br><span class="linenum hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);padding-right: 20px;overflow-wrap: inherit !important;word-break: inherit !important;">4</span>&nbsp;&nbsp;<span class="hljs-keyword" style="font-size: inherit;line-height: inherit;color: rgb(248, 35, 117);overflow-wrap: inherit !important;word-break: inherit !important;">return</span>&nbsp;f(n<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">-1</span>)&nbsp;+&nbsp;f(n<span class="hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);overflow-wrap: inherit !important;word-break: inherit !important;">-2</span>);<br><span class="linenum hljs-number" style="font-size: inherit;line-height: inherit;color: rgb(174, 135, 250);padding-right: 20px;overflow-wrap: inherit !important;word-break: inherit !important;">5</span>}<br></code></pre> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">通过上述三个示例,总结一下如何写递归代码:</p> <ul style="" class=" list-paddingleft-2"> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">1.找到如何将大问题分解为小问题的规律</span></p></li> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">2.通过规律写出递推公式</span></p></li> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">3.通过递归公式的临界点推敲出终止条件</span></p></li> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">4.将递推公式和终止条件翻译成代码</span></p></li> </ul> <h3 style="color: inherit;line-height: inherit;margin-top: 1.6em;margin-bottom: 1.6em;font-weight: bold;border-bottom: 2px solid rgb(239, 112, 96);font-size: 1.3em;"><span style="font-size: inherit;line-height: inherit;display: inline-block;font-weight: normal;background: rgb(239, 112, 96);color: rgb(255, 255, 255);padding: 3px 10px 1px;border-top-right-radius: 3px;border-top-left-radius: 3px;margin-right: 3px;">什么是动态规划</span><span style="display: inline-block;vertical-align: bottom;border-bottom: 36px solid rgb(239, 235, 233);border-right: 20px solid transparent;"> </span></h3> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">介绍动态规划之前先介绍一下分治策略(Divide and Conquer)。</p> <h4 style="color: inherit;line-height: inherit;margin-top: 1.6em;margin-bottom: 1.6em;font-weight: bold;font-size: 1.2em;"><span style="font-size: inherit;color: inherit;line-height: inherit;">分治策略</span></h4> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">将原问题分解为若干个规模较小但类似于原问题的子问题(<strong style="font-size: inherit;line-height: inherit;color: rgb(233, 105, 0);">Divide</strong>),「递归」的求解这些子问题(<strong style="font-size: inherit;line-height: inherit;color: rgb(233, 105, 0);">Conquer</strong>),然后再合并这些子问题的解来建立原问题的解。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">因为在求解大问题时,需要递归的求小问题,因此一般用「递归」的方法实现,即自顶向下。</p> <h4 style="color: inherit;line-height: inherit;margin-top: 1.6em;margin-bottom: 1.6em;font-weight: bold;font-size: 1.2em;"><span style="font-size: inherit;color: inherit;line-height: inherit;">动态规划(Dynamic Programming)</span></h4> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">动态规划其实和分治策略是类似的,也是将一个原问题分解为若干个规模较小的子问题,递归的求解这些子问题,然后合并子问题的解得到原问题的解。 &nbsp;<br>区别在于这些子问题会有重叠,一个子问题在求解后,可能会再次求解,于是我们想到将这些子问题的<strong style="font-size: inherit;line-height: inherit;color: rgb(233, 105, 0);">解存储起来</strong>,当下次再次求解这个子问题时,直接拿过来就是。 &nbsp;<br>其实就是说,动态规划所解决的问题是分治策略所解决问题的一个子集,只是这个子集更适合用动态规划来解决从而得到更小的运行时间。 &nbsp;<br><strong style="font-size: inherit;line-height: inherit;color: rgb(233, 105, 0);">即用动态规划能解决的问题分治策略肯定能解决,只是运行时间长了</strong>。因此,分治策略一般用来解决子问题相互对立的问题,称为标准分治,而动态规划用来解决子问题重叠的问题。</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">与「分治策略」「动态规划」概念接近的还有「贪心算法」「回溯算法」,由于篇幅限制,程序员小吴就不在这进行展开,在后续的文章中将分别详细的介绍「贪心算法」、「回溯算法」、「分治算法」,敬请关注:)</p> <p style="font-size: inherit;color: inherit;line-height: inherit;margin-top: 1.7em;margin-bottom: 1.7em;">将「动态规划」的概念关键点抽离出来描述就是这样的:</p> <ul style="" class=" list-paddingleft-2"> <li><p><span style="font-size: inherit;color: inherit;line-height: inherit;">1.动态规划法试图只解决每个子问题一次</span></p></li> <li><p><span style="font-size: inhe