通过canal+Nginx+lua+redis实现缓存预热和二级缓存

缓存穿透问题: 当客户端发起访问, nginx本地没有缓存, 查询redis也没有缓存, 就会去查mysql, 当mysql中查询不到数据时, nginx和redis中不会也有更新的缓存数据; 当这种无结果的访问被黑客攻击高并发请求时, 就会造成mysql数据库频繁访问, 产生缓存穿透现象.

解决措施: 使用nginx+redis实现缓存预热, 如果从nginx和redis中都无法获得数据, 直接返回给客户端, 不去访问数据库.


缓存预热原理图, 通过nginx和redis将mysql访问压力截断

实现技术: canal监测数据库变化 + rabbitmq消息队列分发+nginx lua执行redis脚本和mysql数据库数据更新


缓存预热以及二级缓存实现技术流程

一. canal

  • canal介绍:

阿里研发的对数据库binlog日志监听的服务器技术
原始作用: 为了跨机房进行mysql数据库同步
现在用来监听mysql服务器, 监测数据库数值的变化, 发送给canal客户端

  • 原理相对比较简单:
  1. canal模拟mysql slave的交互协议,伪装自己为mysql slave,向mysql master发送dump协议
  2. mysql master收到dump请求,开始推送binary log给slave(也就是canal)
  3. canal解析binary log对象(原始为byte流)
  • 开启mysql binary log

(1)查看当前mysql是否开启binlog模式。

SHOW VARIABLES LIKE '%log_bin%'

如果log_bin的值为OFF是未开启,为ON是已开启。

(2)修改/etc/my.cnf 需要开启binlog模式。

[mysqld]
log-bin=mysql-bin
binlog-format=ROW
server_id=1

修改完成之后,重启mysqld的服务。

  • canal服务端安装配置

(1)下载地址canal

https://github.com/alibaba/canal/releases/tag/canal-1.0.24

(2)下载之后 上传到linux系统中,解压缩到指定的目录/usr/local/canal
(3)修改 exmaple下的实例配置

vi conf/example/instance.properties

修改如图所示的几个参数。提供监测的mysql服务器的地址,以及用户名和密码


image.png

一定要注释掉下面这个参数,这样就会扫描全库

#canal.instance.defaultDatabaseName =

(3)启动服务:

[root@localhost canal]# ./bin/startup.sh
  • 配置canal的客户端, 接收服务端监测到的数据变化:
  1. 创建工程模块changgou_canal,pom引入依赖
    我们这里使用的一个开源的项目,它实现了springboot与canal的集成。比原生的canal更加优雅。
    https://github.com/chenqian56131/spring-boot-starter-canal
    使用前需要将starter-canal安装到本地仓库。安装方法参考:将github第三方jar包安装到本地maven
    安装完成后倒入pom依赖:
<dependency>
    <groupId>com.xpand</groupId>
    <artifactId>starter-canal</artifactId>
    <version>0.0.1-SNAPSHOT</version>
</dependency>
  1. 创建包com.changgou.canal ,包下创建启动类
@SpringBootApplication
@EnableCanalClient   //开启canal客户端支持
public class CanalApplication {

    public static void main(String[] args) {
        SpringApplication.run(CanalApplication.class, args);
    }
}
  1. 添加配置文件application.properties
canal.client.instances.example.host=192.168.225.128 //这里是canal服务器端的ip
canal.client.instances.example.port=11111
canal.client.instances.example.batchSize=1000
  1. 创建com.changgou.canal.listener包,包下创建类

@CanalEventListener
public class BusinessListener {

 
    /**
     *  ListenPoint schema: 数据库名  table: 表名
     * @param eventType
     * @param rowData
     */
    @ListenPoint(schema = "changgou_business", table = {"tb_ad"})
    public void adUpdate(CanalEntry.EventType eventType, CanalEntry.RowData rowData) {
        System.err.println("数据发生变化");
         for(CanalEntry.Column column: rowData.getAfterColumnsList()) {
          
        }
    }
}

其中ListenPoint schema代表监测的数据库名, table 代表监测的表名
启动客户端服务, 这时如果修改了changgou_business库中tb_ad表中的值, 就会在控制台上收到打印的内容: 数据发生变化
rowData.getAfterColumnsList() 可以获取到对应产生变化后的那一行的数据.
rowData.getBeforeColumnsList() 可以获取到对应产生变化前的那一行的数据.
监控到mysql数据的变化后可以根据自己的需求发送变化通知, 这里我们之后会发送rabbitmq消息通知队列监控服务做出处理.

二. RabbitMQ

    1. 在rabbitmq管理后台创建队列 ad_update_queue ,用于接收广告更新通知
    1. 引入rabbitmq起步依赖
<dependency>
    <groupId>org.springframework.amqp</groupId>
    <artifactId>spring-rabbit</artifactId>
</dependency>
    1. 配置文件application.properties 添加内容
spring.rabbitmq.host=192.168.225.128  //rabbitMQ的服务器ip
    1. 修改BusinessListener类, 发送消息给rabbitmq的消息队列
@CanalEventListener
public class BusinessListener {

    @Autowired
    private RabbitTemplate rabbitTemplate;

    @ListenPoint(schema = "changgou_business", table = {"tb_ad"})
    public void adUpdate(CanalEntry.EventType eventType, CanalEntry.RowData rowData) {
        System.err.println("广告数据发生变化");

        //修改后数据
        for(CanalEntry.Column column: rowData.getAfterColumnsList()) {
            if(column.getName().equals("position")){
                System.out.println("发送消息到mq  ad_update_queue:"+column.getValue());
                // 参数2ad_update_queue为消息队列的名称
                //参数1: 是交换机exchage. 这个例子中没有使用
               // 参数3为发送的消息内容, 这个列子中我们发送的为position的值 
               rabbitTemplate.convertAndSend("","ad_update_queue",column.getValue());  //发送消息到mq
                break;
            }
        }
    }
}
  • 5.从mq中提取消息执行更新
    创建消息接收更新工程, 引入pom文件
