You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

240 lines
15 KiB
Markdown

This file contains invisible Unicode characters!

This file contains invisible Unicode characters that may be processed differently from what appears below. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to reveal hidden characters.

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

# 03 | 指日可待:一步一步搭建秒杀系统(上)
你好,我是志东,欢迎和我一起从零打造秒杀系统。
在上一节课中我们搭建了本地的依赖环境,这节课我们将依据前篇中做的技术选型,继续搭建我们的开发项目,并在我们搭建好的项目上,**开发一个最简的秒杀系统**。
这个系统模拟通过商详页,进入到结算页,可以展示商品图片、名称、价格等;其次支持输入地址、选择支付方式、修改购买数量等操作行为;然后提交订单;最后在下单成功后模拟跳转到支付页。
麻雀虽小,五脏俱全,这些基本上涵盖了秒杀的整个流程,只不过大平台的结算元素更丰富,也有更多的分支操作,但这些都可以在我们的基础上,按照专栏中介绍的一些原则,进行增减。
整个项目的搭建和秒杀业务系统的具体实现还是比较复杂的,所以我们将分成两节课来完成。这节课我们主要是搭建项目,那么话不多说,我们直接开始吧。
## 职能划分
根据前面我们对秒杀业务做的分析可知要实现秒杀业务我们最少需要3个系统服务
* 一个是Nginx服务我们命名为demo-nginx
* 再一个是Web服务我们称其为demo-web
* 最后一个是RPC服务我们就叫它demo-support。
3个项目的关系如下图所示
![](https://static001.geekbang.org/resource/image/f6/3c/f627a52ea373f6d4cf3a1613fd9cdb3c.jpg?wh=1508x1108)
3个项目的目标职能划分这里我也详细介绍一下。
**首先是demo-nginx主要负责**
* 流量筛选:根据黑白名单、登录态和参数有效性等来筛选流量。
* 流量分发通过设置的负载均衡算法进行流量分发也可以自定义算法比如根据IP做hash或者根据用户ID做hash等。
* 简单业务以及校验:提供活动数据、活动有效性校验、库存数量校验和其他业务相关的简单校验等。
* 限流根据IP或者自定义关键入参做限流。
* 异常提示页面:主要是进结算页失败的提示页,可能是被限流,被业务校验拦截或者是后端服务异常等。
**其次是demo-web主要负责**
* 提供结算页H5。
提供结算页的HTML或者是重定向到CDN上的结算页H5地址。这里你肯定要问为什么不把这个功能前置到Nginx服务那样不是可以更快地返回吗
这里主要考虑到一个上线灰度发布的问题。你有没有想过,如果秒杀上线后,在后续的需求版本迭代中,我们页面新增了一个功能,该功能需要调用一个新的接口,我们该如何制定发布计划,来确保上线的安全以及不影响线上用户的使用体验呢?
我们可以选择全量上线,但这么做的后果就是一旦出现我们测试未覆盖到的业务场景而引发了线上事故,那么就会影响到全部的用户,这对我们来说是不可接受的,所以上线还得采取灰度发布的方式。
**那什么是灰度发布呢?**
大多数同学应该有所了解这里我只简单讲一下灰度发布的做法与目的。一般我们新功能上线都不是一下子全量上线而是有计划地逐步灰度上线即先上1台或几台服务线上观察如果有问题马上回滚。这样做的好处是如果真的出现问题那么受影响的流量也只有灰度的那些机器。也就是说如果你线上有1000台服务你上一台灰度那么受影响的流量也只有千分之一。这个做法最大限度地降低了事故的影响面。
但采用灰度发布的方式又会带来新的问题,即用户点击抢购进入到了新页面,新页面调用新接口,但如果这个请求打到了旧的服务器上,没有新的接口,那岂不是就报错了?
这里你不妨主动思考下:怎么确保灰度上线情况下,使同一个用户进入到新页面所发出的请求也能够打到新的接口服务上。
![](https://static001.geekbang.org/resource/image/cb/13/cbb1666d0c36ab235bbyy7ee8fb09413.jpg?wh=1870x926)
* 业务聚合接口提供结算页H5渲染页面所需数据以及提供用户行为操作所需接口比如下单等。
* 其他功能:部分关键接口的限流,以保证下游接口的安全。
**最后是demo-support主要负责**
提供基础服务、数据的支持,包括活动数据、商品数据、用户维度数据、提单等,主要模拟基础服务,正常情况下,应该是按业务模块做细致划分的,如下图所示:
![](https://static001.geekbang.org/resource/image/c0/1c/c0ac84c48a59c9a8dd25b2d5yy50971c.jpg?wh=1628x590)
这里我们为了开发方便就全部集成到一个demo-support服务里了。
## 项目搭建
现在,我们确定了项目服务职能范围,也做了相关的命名,下面就可以做项目的搭建了。
但在开始之前我要简单说明下在后面的项目搭建以及后续的开发中对涉及到demo-nginx项目的配置或Lua语法等我会多讲一些这部分可能大家都比较陌生而对于demo-web、demo-support项目的一些常用配置则会稍加精简因为大家会比较熟悉特别的地方会再重点说明。
以下除了demo-nginx项目外你在搭建项目时都可以通过自己熟悉的技术栈来实现只需要让目标系统达到我们所希望它具备的能力即可技术不限。
### **demo-nginx项目搭建**
这个项目主要是用Lua语言开发在上节课中我们安装了OpenResty本地创建的nginx.conf文件的路径在 ~/Documents/seckillproject/nginx/conf/nginx.conf这个是Nginx服务配置的核心文件Nginx启动就是根据这个配置来做一些初始化工作的。
一个nginx.conf文件的正常配置结构如下图所示
![](https://static001.geekbang.org/resource/image/24/b5/2418515a40bba14bcef425a35a5cbab5.jpg?wh=1726x1656)
这里的“全局模块配置”“events模块配置”以及HTTP模块配置中的“HTTP全局模块配置”都是直接配置在nginx.conf中属于核心配置。而“server模块配置”以及下面的“location块”配置都是我们写业务逻辑的地方比较灵活所以写到我们的demo-nginx项目中并通过Nginx的include关键字将对应的路径在nginx.conf中的“HTTP模块配置”里引用进来。
这里先粗略地介绍一下各个模块的作用:
* **全局模块配置**这里一般配置Nginx的进程数、日志目录与级别、CPU绑核等。
* **events模块配置**主要配置Nginx使用的工作模型进程连接数限制等
* **HTTP模块配置**这里就是处理HTTP请求的相关配置包括监控的域名、端口、URL以及业务代码的配置引用等。
那么在了解了项目的基本结构之后,我们就开始动手搭建吧。
**第一步:**我们先在本地创建一个demo-nginx项目这里我用的开发工具是IDEA如下图所示
![](https://static001.geekbang.org/resource/image/55/c4/55eaee8109975755b2dc6453d92d9fc4.png?wh=2594x936)
这里我创建了三个文件夹依次是config、domain、lua。其中config文件夹是用来存放一些常用配置的像一些常量配置和负载均衡配置等domain文件夹主要是用来存放“HTTP配置”中的server模块配置文件lua文件夹则主要用来存放用Lua语言编写的业务逻辑代码文件。
**第二步:**在domain文件下新增domain.com文件用来配置server和location。
**第三步:**修改我们的nginx.conf文件引入刚刚新建的domain.com文件以及lua文件夹下即将要写的.lua文件。nginx.conf内容修改如下
```plain
worker_processes 1; #工作进程数
error_log logs/error.log error;#日志路径  日志级别
events {
worker_connections 256;#单进程最大连接数
}
http {
lua_package_path "~/Documents/seckillproject/demo-nginx/lua/?.lua;;";
include ~/Documents/seckillproject/demo-nginx/domain/domain.com;
}
```
这里lua\_package\_path 的功能是导入项目中的 .lua文件。Include 是导入一些server模块配置文件就像在HTML里引用JavaScript一样。
**第四步:**配置server和location这里先只简单配置一个监控端口以及一个请求的匹配URL简单输出“hello world”配置内容如下
```plain
server {
    listen 7081;
    location /sayhello {
        default_type text/plain;
        content_by_lua_block {
            ngx.say("hello world!!!")
        }
    }
}
```
**第五步:**启动Nginx。还记得Nginx的命令吗不记得可以回头看看我们之前搭建OpenResty时教的。启动Nginx之后在终端或者浏览器输入请求地址就可以看到输出结果了如下图所示
![图片](https://static001.geekbang.org/resource/image/cd/76/cd850cda3dea612bf19c0f605b402076.png?wh=866x245)
这样我们的demo-nginx项目就搭建完成了剩下的就是填充配置与业务逻辑的开发了。
现在让我们马不停蹄,继续下一个项目的搭建。
### **demo-web项目搭建**
因为demo-web项目提供的是HTTP接口所以我们的基础框架使用的是SpringMVC新建了一个maven项目命名为demo-web并且新建了3个子module分别是demo-gatewaydemo-commondemo-service。
其中demo-gateway主要负责对外HTTP接口定义以及SpringMVC相关文件配置并且也是要打包部署的moduledemo-service主要用来做业务逻辑处理demo-common用来放一些公用方法或者工具类等。
整体项目结构如下图所示:
![](https://static001.geekbang.org/resource/image/ac/71/acdd356c114659c3a8bf2b05e4500b71.png?wh=2139x1500)
之后进行SpringMVC相关配置配置完成后可以自己写个controller测试一下是否搭建成功这里因为都是大家比较熟悉的领域所以就不做过多赘述了。
当demo-web项目搭建完成后就可以和我们之前搭建的demo-nginx项目联动一下了即请求先到Nginx被其配置的location拦截经过处理后将其分发给demo-web。所以接下来我们需要对demo-nginx做一些修改。
**第一步:**在config文件夹新建upstream.conf的配置文件用来配置打到后端Web服务器的IP和端口号。如果是线上这里可以配置服务器集群内容如下
```plain
upstream backend {
 server 127.0.0.1:8080;
}
```
其中upstream是Nginx关键字backend是自定义名称。我们了解的Nginx的负载均衡功能就是在这里配置的。修改后的项目结构如下图所示
![](https://static001.geekbang.org/resource/image/71/e5/71557d5a71295654dbb02b29fca5a2e5.png?wh=2600x972)
**第二步:**修改domain.com的server配置文件新增proxy\_pass配置项。意在将请求分发给后端服务器其中backend就是我们在第一步中自定义的名字如代码第5行所标
```plain
server {
listen 7081;
location /sayhello{
    default_type text/plain;
        proxy_pass http://backend;
}
}
```
**第三步:**在nginx.conf中将upstream.conf文件加载进去。
```plain
worker_processes 1;
error_log logs/error.log error;
events {
    worker_connections 256;
}
http {
    lua_package_path "~/Documents/seckillproject/demo-nginx/lua/?.lua;;";
    include ~/Documents/seckillproject/demo-nginx/domain/domain.com;
    include ~/Documents/seckillproject/demo-nginx/config/upstream.conf;
}
```
**第四步:**demo-nginx修改完成后我们在demo-web新建一个controller并启动。这里就要重点说明一下了Nginx配置的location URL需要和demo-web中定义的controller下对应的方法mapping保持一致不然两者是串不起来的。
准备完成后我们来启动一下Nginx在浏览器输入请求URL[http://localhost:7081/sayhello](http://localhost:7081/sayhello),你就会看到预期结果,这里就不展示截图了。
到这里我们的demo-nginx与demo-web两个项目就可以串联起来工作了。现在就只剩下一个demo-support项目了我们马上着手搭建它。
### **demo-support搭建**
可以参考刚刚的步骤与配置。首先新建demo-support项目然后新建5个子module如下图所示
![](https://static001.geekbang.org/resource/image/00/b1/008ddda95316ecd1ba38e5ae2342f3b1.png?wh=2133x1498)
* demo-support-common存放一些基本的工具方法或者常量等。
* demo-support-dao持久层主要存放数据库相关的SQL文件、实体、操作方法。
* demo-support-export对外接口定义层主要定义提供的RPC接口方法以及实体等。
* demo-support-service业务逻辑层。
* demo-support-launcher项目文件配置、监控、拦截器等同时也是打包部署module。
这里也同样不做过多的讲解你最终搭建起来项目使其能够操作数据库并能够提供RPC服务即可。这里特别说明一下就是在集成RPC框架Dubbo时因为是本地服务间调用所以不需要注册中心像下面这样配置即可线上再正常配置。
![](https://static001.geekbang.org/resource/image/db/50/db3234a5607018589946f04e9b9dd050.png?wh=2602x1118)
![](https://static001.geekbang.org/resource/image/31/95/31496dfb4df279ca07459a77098de095.png?wh=2603x1107)
## 小结
在这节课中我们为了开发一个最简的秒杀系统设计了3个系统项目一个是demo-nginx用来做真正的网关入口另一个是demo-web用来做业务的聚合最后一个是web-support用来做基础数据和服务的支撑。
同时我们给每个项目做了比较明确的职能划分然后按照顺序依次搭建起了3个项目。
在这3个项目中我们着重介绍了demo-nginx项目的搭建其他两个因为是大家比较熟悉的内容所以就没有做太多着墨。
由于在上节课的OpenResty搭建和这节课的demo-nginx搭建教学里一共出现了好几个和Nginx相关的文件在这里我们做个归纳总结以防你混淆。
![图片](https://static001.geekbang.org/resource/image/31/ee/31266d58627f134248ceebef3175daee.png?wh=1422x868)
通过表格,我们可以更清楚明了地看到这些文件的作用。同时到这里,我们的所有依赖环境和项目都已经准备就绪了,那么下节课我们将正式开始秒杀业务代码的编写,来实现我们课程开头设定的小目标。
## **思考题**
课程中布置过的一道思考题H5在做灰度上线时如何让新版本的页面请求始终打到新的灰度机器而老页面的请求始终打到旧版本的服务上两者不出现交叉呢
![](https://static001.geekbang.org/resource/image/cb/13/cbb1666d0c36ab235bbyy7ee8fb09413.jpg?wh=1870x926)
以上就是这节课的全部内容,欢迎你在评论区和我讨论问题,交流经验!