Java后端實(shí)現(xiàn)短鏈接生成功能
生成的邏輯
短鏈接生成的核心邏輯主要包括接收長(zhǎng)鏈接輸入、生成唯一的短碼、存儲(chǔ)映射關(guān)系以及生成短鏈接。然后,通過(guò)重定向機(jī)制實(shí)現(xiàn)短鏈接的訪問(wèn)跳轉(zhuǎn)。為了保證系統(tǒng)的性能,短鏈接服務(wù)會(huì)涉及哈希算法、數(shù)據(jù)庫(kù)存儲(chǔ)優(yōu)化和緩存等技術(shù)。
短鏈接生成主要在于把指定的接口和參數(shù)實(shí)現(xiàn)加密生成比較短的字符串(這里注意一點(diǎn)不要加入中文去生成,加入中文會(huì)提示異常),再進(jìn)行拼接通過(guò)指定的域名或者ip實(shí)現(xiàn)鏈接的跳轉(zhuǎn),如:http://你的域名活ip地址<加密后的字符> 就像 http://127.0.0.1:9001/37kRU 這樣,請(qǐng)求的時(shí)候會(huì)去匹配指定的接口再通過(guò)重定向去找到對(duì)應(yīng)的地址
實(shí)現(xiàn)目的
Java短鏈接服務(wù)是一個(gè)基于Java編程語(yǔ)言開(kāi)發(fā)的應(yīng)用程序,能夠?qū)㈤L(zhǎng)網(wǎng)址轉(zhuǎn)換為短網(wǎng)址。它的主要作用包括:
- URL縮短:將用戶提供的長(zhǎng)網(wǎng)址轉(zhuǎn)換為簡(jiǎn)短、易于分享的鏈接。通常,長(zhǎng)網(wǎng)址包含大量無(wú)意義的字符,縮短后便于在社交媒體、短信等平臺(tái)上快速分享。例如,將 www.example.com/some/very/l… 轉(zhuǎn)換為 short.ly/abc123
- 節(jié)省空間:短網(wǎng)址比長(zhǎng)網(wǎng)址占用更少的字符,適合字符數(shù)有限制的平臺(tái)(如短信)或者需要在有限空間內(nèi)展示鏈接的場(chǎng)合
- 跟蹤和分析:通過(guò)生成的短鏈接,可以進(jìn)行訪問(wèn)統(tǒng)計(jì)與分析,跟蹤短鏈接被點(diǎn)擊的次數(shù)、用戶來(lái)源、地理位置等數(shù)據(jù)。這有助于了解用戶行為、市場(chǎng)趨勢(shì)等
- 優(yōu)化用戶體驗(yàn):提供簡(jiǎn)潔的鏈接,用戶更容易記住并分享,尤其是在電子郵件、廣告、文檔等地方
- 提高鏈接安全性:短鏈接服務(wù)可以進(jìn)行重定向管理,確保用戶訪問(wèn)的是正確和安全的目標(biāo)網(wǎng)址,避免惡意重定向和釣魚(yú)攻擊
- 鏈接有效期管理:有些短鏈接服務(wù)允許設(shè)置鏈接的有效期,鏈接在過(guò)期后將不再有效,增強(qiáng)了鏈接的管理性和控制性
首先引入依賴
使用SpringBoot項(xiàng)目構(gòu)建
<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>
實(shí)現(xiàn)短鏈接服務(wù)
新增短鏈接工具類
復(fù)制其他博主工具類,也可自己實(shí)現(xiàn),主要在于加命生成短地址
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);
}
}
新增短鏈接生成地址和訪問(wèn)地址
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: 必須描述類做什么事情, 實(shí)現(xiàn)什么功能
*/
@Controller
public class UrlController {
/**
* 存儲(chǔ)長(zhǎng)鏈接和短鏈接的映射關(guān)系(亦可存儲(chǔ)在其他中間件中,如: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;
}
/**
* 短鏈接點(diǎn)擊跳轉(zhuǎn)
* @param shortURL
*/
@GetMapping("/{shortURL}")
public String redirect(@PathVariable String shortURL) {
if (URL_MAP.containsKey(shortURL)) {
return "redirect:" + URL_MAP.get(shortURL);
} else {
return "redirect:/";
}
}
/**
* 發(fā)送郵件
*/
@GetMapping("/sendMeg")
@ResponseBody
public String sendMeg(String email, String msg){
return "發(fā)送成功:" + email + ":" + msg;
}
}
1.先生成短鏈接

