架构高性能网站秘笈(五)——Web组件分离
什么是Web组件?
网站的静态网页HTML、JavaScript脚本、CSS样式、图片、动态数据称为网站的Web组件。也就是说,一个Web应用由各种各样的Web组件构成。
为什么要进行Web组件分离?
一个网站的Web组件往往有各自的特点,比如:HTML页面属于静态文件,当用户请求一个HTML页面的时候Web服务器会进行IO操作,读取HTML文件;而用户请求动态数据的时候IO操作会比较少,但会涉及到大量的CPU计算;因此,如果静态内容和动态内容都使用相同服务器配置的话显然不能发挥Web应用最好的性能,因此我们需要对不同的Web组件采取不同的服务器配置方案。因此需要组件分离。
如何进行组件分离?
我们可以把不同的组件放在不同的服务器上,并且根据组件的特点,定制服务器配置,从而发挥组件最好的性能。要实现不同组件指向不同的服务器,我们首先需要为网站解析更多的子域名。
域名解析
假设我们已经拥有顶级域名www.5188.help,那么我们可以到购买域名的网站上设置域名的A标签,从而分出二级域名。以下是我解析的二级域名: - static.5188.help #用于存放静态数据 - api.5188.help #存放动态数据 - css.5188.help #存放css - js.5188.help #存放js - upload.5188.help #存放图片、音频、文档
如何对待不同的组件?
下面具体介绍针对具体组件的服务器配置方案。
1. 静态页面
静态页面HTML以文件的形式存储在存储设备,因此存储HTML页面的服务器需要有较高的IO读写速度,对IO密集型操作,我们要进行如下优化:
- 支持epoll。使得Web在高并发情况下仍然保持稳定的吞吐率。
- 非阻塞IO。避免不必要的IO等待。
- 异步IO
- 使用sendfile()系统调用。避免文件系统磁盘缓冲区到用户地址空间的数据复制。
- 单进程。避免多进程切换的不必要开销。对于IO密集型的静态内容处理,多进程并不能带来多大的意义。
- 使用高转速磁盘。
- 使用RAID分区。使得磁盘实现并行读写,提高磁盘吞吐量。
- 购买大带宽。
2. 动态内容
动态内容的数据都实时计算生成,或查询数据库得到,为了提升运算速度,因此需要增加CPU核数,增加内存容量,具体做法如下: 1. 使用快的CPU 2. 使用大内存 3. 使用多进程 4. 使用数据库连接池,减去连接建立和释放的开销。
3. CSS样式表和avaScript脚本
一般网站上线后CSS样式表和JavaScript脚本几乎不会发生变化,因此完全可以将css和js在用户浏览器的缓存有效期设置更长的时间。
注:在css、js的URL后可以加一个参数,用来标注当前css、js的版本,如:
<link href="css.5188.help/BSB/css/xxx.css?ver=1.0" />
当服务器中css、js发生修改后,需要将参数进行修改:
<link href="css.5188.help/BSB/css/xxx.css?ver=2.0" />
那么,当浏览器发现css后的参数发生修改时,会重新向服务器请求,而不会使用本地缓存。
4. 图片
对于图片较多的页面,如果每个图片都向服务器请求的话需要消耗大量的时间带带宽,因此服务器向浏览器返回响应信息时一定要设置图片的Keep-Alive参数设为true,延用TCP连接。
Web组件分离的好处
浏览器对于同一域名的并发数会有限制。Web组件分离之后,不同类型的Web组件需要请求不同的域名,从而能够支持更大的并发量,从而能够提升Web组件的下载速度。
- XGoServer 一个基础性、模块完整且安全可靠的服务端框架
- Bing 每日一图 & 随机图片 API
- 可视化数据库MapD安装——GPU模式
- 12步轻松搞定Python装饰器
- 使用shell脚本检测数据库连接访问情况(r10笔记第98天)
- 贝叶斯分类器及Python实现
- Docker 简介与安装
- 三种决策树算法(ID3, CART, C4.5)及Python实现
- Logistic 回归算法及Python实现
- MySQL主从不一致的修复过程(r10笔记第96天)
- ML中相似性度量和距离的计算&Python实现
- Oracle中的ROWID实现(r10笔记第95天)
- 100个Numpy练习【3】
- 100个Numpy练习【4】
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- 把redux当做观察者单独使用
- 用Spring Boot Admin来监控我们的微服务
- java+testNG测试框架搭建(接口测试或者ui测试)
- 解决Selenium testNG执行测试时,每个测试方法都打开一个浏览器窗口的问题
- selenium元素定位中css或者xpath不选择某一类元素
- TRTC横竖屏切换
- Swift 元祖
- Flutter - 解决混合开发iOS脚本打包遇到的问题
- Shader 特效 —— Film Burn (炫光光晕)效果【GLSL】
- java selenium chromedriver浏览器驱动放在哪里?【两种位置】
- 56. Vue原生js的组件拆分结构设计
- 一步一步教你把 Redux Saga 添加到 React&Redux 程序中
- Octave的基本语句及函数的使用入门—ML Note 31
- JAVA的Lock锁接口实现
- 抽象语法树为什么抽象