生成的逻辑
短链接生成的核心逻辑主要包括接收长链接输入、生成唯一的短码、存储映射关系以及生成短链接。然后,通过重定向机制实现短链接的访问跳转。为了保证系统的性能,短链接服务会涉及哈希算法、数据库存储优化和缓存等技术。
短链接生成主要在于把指定的接口和参数实现加密生成比较短的字符串(这里注意一点不要加入中文去生成,加入中文会提示异常),再进行拼接通过指定的域名或者ip实现链接的跳转,如:http://你的域名活ip地址<加密后的字符>
就像 http://127.0.0.1:9001/37kRU
这样,请求的时候会去匹配指定的接口再通过重定向去找到对应的地址
实现目的
Java短链接服务是一个基于Java编程语言开发的应用程序,能够将长网址转换为短网址。它的主要作用包括:
- URL缩短:将用户提供的长网址转换为简短、易于分享的链接。通常,长网址包含大量无意义的字符,缩短后便于在社交媒体、短信等平台上快速分享。例如,将 https://www.example.com/some/very/long/url/that/is/difficult/to/remember 转换为 https://short.ly/abc123
- 节省空间:短网址比长网址占用更少的字符,适合字符数有限制的平台(如短信)或者需要在有限空间内展示链接的场合
- 跟踪和分析:通过生成的短链接,可以进行访问统计与分析,跟踪短链接被点击的次数、用户来源、地理位置等数据。这有助于了解用户行为、市场趋势等
- 优化用户体验:提供简洁的链接,用户更容易记住并分享,尤其是在电子邮件、广告、文档等地方
- 提高链接安全性:短链接服务可以进行重定向管理,确保用户访问的是正确和安全的目标网址,避免恶意重定向和钓鱼攻击
- 链接有效期管理:有些短链接服务允许设置链接的有效期,链接在过期后将不再有效,增强了链接的管理性和控制性
首先引入依赖
使用SpringBoot项目构建
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-all</artifactId>
<version>5.8.31</version>
</dependency>
实现短链接服务
新增短链接工具类
复制其他博主工具类,也可自己实现,主要在于加命生成短地址
package com.boot.utils;
import cn.hutool.core.lang.hash.MurmurHash;
/**
* @date: 2025/3/7 15:32
* @description: 生成短链接的工具类
*/
public class HashUtils {
private static char[] CHARS = new char[]{
'0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z',
'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z'
};
private static int SIZE = CHARS.length;
private static String convertDecToBase62(long num) {
StringBuilder sb = new StringBuilder();
while (num > 0) {
int i = (int) (num % SIZE);
sb.append(CHARS[i]);
num /= SIZE;
}
return sb.reverse().toString();
}
public static String hashToBase62(String str) {
int i = MurmurHash.hash32(str);
long num = i < 0 ? Integer.MAX_VALUE - (long) i : i;
return convertDecToBase62(num);
}
}
新增短链接生成地址和访问地址
package com.boot.controller;
import com.boot.utils.HashUtils;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.*;
import java.util.HashMap;
import java.util.Map;
/**
* @author: Re
* @date: 2025/3/7 15:34
* @description: 必须描述类做什么事情, 实现什么功能
*/
@Controller
public class UrlController {
/**
* 存储长链接和短链接的映射关系(亦可存储在其他中间件中,如:Redis,MySQL等)
*/
private static Map<String, String> URL_MAP = new HashMap<>();
/**
* 生成短链接
* @param longUrl
*/
@PostMapping("/generate")
@ResponseBody
public String generate(@RequestParam String longUrl) {
String shortUrl = HashUtils.hashToBase62(longUrl);
URL_MAP.put(shortUrl, longUrl);
return shortUrl;
}
/**
* 短链接点击跳转
* @param shortURL
*/
@GetMapping("/{shortURL}")
public String redirect(@PathVariable String shortURL) {
if (URL_MAP.containsKey(shortURL)) {
return "redirect:" + URL_MAP.get(shortURL);
} else {
return "redirect:/";
}
}
/**
* 发送邮件
*/
@GetMapping("/sendMeg")
@ResponseBody
public String sendMeg(String email, String msg){
return "发送成功:" + email + ":" + msg;
}
}
1.先生成短链接
2.访问后端地址ip+加密字符,这里为:http://127.0.0.1:9001/s37kRU
3.在浏览器请求后会自动跳转输出如下截图内容
至此短链接生成和访问都能正常使用,其中生成内容和对应的接口则需要根据自己的需求去设定。
注意点:
在使用本案例生成短链接和访问短链接时,需要注意控制层注解是否是`@Controller`或者`@RestController`
这两个注解对应的访问短链接跳转是不一样的,原因如下:
@RestController
中使用返回字符串 “redirect:” 不起作用,那是因为 @RestController 默认是返回数据的响应而非视图,所以 Spring 并不会像在 @Controller
中那样处理 “redirect:” 语法,所以当使用的是@RestController
注解控制层时,跳转接口需进行改造,解决方案如下:
@RestController
public class UrlController {
/**
* 你的跳转地址ip或者域名
*/
private final String BASE_URL = "http://127.0.0.1:9001/";
/**
* 存储长链接和短链接的映射关系
*/
private static Map<String, String> URL_MAP = new HashMap<>();
/**
* 生成短链接
* @param longUrl
*/
@PostMapping("/generate")
public String generate(@RequestParam String longUrl) {
String shortUrl = HashUtils.hashToBase62(longUrl);
URL_MAP.put(shortUrl, BASE_URL + longUrl);
return shortUrl;
}
/**
* 短链接点击跳转
* @param shortURL
*/
@GetMapping("/{shortURL}")
public void redirect(@PathVariable String shortURL, HttpServletResponse response) throws IOException {
if (URL_MAP.containsKey(shortURL)) {
response.sendRedirect(URL_MAP.get(shortURL));
} else {
response.sendRedirect("/");
}
}
/**
* 发送邮件
*/
@PostMapping("/sendMeg")
public String sendMeg(String email, String msg){
return "发送成功:" + email + ":" + msg;
}
}
如果不改造的会直接输出:redirect:http://127.0.0.1:9001/sendMeg?email=2281077633@qq.com&msg=123456 就是因为@RestController
不能进行视图解析
结论:
- 如果使用
@Controller
,返回"redirect:"
是有效的,可以让 Spring 自动处理视图重定向 - 如果使用
@RestController
,需要手动通过response.sendRedirect()
来触发 HTTP 重定向
你可以根据自己的需求选择合适的方式。如果你在处理的是 RESTful 接口,response.sendRedirect()
更合适;如果是传统的 Spring MVC 页面控制器,则使用 redirect:
更为简洁