有一个短链接跳转的sass系统,假设客户很多,在短链接进行跳转时肯定会用到redis这就涉及到了缓存穿透 缓存雪崩 缓存击穿等问题
短链接平台是一种在线服务,它将长的网址(URL)转换为更短的链接。这些短链接更便于分享,特别是在字符数有限的环境中,比如社交媒体平台。使用短链接平台不仅可以节省空间,还可以提供额外的功能,如点击统计、自定义短链接、以及访问控制等。 短链接的典型格式是由平台的域名加上一串字符组成,这串字符代表了原始的长链接。当用户点击这个短链接时,短链接平台会自动将用户重定向到原始的长链接所指向的网页。这个过程对用户来说是透明的,他们可能根本意识不到链接已经被转换和重定向了。 短链接平台的一些常见应用包括但不限于:
Redis作为一种常用的内存数据存储系统,经常被用作缓存来提高数据访问的速度和效率。然而,在使用Redis作为缓存时,可能会遇到几种典型的问题,包括缓存穿透、缓存雪崩和缓存击穿。这些问题都可能对系统的性能和稳定性产生负面影响。下面分别解释这三种情况:
缓存穿透是指查询一个数据库中不存在的数据。由于缓存是不命中的,每次查询都会穿过缓存去查询数据库。如果有大量这样的查询,数据库就会受到很大的压力。缓存穿透的一个典型场景是恶意用户故意查询不存在的数据,使得数据库压力增大。
解决办法:
缓存雪崩是指在某一个时间点,由于大量的缓存同时过期,导致原本应该命中缓存的请求都落到了数据库上,从而引发数据库瞬时压力过大。这种情况可能由缓存服务器重启或者大量缓存设置了相同的过期时间引起。
解决办法:
缓存击穿与缓存穿透不同,它是指缓存中有这个数据,但是已经过期,此时有大量并发请求这个数据。因为缓存没有命中,所有的请求都去数据库查询数据,然后重新设置到缓存中,这可能会对数据库造成巨大压力。
解决办法:
@Override
public void restoreUrl(String shortUri, ServletRequest request, ServletResponse response) throws IOException {
// 获取完整短链接
final String fullShortUrl = request.getServerName() + "/" + shortUri;
// 从缓存中获取短链接所对应的完整链接
String originalLink = stringRedisTemplate.opsForValue().get(String.format(RedisKeyConstant.GOTO_SHORT_LINK_KEY, fullShortUrl));
// 缓存存在的话直接进行短链接跳转
if (Opp.ofStr(originalLink).isPresent()) {
((HttpServletResponse) response).sendRedirect(originalLink);
return;
}
// 从布隆过滤器中查看有没有这个短链接
final boolean contains = shortUriCreateCachePenetrationBloomFilter.contains(fullShortUrl);
if (!contains){
// 不存在的话直接跳转自定义404界面
((HttpServletResponse) response).sendRedirect("/page/notfound");
return;
}
// 如果存在于布隆过滤器,可能存在误判。所以缓存中存放了一个数据库中短链接是否为null的
final String link = stringRedisTemplate.opsForValue().get(String.format(RedisKeyConstant.GOTO_NULL_SHORT_LINK_KEY, fullShortUrl));
// 如果为null的话还是直接跳转自定义404界面
if (Opp.ofStr(link).isPresent()) {
((HttpServletResponse) response).sendRedirect("/page/notfound");
return;
}
// 添加分布式锁
final RLock lock = redissonClient.getLock(String.format(RedisKeyConstant.LOCK_GOTO_SHORT_LINK_KEY, fullShortUrl));
lock.lock();
try {
// 加锁之后再去缓存中判断一次
originalLink = stringRedisTemplate.opsForValue().get(String.format(RedisKeyConstant.GOTO_SHORT_LINK_KEY, fullShortUrl));
if (Opp.ofStr(originalLink).isPresent()) {
// 如果存在直接跳转
((HttpServletResponse) response).sendRedirect(originalLink);
return;
}
// 如果不存在的话,去数据库中查询
final ShortLinkGotoDO shortLinkGotoDO = One.of(ShortLinkGotoDO::getFullShortUrl).eq(fullShortUrl).query();
if (shortLinkGotoDO == null) {
// 如果数据库不存在的话存放一个临时的空值,防止缓存穿透
stringRedisTemplate.opsForValue().set(String.format(RedisKeyConstant.GOTO_NULL_SHORT_LINK_KEY, fullShortUrl), "-",30 , TimeUnit.SECONDS);
((HttpServletResponse) response).sendRedirect("/page/notfound");
return;
}
// 从数据库获取完整短链接
final ShortLinkDO shortLinkDO = One.of(ShortLinkDO::getGid).eq(shortLinkGotoDO.getGid()).condition(w -> w.eq(ShortLinkDO::getFullShortUrl, fullShortUrl).eq(ShortLinkDO::getEnableStatus, 0)).query();
if (Opp.of(shortLinkDO).isPresent()) {
// 判断短链接是否已经过期
if (shortLinkDO.getValidDate() != null && shortLinkDO.getValidDate().before(new Date())) {
// 证明已经过期
stringRedisTemplate.opsForValue().set(String.format(RedisKeyConstant.GOTO_NULL_SHORT_LINK_KEY, fullShortUrl), "-", 30, TimeUnit.MINUTES);
((HttpServletResponse) response).sendRedirect("/page/notfound");
return;
}
// 如果数据库存在的话设置缓存到redis,并进行跳转
stringRedisTemplate.opsForValue()
.set(
String.format(RedisKeyConstant.GOTO_SHORT_LINK_KEY,
fullShortUrl),
shortLinkDO.getOriginUrl(),
LinkUtil.getLinkCacheValidTime(shortLinkDO.getValidDate()));
((HttpServletResponse) response).sendRedirect(shortLinkDO.getOriginUrl());
}
}finally {
lock.unlock();
}
}
对应的时序图
final String fullShortUrl = request.getServerName() + "/" + shortUri;
这行代码拼接了服务器的名称和短链接的唯一标识符shortUri来构成完整的短链接fullShortUrl。
String originalLink = stringRedisTemplate.opsForValue().get(String.format(RedisKeyConstant.GOTO_SHORT_LINK_KEY, fullShortUrl));
这行代码尝试从Redis缓存中获取短链接所对应的原始链接。这是为了减少对数据库的访问,提高响应速度。
if (Opp.ofStr(originalLink).isPresent()) {
((HttpServletResponse) response).sendRedirect(originalLink);
return;
}
如果缓存中存在原始链接,则直接重定向到原始链接,这一步骤帮助防止缓存击穿。
final boolean contains = shortUriCreateCachePenetrationBloomFilter.contains(fullShortUrl);
这行代码使用布隆过滤器检查短链接是否存在,这是为了防止缓存穿透,即防止恶意用户通过不断请求不存在的短链接来使得服务直接访问数据库。
if (!contains) {
((HttpServletResponse) response).sendRedirect("/page/notfound");
return;
}
如果布隆过滤器判断短链接不存在,则直接重定向到404页面,避免了对数据库的无效访问。
final String link = stringRedisTemplate.opsForValue().get(String.format(RedisKeyConstant.GOTO_NULL_SHORT_LINK_KEY, fullShortUrl));
这行代码检查是否缓存了一个表示数据库中没有对应记录的空值,这是为了处理布隆过滤器的误判。
if (Opp.ofStr(link).isPresent()) {
((HttpServletResponse) response).sendRedirect("/page/notfound");
return;
}
如果缓存中存储了一个表示短链接在数据库中不存在的值,则直接重定向到404页面。
final RLock lock = redissonClient.getLock(String.format(RedisKeyConstant.LOCK_GOTO_SHORT_LINK_KEY, fullShortUrl));
这行代码为当前操作的短链接添加了一个分布式锁,这是为了防止缓存击穿,即在缓存失效的瞬间,大量的并发请求直接打到数据库。
lock.lock();
try {
originalLink = stringRedisTemplate.opsForValue().get(String.format(RedisKeyConstant.GOTO_SHORT_LINK_KEY, fullShortUrl));
if (Opp.ofStr(originalLink).isPresent()) {
((HttpServletResponse) response).sendRedirect(originalLink);
return;
}
这部分代码首先对短链接加锁,然后再次检查缓存,如果这时候缓存中存在原始链接,则直接重定向,这可以处理高并发下的缓存击穿问题。
final ShortLinkGotoDO shortLinkGotoDO = One.of(ShortLinkGotoDO::getFullShortUrl).eq(fullShortUrl).query();
如果缓存中没有找到原始链接,代码会继续从数据库查询。这里使用了某种ORM框架的查询语法来获取短链接对应的数据对象。
if (shortLinkGotoDO == null) {
stringRedisTemplate.opsForValue().set(String.format(RedisKeyConstant.GOTO_NULL_SHORT_LINK_KEY, fullShortUrl), "-",30 , TimeUnit.SECONDS);
((HttpServletResponse) response).sendRedirect("/page/notfound");
return;
}
如果数据库中也不存在该短链接,则在缓存中设置一个短期的空值并重定向到404页面,这是为了防止缓存穿透。
final ShortLinkDO shortLinkDO = One.of(ShortLinkDO::getGid).eq(shortLinkGotoDO.getGid()).condition(w -> w.eq(ShortLinkDO::getFullShortUrl, fullShortUrl).eq(ShortLinkDO::getEnableStatus, 0)).query();
这行代码进一步查询获取短链接的详细信息。
if (Opp.of(shortLinkDO).isPresent()) {
if (shortLinkDO.getValidDate() != null && shortLinkDO.getValidDate().before(new Date())) {
stringRedisTemplate.opsForValue().set(String.format(RedisKeyConstant.GOTO_NULL_SHORT_LINK_KEY, fullShortUrl), "-", 30, TimeUnit.MINUTES);
((HttpServletResponse) response).sendRedirect("/page/notfound");
return;
}
stringRedisTemplate.opsForValue()
.set(
String.format(RedisKeyConstant.GOTO_SHORT_LINK_KEY,
fullShortUrl),
shortLinkDO.getOriginUrl(),
LinkUtil.getLinkCacheValidTime(shortLinkDO.getValidDate()));
((HttpServletResponse) response).sendRedirect(shortLinkDO.getOriginUrl());
}
如果查询到短链接且未过期,则更新缓存并重定向到原始链接,这样可以防止后续的缓存穿透和击穿问题。
} finally {
lock.unlock();
}
最后释放分布式锁,以允许其他线程处理其他短链接。