nginx反向代理缓存

news/2024/10/17 22:13:50/

背景

nginx 一般用来做反向代理和负载均衡,将客户端请求发送到后端的 jetty,并将 jetty 的响应发送给客户端。后端的 jetty 通常不止一个,nginx 根据配置来选择其中一个 jetty,比较常见的选择策略是轮询。示意图如下

启动缓存支持以后,nginx 在将请求转发到 jetty 之前会在缓存里查找结果,如果缓存里存在匹配客户端请求的结果,则直接返回给客户端,不会将请求发送给 jetty;如果缓存没有命中,才会将请求发送给 jetty,接收到 jetty 返回的结果后,会将结果进行缓存以备客户端下次的请求。示意图如下:
在这里插入图片描述

缓存命中

在这里插入图片描述

缓存未命中

在这里插入图片描述

最简单的缓存设置

web 程序

首先我们有个简单的 web 程序,返回当前时间,如下

@Controller
public class TestController {@ResponseBody@RequestMapping("time.do")public Map<String, Long> time() {Map<String, Long> map = new HashMap<String, Long>();map.put("time", System.currentTimeMillis());return map;}
}

这个接口会返回当前时间,精确到毫秒,正常情况下每次刷新返回的值都应该是不一样的

nginx 配置

现在配置 nginx 的缓存,首先在 http 小节添加如下配置

http {...# 设置缓存的路径和其他参数# proxy_cache_path path [levels=levels] keys_zone=name:size [inactive=time] [max_size=size] [loader_files=number] [loader_sleep=time] [loader_threshold=time];# 缓存路径 /data/nginx/cache 缓存结构为 2 层,即该路径下会有 2 层子目录,缓存文件会保存在最下层子目录# 缓存的 key 会保存在名为 web_cache 的内存区域,该内存区域大小为 50 m# 10 分钟内缓存没有被访问就会过期# 缓存文件最多占用 1g 空间proxy_cache_path /data/nginx/cache levels=1:2 keys_zone=web_cache:50m inactive=10m max_size=1g;# proxy_cache_path 指令必须在 include 指令之前include vhosts/*.conf;...
}

接下来在 server 小节 的 location / 小节添加如下配置

