教你怎么实现缩短网址功能

本文将简单介绍,如何去完成一个缩短网址的功能。javascript

Node.js + MySQL + Redis版本的源码地址:githubhtml

Demo地址:www.ecool.fun/shortLink前端

文章阅读大概须要8分钟。java

什么是短连接

短连接,通俗来讲,就是将长的URL网址,经过程序计算等方式,转换为简短的网址字符串。git

你们常常能够从微博或者各种营销短信中,看到短连接,形式通常相似于 t.cn/xxxxxx,点击后,就能跳转到对应的页面。github

早期短连接普遍应用于图片上传网站,经过缩短网址URL连接字数,达到减小代码字符串的目的。更便于使用者引用网址,写入代码中,节省字符数空间。常见于网店图片分类的使用,因有字符限制,运用短连接,达到外链图片的目的,自微博盛行以来,在微博字数有限的特点下,短连接也盛行于微博网站,以节省字数,给博主发布更多文字的空间。算法

将长连接转成短连接,通常是为了方便记忆或者传播。数据库

须要完成的功能

从上面的介绍中,咱们得出,缩短网址须要完成如下两个功能点:segmentfault

  • 将长网址缩短成短连接
  • 点击生成短连接,能正常跳转到原来的长网址页面

全流程设计

其实上述功能点的原理很简单,简单描述一下:缓存

  • 用户输入长网址,服务端接收后进行处理,并根据长网址的内容,生成一个短码,并将映射关系进行存储。而后根据短码拼接出短连接,返回给用户;
  • 用户点击短连接,服务器端根据短连接中的短码,查找到对应的长网址,并302跳转到对应的页面。

知识点:为何要使用302跳转,而不是301跳转呢?

301是永久重定向,302是临时重定向。短地址一经生成就不会变化,因此用301是符合http语义的。可是若是用了301, Google,百度等搜索引擎,搜索的时候会直接展现真实地址,那咱们就没法统计到短地址被点击的次数了,也没法收集用户的Cookie, User Agent 等信息,这些信息能够用来作不少有意思的大数据分析,也是短网址服务商的主要盈利来源。

引自知乎-武林的回答,原文连接

整个流程的设计以下图所示:

短网址全流程

能够看到,我用到了MySQLRedis来存储长网址和短码之间的映射关系。

MySQL想必你们都能理解,可是为何要用 Redis 呢,直接用数据库不就行了吗?

这个主要是考虑到生成短连接,在投放以后的访问量会比较大,使用 Redis 缓存后,能有效下降数据库的压力。

生成短码的方法

经过上面的全流程设计,会发现主要的问题就是如何经过长网址,去生成对应的短码。

短码通常是由 [a - z, A - Z, 0 - 9] 这62 个字母或数字组成,短码的长度也能够自定义,但通常不超过8位。比较经常使用的都是6位,6位的短码已经能有568亿种的组合:(26+26+10)^6 = 56800235584,已知足绝大多数的使用场景。

目前比较流行的生成短码方法有:自增id摘要算法普通随机数

  • 自增id

该方法是一种无碰撞的方法,原理是,每新增一个短码,就在上次添加的短码id基础上加1,而后将这个10进制的id值,转化成一个62进制的字符串。

通常利用数据表中的自增id来完成:每次先查询数据表中的自增id最大值max,那么须要插入的长网址对应自增id值就是 max+1,将max+1转成62进制便可获得短码。

可是短码 id 是从一位长度开始递增,短码的长度不固定,不过能够用 id 从指定的数字开始递增的方式来处理,确保全部的短码长度都一致。同时,生成的短码是有序的,可能会有安全的问题,能够将生成的短码id,结合长网址等其余关键字,进行md5运算生成最后的短码。

10进制转成62进制的具体实现:

function string10to62(number) {
    const chars = '0123456789abcdefghigklmnopqrstuvwxyzABCDEFGHIGKLMNOPQRSTUVWXYZ';
    const charsArr = chars.split('');
    const radix = chars.length;
    let qutient = +number;
    let arr = [];
    do{
        let mod = qutient % radix;
        qutient = (qutient - mod) / radix;
        arr.unshift(charsArr[mod]);
    }while(qutient);
    return arr.join('');
}
复制代码
  • 摘要算法

摘要算法又称哈希算法,它表示输入任意长度的数据,输出固定长度的数据。相同的输入数据始终获得相同的输出,不一样的输入数据尽可能获得不一样的输出。

算法思路:

一、将长网址经过 md5 运算,生成 32 字符的 hex string,分为 4 段,每段 8 个字符;

二、对这四段循环处理,取 8 个字节,将其当作 16 进制串,并与 0x3fffffff(30位1) 与操做,即超过 30 位的忽略处理;

三、这 30 位分红 6 段,每 5 位的数字做为字母表的索引取得特定字符,依次进行得到 6 位字符串。

四、总的 md5 串能够得到 4 个 6 位串,取里面的任意一个就可做为这个长网址的短连接 url 地址。

虽然概率很小,可是该方法依然存在碰撞的可能性,解决冲突会比较麻烦。不过该方法生成的短码位数,是固定的,也不存在连续生成的短码有序的状况。

  • 普通随机数

该方法是从62个字符串中随机取出一个6位短码的组合,而后去数据库中查询该短码是否已存在。若是已存在,就继续循环该方法从新获取短码,不然就直接返回。

该方法是最简单的一种实现,不过因为Math.round()方法生成的随机数属于伪随机数,碰撞的可能性也不小。在数据比较多的状况下,可能会循环不少次,才能生成一个不冲突的短码。

具体实现:

// 获取惟一的Link
async getShortLink() {
    const shortLink = this.generateShortLink();

    // 查询数据库中是否存在该连接,若是存在,就直接返回
    const searchResult = await this.searchByLinkInMySQL(shortLink);

    if (searchResult && searchResult.length > 0) {
    // 若是shortLink已经存在,就遍历从新生成
        return this.getShortLink();
    }
    return shortLink;

}

// 生成随机的Link
generateShortLink() {
    let str = '';
    const arr = [
        '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',
    ];

    for (let i = 0; i < 6; i++) {
        const pos = Math.round(Math.random() * (arr.length - 1));
        str += arr[pos];
    }
    return str;
}
复制代码

综上,比较推荐使用第一种方法来实现短码的生成。

广告时间

最后,欢迎你们star咱们的人人贷大前端团队博客,全部的文章还会同步更新到知乎专栏掘金帐号,咱们每周都会分享几篇高质量的大前端技术文章。

参考文章

相关文章
相关标签/搜索