2.訪問(wèn)后端地址ip+加密字符,這里為:http://127.0.0.1:9001/s37kRU3.在瀏覽器請(qǐng)求后會(huì)自動(dòng)跳轉(zhuǎn)輸出如下截圖內(nèi)容

至此短鏈接生成和訪問(wèn)都能正常使用,其中生成內(nèi)容和對(duì)應(yīng)的接口則需要根據(jù)自己的需求去設(shè)定。
注意點(diǎn)
在使用本案例生成短鏈接和訪問(wèn)短鏈接時(shí),需要注意控制層注解是否是`@Controller`或者`@RestController`
這兩個(gè)注解對(duì)應(yīng)的訪問(wèn)短鏈接跳轉(zhuǎn)是不一樣的,原因如下:
@RestController 中使用返回字符串 "redirect:" 不起作用,那是因?yàn)?@RestController 默認(rèn)是返回?cái)?shù)據(jù)的響應(yīng)而非視圖,所以 Spring 并不會(huì)像在 @Controller 中那樣處理 "redirect:" 語(yǔ)法,所以當(dāng)使用的是@RestController注解控制層時(shí),跳轉(zhuǎn)接口需進(jìn)行改造,解決方案如下:
@RestController
public class UrlController {
/**
* 你的跳轉(zhuǎn)地址ip或者域名
*/
private final String BASE_URL = "http://127.0.0.1:9001/";
/**
* 存儲(chǔ)長(zhǎng)鏈接和短鏈接的映射關(guān)系
*/
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;
}
/**
* 短鏈接點(diǎn)擊跳轉(zhuǎn)
* @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("/");
}
}
/**
* 發(fā)送郵件
*/
@PostMapping("/sendMeg")
public String sendMeg(String email, String msg){
return "發(fā)送成功:" + email + ":" + msg;
}
}
如果不改造的會(huì)直接輸出:redirect:http://127.0.0.1:9001/sendMeg?email=2281077633@qq.com&msg=123456 就是因?yàn)?code>@RestController不能進(jìn)行視圖解析
結(jié)論:
- 如果使用
@Controller,返回"redirect:"是有效的,可以讓 Spring 自動(dòng)處理視圖重定向 - 如果使用
@RestController,需要手動(dòng)通過(guò)response.sendRedirect()來(lái)觸發(fā) HTTP 重定向
你可以根據(jù)自己的需求選擇合適的方式。如果你在處理的是 RESTful 接口,response.sendRedirect() 更合適;如果是傳統(tǒng)的 Spring MVC 頁(yè)面控制器,則使用 redirect: 更為簡(jiǎn)潔
到此這篇關(guān)于Java后端實(shí)現(xiàn)短鏈接生成功能的文章就介紹到這了,更多相關(guān)Java短鏈接內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
一文教你掌握J(rèn)ava如何實(shí)現(xiàn)判空
實(shí)際項(xiàng)目中我們會(huì)有很多地方需要判空校驗(yàn),如果不做判空校驗(yàn)則可能產(chǎn)生NullPointerException異常。所以本文小編為大家整理了Java中幾個(gè)常見(jiàn)的判空方法,希望對(duì)大家有所幫助2023-04-04
一文詳解SpringBoot?Redis多數(shù)據(jù)源配置
Spring?Boot默認(rèn)只允許一種?Redis?連接池配置,且配置受限于?Lettuce?包,不夠靈活,所以本文將為大家介紹如何自定義Redis配置方案實(shí)現(xiàn)多數(shù)據(jù)源支持,需要的可以參考下2024-11-11
springboot項(xiàng)目test文件夾下帶main方法的類不能運(yùn)行問(wèn)題
這篇文章主要介紹了springboot項(xiàng)目test文件夾下帶main方法的類不能運(yùn)行問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-11-11
java基礎(chǔ)之字符串編碼知識(shí)點(diǎn)總結(jié)
這篇文章主要介紹了java基礎(chǔ)之字符串編碼總結(jié),文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java基礎(chǔ)的小伙伴們有很好的幫助,要的朋友可以參考下2021-04-04
Java核心教程之常見(jiàn)時(shí)間日期的處理方法
這篇文章主要給大家介紹了關(guān)于Java核心教程之常見(jiàn)時(shí)間日期的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02
通過(guò)實(shí)例了解cookie機(jī)制特性及使用方法
這篇文章主要介紹了通過(guò)實(shí)例了解cookie機(jī)制特性及使用方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-09-09