<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
<dependency>
  <groupId>com.squareup.okhttp3</groupId>
  <artifactId>okhttp</artifactId>
  <version>3.9.0</version>
</dependency>

在spring节点下配置rabbitmq的host地址

spring:
  rabbitmq:
    host: 192.168.200.128

创建rabbitmq消息监听类, 注解@RabbitListener设置监听的队列名称

@Component
@RabbitListener(queues = "ad_update_queue")
public class AdListener {

    /**
     * 获取更新广告通知
     * @param message
     */
    @RabbitHandler
    public void updateAd(String message){
        System.out.println("接收到消息:"+message);      
}

三. nginx+lua+redis

当我们接收到数据库信息变更时, 最重要的是通知nginx进行本地更新并且将数据缓存至redis, 使用户能访问获得最新数据.
这里我们使用openRestry,OpenResty(又称:ngx_openresty) 是一个基于 NGINX 的可伸缩的 Web 平台,由中国人章亦春发起,提供了很多高质量的第三方模块。OpenResty 简单理解成 就相当于封装了nginx,并且集成了LUA脚本,开发人员只需要简单的其提供了模块就可以实现相关的逻辑,而不再像之前,还需要在nginx中自己编写lua的脚本,再进行调用了。

  • 安装openRestry

1.添加仓库执行命令

 yum install yum-utils
 yum-config-manager --add-repo https://openresty.org/package/centos/openresty.repo

2.执行安装

yum install openresty

3.安装成功后 会在默认的目录如下:

/usr/local/openresty

修改/usr/local/openresty/nginx/conf/nginx.conf ,将配置文件使用的根设置为root,目的就是将来要使用lua脚本的时候 ,直接可以加载在root下的lua脚本。

#user nobody; 配置文件第一行原来为这样, 现改为下面的配置
user root root;

四. 实现缓存预热

实现思路:
(1)用户请求获取广告数据, 先从nginx缓存中读取, 若没有, 则从redis中读取
(2)监控mysql广告数据发生变化, 通知nginx进行数据更新, 同时将更新的json数据保存到redis, 保证缓存到最新的数据

        location /ad_read {
            content_by_lua_file /root/lua/ad_read.lua;
        }

路由将有/root/lua/ad_read.lua;脚本执行, 脚本的内容会从nginx本地或redis中获取缓存数据, openResty需要开启共享内存, 提供二级缓存功能, 减轻redis压力.
ad_read.lua脚本内容:

--设置响应头类型
ngx.header.content_type="application/json;charset=utf8"
--获取请求中的参数ID
local uri_args = ngx.req.get_uri_args();
local position = uri_args["position"];

--获取本地缓存
local cache_ngx = ngx.shared.dis_cache;
--根据ID 获取本地缓存数据
local adCache = cache_ngx:get('ad_cache_'..position);

if adCache == "" or adCache == nil then

    --引入redis库
    local redis = require("resty.redis");
    --创建redis对象
    local red = redis:new()
    --设置超时时间
    red:set_timeout(2000)
    --连接
    local ok, err = red:connect("192.168.225.128", 6379)
    --获取key的值
    local rescontent=red:get("ad_"..position)
    --输出到返回响应中
    ngx.say(rescontent)
    --关闭连接
    red:close()
    --将redis中获取到的数据存入nginx本地缓存
    cache_ngx:set('ad_cache_'..position, rescontent, 10*60);
else
    --nginx本地缓存中获取到数据直接输出
    ngx.say(adCache)
end
  • 修改nginx配置文件vi /usr/local/openresty/nginx/conf/nginx.conf ,http节点下添加配置:
#包含redis初始化模块
lua_shared_dict dis_cache 5m;  #共享内存开启
    1. mysql数据库发生变化, 调用http://192.168.225.128/ad_update?position=web_index_lb, 访问nginx服务器, nginx服务器同样将update路由到本地的lua脚本中进行执行,
      ad_update.lua脚本主要是查询mysql数据库, 将最新符合条件的数据缓存到redis中,完成缓存预热的功能

修改/usr/local/openresty/nginx/conf/nginx.conf文件:

 server {
        ....
        # 添加
        location /ad_update {
            content_by_lua_file /root/lua/ad_update.lua;
        }
        ....
         
    }

ad_update.lua

ngx.header.content_type="application/json;charset=utf8"
local cjson = require("cjson")
local mysql = require("resty.mysql")
local uri_args = ngx.req.get_uri_args()
local position = uri_args["position"]

local db = mysql:new()
db:set_timeout(1000)  
local props = {  
    host = "192.168.200.128",  
    port = 3306,  
    database = "changgou_business",  
    user = "root",  
    password = "123456"  
}

local res = db:connect(props)  
local select_sql = "select url,image from tb_ad where status ='1' and position='"..position.."' and start_time<= NOW() AND end_time>= NOW()"  
res = db:query(select_sql)  
db:close()  

local redis = require("resty.redis")
local red = redis:new()
red:set_timeout(2000)

local ip ="192.168.200.128"
local port = 6379
red:connect(ip,port)

red:set("ad_"..position,cjson.encode(res))
red:close()

ngx.say("{flag:true}")
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,875评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,569评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,475评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,459评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,537评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,563评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,580评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,326评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,773评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,086评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,252评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,921评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,566评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,190评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,435评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,129评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,125评论 2 352

推荐阅读更多精彩内容