- 使用 Spring Initializr 创建 Spring Boot 应用程序
- 在Spring Boot中配置Cassandra
- 在 Spring Boot 上配置 Tomcat 连接池
- 将Camel消息路由到嵌入WildFly的Artemis上
DB主从分离、分库分表后,随并发和数据量增长,磁盘I/O成为系统性能瓶颈,于是缓存上场了!
一种存储数据的组件,让对数据请求更快返回。
某些场景下可能还会使用SSD作为冷数据的缓存。比如说360开源的Pika就是使用SSD存储数据解决Redis的容量瓶颈的。
凡是位于速度相差较大的两种硬件之间,用于协调两者数据传输速度差异的结构,均可称为缓存。
常见硬件组件延时:
做一次内存寻址大概需要100ns,而一次磁盘查找则需10ms。
使用内存作为缓存的存储介质相比以磁盘作为主要存储介质的DB,性能会提高多个数量级,也能支撑更高并发。所以,内存是最常见的一种缓存数据的介质。
实现从虚拟地址到物理地址的转换,但若每次转换都要做这么复杂计算,无疑会造成性能损耗,所以借助TLB(Translation Lookaside Buffer)缓存最近转换过的【虚拟地址和物理地址的映射】。这就是一种缓存组件,缓存复杂运算的结果。
实际上是使用内置的网络播放器来完成的。网络播放器接收的是数据流,将数据下载下来之后经过分离音视频流,解码等流程后输出到外设设备上播放。若在打开一个视频时,才开始下载数据,无疑增加视频打开速度(首播时间),且播放过程中会卡顿。所以播放器中通常会设计一些缓存组件,在未打开视频时缓存一部分视频数据,比如打开x音,服务端可能一次返回三个视频信息,播放第一个视频时,播放器已帮我们缓存第二、三个视频的部分数据,这样在看第二个视频的时候就可以给用户“秒开”体验。
第一次请求静态资源时,如一张图片,服务端除返回图片信息,在响应头里还有一个“etag”字段。ETag
HTTP响应头是资源的特定版本标识符。可让缓存更高效,并节省带宽,因为若内容没变,Web服务器无需发送完整响应。而若内容发生变化,使用ETag有助于防止资源的同时更新相互覆盖。
若给定URL中的资源更改,则一定要生成新Etag值。 因此Etags类似于指纹,也可能被某些服务器用于跟踪。 比较etags能快速确定此资源是否变化,但也可能被跟踪服务器永久存留。
浏览器会缓存图片信息及该字段值。当下一次再请求该图片时,浏览器发起的请求头里有个“If-None-Match”的字段,并且把缓存的“Etag”的值写进去发给服务端。
缓存:
理论上可通过缓存解决所有“慢”,如从磁盘随机读取数据慢,从DB查询数据慢,只是不同场景消耗的存储成本不同。
而缓冲区是块临时存储数据的区域,这些数据后面会被传输到其他设备。缓冲区更像MQ,用以弥补高速设备和低速设备通信时的速度差。如将数据写入磁盘时并不是直接刷盘,而是写到一块缓冲区,内核会标识该缓冲区为脏。当经过一定时间或脏缓冲区比例到达阈值,由单独线程把脏块刷盘,避免每次写数据都要刷盘的性能消耗。
Web 1.0时盛行,通过生成Velocity模板或静态HTML文件实现静态缓存。
在Nginx上部署静态缓存可减少对后台服务器压力。如做内容管理系统,后台会录入很多文章,前台在网站上展示文章内容,就像新浪,网易这种门户网站。
也可将文章录DB,然后前端展示时,穿透查询DB,但这样会对DB很大压力。即使使用分布式缓存挡读请求,但对于像日均PV几十亿的大型门户网站来说,基于成本考虑仍不划算。
所以解决思路是每篇文章在录入的时候渲染成静态页面,放置在所有前端Nginx或Squid等Web服务器上,这样用户在访问时,优先访问Web服务器上的静态页面,在对旧文章执行一定清理策略后,依然可保证99%以上缓存命中率。
这种缓存只能针对静态数据缓存,动态请求无能为力,就需要分布式缓存了。
Redis就是分布式缓存典型,性能强劲,通过分布式方案组成集群可以突破单机的限制。
那何时考虑热点本地缓存呢?
遇到极端热点数据查询时,热点本地缓存主要部署在应用服务器代码中,以阻挡热点查询对分布式缓存节点或DB的压力。
如某明星离婚,吃瓜们会到他微博下围观,这就会引发该用户信息的热点查询。这些查询通常会命中某一缓存节点或某一DB分区,短时间内会形成极高热点查询。
如HashMap,Guava Cache,它们和应用程序部署在同一进程,好处就是无需跨网络调度,速度极快,可阻挡短时间内热点查询。
如垂直电商系统首页的推荐商品,这些商品信息由运营在后台录入和变更。你分析运营录入新的商品或变更某商品信息后,在页面的展示允许有一些延迟,如30s延迟,且首页请求量最大,即使使用分布式缓存也很难抗,所以使用Guava Cache将所有推荐商品的信息缓存起来设置每隔30s重新从DB加载最新的所有商品。
首先,初始化Guava 的Loading Cache:
获取所有商品信息时,可调用Loading Cache的get,优先从本地缓存获取商品信息,若不存在,会使用CacheLoader中逻辑从DB加载所有商品。
由于本地缓存部署在应用服务器,通常集群部署,当数据更新时,不能确定哪台服务器本地中了缓存,更新或删除所有服务器的缓存不是好选择,所以通常会等待缓存过期。因此,这种缓存的有效期很短,通常为min或s级别,以避免返给前端脏数据。
缓存不是银弹:
缓存受限存储介质,不可能缓存所有数据,当数据有热点属性时,才能保证缓存命中率。如朋友圈这种20%内容会占80%流量。所以,一旦当业务场景读少写多时或无明显热点,如搜索场景,每个人搜索词都不同,无明显热点,缓存无明显意义。
当更新DB成功,更新缓存失败场景下,缓存中就会存在脏数据。
因此,使用缓存时要做数据存储量级评估,需消耗极大存储成本的数据,慎用缓存。缓存一定要设置TTL,可保证缓存中的会是热点数据。
运维需要对缓存组件有一定的了解,排查问题时,也多了考量。
缓存可以有多层,如:
需要将请求尽量挡在上层,因为越往下层,对于并发的承受能力越差。缓存命中率是缓存最重要的监控项。
缓存不仅仅是一种组件的名字,更是一种设计思想,任何能够加速读请求的组件和设计方案都是缓存思想的体现。而这种加速通常是通过两种方式来实现:
当你在实际工作中碰到“慢”问题,缓存就是你的第一考量。
参考
在 JavaScript 中,我们可以动态创建 元素并附加到 部分,以便为大量元素应用 CSS 规则。 这种方法的优点或缺点是什么? 如果它确实提供了与元素上的 javascript 迭代相比的性
我有这个代码 import "./HTTPMethod.dart"; import '../../DataModel/DataModel.dart'; mixin RouterMixin { HT
哪些 OLAP 工具支持动态、动态地创建维度或层次结构? 例如,层次结构将成员定义为:“前 5 名”、“前 6-10 名”、“其他”... 计算成员是通常的答案,我正在寻找不同的东西。计算器的问题。成
我正在 CakePHP 中创建一个“表单编辑器”。 该界面允许用户选择要应用于字段的验证,例如数字、电子邮件等 因此,我需要根据用户输入为模型动态创建验证。为此,我可以使用验证对象:https://b
这是一个场景: 我有一个Web服务,我们将其称为部署在tomcat(轴)上的StockQuoteService。通过此 Web 服务公开了 getStockQuote() 方法。 现在,我想构建一个
我正在尝试从服务器获取 JSON 响应并将其输出到控制台。 Future login() async { var response = await http.get( Uri.
我从另一个问题中得到了这段代码(感谢 chunhunghan)。我需要创建一个登录屏幕,并尝试根据服务器发回给我的响应来验证用户凭据,但是每次我尝试运行代码时,它都会给我“未处理的异常:Interna
当我在“Dart”主程序中运行它时,一切正常,并且我得到了一个与会者列表。但是,当我在我的 Flutter 应用程序中调用它时,出现错误: flutter:“List”类型不是“List>”类型的子类
本文实例为大家分享了js实现验证码动态干扰的具体代码,供大家参考,具体内容如下 效果一 效果二 代码一 ?
目前我正在为我的网站使用 No-Ip,我想使用 cloudflare 来抵御 ddos 和机器人程序。我注意到您需要一个用于 cloudflare 的域。我还搜索了网络,发现了一个叫做 cloud
有没有办法在 Excel VBA 中构建动态 if 语句?基本上我正在尝试创建一个参数化计算,用户将能够输入不同的变量,即 变量 1 “变量 2” “变量 3” 在这种情况下 变量 1 是单元格引用
大家好, 请查看上面的图片,我有两张 table 。在下面代码的第一个表中,我得到了这种格式。 但我想像 Table2 那样格式化,每个合并单元格中的行数是动态的,而且不一样。 有没有办法像table
如何根据我添加的 View 修改标题部分的高度?heightForHeaderInSection在 viewForHeaderInSection 之前被调用我不知道 View 大小,直到我创建它。 最
是否存在在运行时生成 AST/解析树的解析器?有点像一个库,它会接受一串 EBNF 语法或类似的东西并吐出数据结构? 我知道 antlr、jlex 和他们的同类。他们生成可以做到这一点的源代码。 (喜
我在持有汽车制造商的表格上有一个 MultipleChoiceField。我想将我的汽车数据库过滤到已检查的品牌,但这会导致问题。如何动态获取所有 Q(make=...) 语句? 我如何开始:['va
$end = preg_replace($pattern, $replacement, $str); 如何使替换字符串 $replacement 随 $str 中的每次匹配而变化?例如,我想用关联的图
我正在编写一个 VBA 程序,用于过滤表中的值。我试图使其成为一个适用于您提供的所有表格的通用程序。在我的程序中,我必须设置它正在过滤的表的范围:Set rng = dataSheet.Range("
我正在循环一个元素数组,并且我想使用给定的模板递归地显示该元素 然后在该模板内使用带有切换功能的按钮来显示/隐藏给定元素的Child的更深级别模板(Child也是一个元素) 这是我的模板
从客户端(html)发送表单,服务器端通过选择选项之一决定运行哪个函数。 const decideWho = (form) => { const choice = form.choice; c
我有一个具有以下属性的按钮: circle_normal.xml(在 res/drawable 中) circle.xml(在 res/drawable 中)
我是一名优秀的程序员,十分优秀!