location / {proxy_store off;proxy_redirect off;proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;proxy_set_header X-Real-IP $remote_addr;proxy_set_header Host $http_host;proxy_pass http://my-server/;# 缓存使用前面定义的内存区域proxy_cache web_cache;# 这个貌似没啥用proxy_ignore_headers   Expires Set-Cookie;# 对于 200 和 304 的响应码进行缓存,过期时间为 2 分钟,这会覆盖前面定义的 10 分钟过期时间proxy_cache_valid 200 304 2m;# 设置缓存的 key,这里用到了 nginx 的内嵌变量,表示用整个 url 作 keyproxy_cache_key  $scheme$proxy_host$request_uri;# 在返回的响应里添加响应头 X-Proxy-Cache,其值表示是否命中了缓存,貌似没啥用add_header X-Proxy-Cache $upstream_cache_status;# 返回的响应头里,设置过期时间为 10 分,貌似没啥用expires 10m;# 貌似没啥用add_header Access-Control-Allow-Origin "http://myserver.net";# 貌似没啥用add_header Access-Control-Allow-Credentials true;}

缓存特定的参数

有的参数每个客户端都不一样,而且该参数的值并不影响返回的结果,如果我们用整个 url 作为缓存的 key,那么 key 的数量就太多了,导致缓存的命中率非常低。
例如,用户手机的 imei 是唯一的,如果 key 包括了 imei,那就会导致每个用户都有一个缓存,这显然不是我们需要的
那么,能不能把某些参数排除在 key 之外呢?答案是肯定滴,看看下面这个 web 程序

@ResponseBody
@RequestMapping("/test/user.do")
public Map<String, String> user(String imei, String sn, String phone, String name) {Map<String, String> map = new HashMap<String, String>();map.put("imei", imei);map.put("sn", sn);map.put("phone", phone);map.put("name", name);return map;
}

只用 phone 和 name 参数来构造缓存的 key,可以如下配置

location / {proxy_store off;proxy_redirect off;proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;proxy_set_header X-Real-IP $remote_addr;proxy_set_header Host $http_host;proxy_pass http://my-server/;# 缓存使用前面定义的内存区域proxy_cache web_cache;# 对于 200 和 304 的响应码进行缓存,过期时间为 2 分钟,这会覆盖前面定义的 10 分钟过期时间proxy_cache_valid 200 304 2m;# 设置缓存的 key,只用 phone 和 name 构造 keyproxy_cache_key  $host$uri$is_args$arg_phone&$arg_name;}

这次把貌似没用的配置都移除了,要指定某些参数来构造缓存的 key,关键是以下的内嵌变量

  • $is_args
  • $arg_{name}

缓存特定的接口

有时候不是所有接口都需要被 nginx 缓存,或者缓存的 key 采用不同的规则,可以通过不同的 location 来配置
例如,下面的配置就只对以 /test/ 开头的接口做缓存

location / {proxy_store off;proxy_redirect off;proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;proxy_set_header X-Real-IP $remote_addr;proxy_set_header Host $http_host;proxy_pass http://my-server/;# 不缓存 proxy_cache 默认值就是 off,所以也可以省略这一行proxy_cache off
}
location /test/ {proxy_store off;proxy_redirect off;proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;proxy_set_header X-Real-IP $remote_addr;proxy_set_header Host $http_host;proxy_pass http://my-server/test/;# 缓存使用前面定义的内存区域proxy_cache web_cache;# 对于 200 和 304 的响应码进行缓存,过期时间为 2 分钟,这会覆盖前面定义的 10 分钟过期时间proxy_cache_valid 200 304 2m;# 设置缓存的 key,只用 phone 和 name 构造 keyproxy_cache_key  $host$uri$is_args$arg_phone&$arg_name;}

http://www.ppmy.cn/news/86626.html

相关文章

谷歌浏览器被2345劫持

方法1&#xff1a; 打开控制面板的卸载程序&#xff0c;搜索2345&#xff0c;把那个恶心的“安全组件-2345”卸载掉&#xff01;&#xff01; 这个方法比修改 host 以及注册表要好使地多&#xff01; 参考网址&#xff1a; 【小技巧】修复chrome被2345劫持 方法2&#xff1a; …

【利用AI让知识体系化】V8引擎相关知识

文章目录 I. 引言V8引擎的背景和概述 II. V8的设计和工作原理V8的整体设计V8的工作流程和运行机制V8在浏览器中的应用场景 III. 内存管理内存模型和内存管理策略垃圾回收机制和算法内存泄漏和内存优化 IV. JIT编译器JIT编译器的作用和优势V8的编译流程和编译器类型编译器优化技…

机器学习常识 4: 分类问题的训练与测试

摘要: 本贴以最为典型的分类问题为例, 描述训练与测试. 1. 基本概念 上午来了 60 个患者, 根据他们的各项检测指标 (即数据), 主治医生给出了诊断结论 (如是否患病, 以及患哪种病), 但不会告诉实习生诊断的方法. 实习生根据这 60 条数据, 归纳总结出了诊断模型 (方法), 这是一…

第三十七回:如何正确地显示BottomeSheet

文章目录 概念介绍显示方法使用showBottomSheet()方法使用showModalBottomSheet()方法 示例代码 我们在上一章回中介绍了 BottomSheet Widget相关的内容,本章回中将介绍 如何正确地显示BottomSheet Widget.闲话休提&#xff0c;让我们一起Talk Flutter吧。 概念介绍 我们在上…

Java技术接单

今天给大家介绍一个阶段性&#xff08;周期性&#xff09;能获取一定收益的Java技术接单群&#xff0c;分享给大家&#xff01;主要对搞Java的粉丝有帮助&#xff0c;因为可以赚点小钱&#xff0c;对Java技术的要求不高&#xff01; 那些感兴趣或者想直接加技术群的我给大家讲一…

快速理解会话跟踪技术Cookie和Session

文章目录 会话跟踪技术客户端会话跟踪技术Cookie服务端会话跟踪技术Session 会话跟踪技术 会话&#xff1a;客服端和服务端的多次请求与响应称为会话。 会话跟踪&#xff1a;服务器需要识别多次请求是否来自同一浏览器&#xff0c;在同一次会话多次请求中共享数据。 HTTP协议是…

C++中的继承、以及赋值兼容转换。

一、继承的概念及定义 继承可以使代码复用&#xff0c;允许在保持原有类特性的基础上进行扩展。 举个例子&#xff1a;就好比我现在要封装老师、学生等这些人的属性和方法&#xff0c;但是这些人都有重复的属性和方法&#xff0c;比如name、age、sex等等&#xff0c;那么我可…

虚拟化与云计算有什么区别?

两者核心理念都是从硬件中分离资源&#xff0c;从而创建可用的环境&#xff0c;所以很容易被混为一谈。 虚拟化有助于创建云计算&#xff0c;但它并非实现云计算的决定性技术&#xff0c;也可能只是其中的一种。你可以这样理解&#xff1a; 虚拟化是一种将功能与硬件分离的技…