短 URL 生成器设计:百亿短 URL 怎样做到无冲突?

Java全能学习面试指南:https://javaxiaobear.cn

我们先来看看,当高并发遇到海量数据处理时的架构。在社交媒体上,人们经常需要分享一些 URL,但是有些 URL 可能会很长,比如:

https://time.geekbang.org/hybrid/pvip?utm_source=geektime-pcdiscover-banner&utm_term=geektime-pc-discover-banner

这样长的 URL 显然体验并不友好。我们期望分享的是一些更短、更易于阅读的短 URL,比如像 http://javaxiaobear.cn/study-tutorial/这样的。当用户点击这个短 URL 的时候,可以重定向访问到原始的链接地址。为此我们将设计开发一个短 URL 生成器,产品名称是“Fuxi(伏羲)”。

我们预计 Fuxi 需要管理的短 URL 规模在百亿级别,并发吞吐量达到数万级别。这个量级的数据对应的存储方案是什么样的?用传统的关系数据库存储,还是有其他更简单的办法?此外,如何提升系统的并发处理能力呢?这些是我们今天要重点考虑的问题。

1、需求分析

短 URL 生成器,也称作短链接生成器,就是将一个比较长的 URL 生成一个比较短的URL,当浏览器通过短 URL 生成器访问这个短 URL 的时候,重定向访问到原始的长 URL目标服务器,访问时序图如下:

image-20231127210345356

对于需要展示短 URL 的应用程序,由该应用调用短 URL 生成器生成短 URL,并将该短URL 展示给用户,用户在浏览器中点击该短 URL 的时候,请求发送到短 URL 生成器(短URL 生成器以 HTTP 服务器的方式对外提供服务,短 URL 域名指向短 URL 生成器),短URL 生成器返回 HTTP 重定向响应,将用户请求重定向到最初的原始长 URL,浏览器访问长 URL 服务器,完成请求服务。

1、短 URL 生成器的用例图

image-20231127210845791

  1. 用户 client 程序可以使用短 URL 生成器 Fuxi 为每个长 URL 生成唯一的短 URL,并存储起来。
  2. 用户可以访问这个短 URL,Fuxi 将请求重定向到原始长 URL。
  3. 生成的短 URL 可以是 Fuxi 自动生成的,也可以是用户自定义的。用户可以指定一个长URL 对应的短 URL 内容,只要这个短 URL 还没有被使用。
  4. 管理员可以通过 web 后台检索、查看 Fuxi 的使用情况。
  5. 短 URL 有有效期(2 年),后台定时任务会清理超过有效期的 URL,以节省存储资源,同时回收短 URL 地址链接资源。

2、性能指标估算

Fuxi 的存储容量和并发量估算如下:

预计每月新生成短 URL 5 亿条,短 URL 有效期 2 年,那么总 URL 数量 120 亿

5亿 × 12月 × 2年 = 120亿
1、存储空间

每条短 URL 数据库记录大约 1KB,那么需要总存储空间 12TB(不含数据冗余备份):

120亿 × 1KB = 12TB

2、吞吐量

每条短 URL 平均读取次数 100 次,那么平均访问吞吐量(每秒访问次数)2 万:

(5亿 × 100) ÷ (30 × 24 × 60 × 60) ≈ 20000

一般系统高峰期访问量是平均访问量的 2 倍,因此系统架构需要支持的吞吐能力应为 4万。

3、网络带宽

短 URL 的重定向响应包含长 URL 地址内容,长 URL 地址大约 500B,HTTP 响应头其他内容大约 500B,所以每个响应 1KB,高峰期需要的响应网络带宽 40MB

4万(每秒)次请求 × 1KB = 40MB

Fuxi 的短 URL 长度估算如下:

  1. 短 URL 采用 Base64 编码,如果短 URL 长度是 7 个字符的话,大约可以编码 4 万亿个短URL
    6 4 7 = 4 万亿 64^7=4万亿 647=4万亿

  2. 如果短 URL 长度是 6 个字符的话,大约可以编码 680 亿个短 URL。
    6 4 6 = 680 亿 64^6=680亿 646=680亿

按我们前面评估,总 URL 数 120 亿,6 个字符的编码就可以满足需求。因此 Fuxi 的短URL 编码长度 6 个字符,形如 http://javaxiaobear.cn/ScW4dt 。

3、非功能需求

  1. 系统需要保持高可用,不因为服务器、数据库宕机而引起服务失效。
  2. 系统需要保持高性能,服务端 80% 请求响应时间应小于 5ms,99% 请求响应时间小于20ms,平均响应时间小于 10ms。
  3. 短 URL 应该是不可猜测的,即不能猜测某个短 URL 是否存在,也不能猜测短 URL 可能对应的长 URL 地址内容。

2、概要设计

短 URL 生成器的设计核心就是短 URL 的生成,即长 URL 通过某种函数,计算得到一个 6个字符的短 URL。短 URL 有几种不同的生成算法。

1、单项散列函数生成短 URL

通常的设计方案是,将长 URL 利用 MD5 或者 SHA256 等单项散列算法,进行 Hash 计算,得到 128bit 或者 256bit 的 Hash 值。然后对该 Hash 值进行 Base64 编码,得到 22个或者 43 个 Base64 字符,再截取前面的 6 个字符,就得到短 URL 了,如图

image-20231128162918786

但是这样得到的短 URL,可能会发生 Hash 冲突,即不同的长 URL,计算得到的短 URL是相同的(MD5 或者 SHA256 计算得到的 Hash 值几乎不会冲突,但是 Base64 编码后再截断的 6 个字符有可能会冲突)。所以在生成的时候,需要先校验该短 URL 是否已经映射为其他的长 URL,如果是,那么需要重新计算(换单向散列算法,或者换 Base64 编码截断位置)。重新计算得到的短 URL 依然可能冲突,需要再重新计算。但是这样的冲突处理需要多次到存储中查找 URL,无法保证 Fuxi 的性能要求。

2、自增长短 URL

一种免冲突的算法是用自增长自然数来实现,即维持一个自增长的二进制自然数,然后将该自然数进行 Base64 编码即可得到一系列的短 URL。这样生成的的短 URL 必然唯一,而且还可以生成小于 6 个字符的短 URL,比如自然数 0 的 Base64 编码是字符“A”,就可以用 http://javaxiaobear.cn/A 作为短 URL。

但是这种算法将导致短 URL 是可猜测的,如果某个应用在某个时间段内生成了一批短URL,那么这批短 URL 就会集中在一个自然数区间内。只要知道了其中一个短 URL,就可以通过自增(以及自减)的方式请求访问其他 URL。Fuxi 的需求是不允许短 URL 可预测。

3、预生成短 URL

因此,Fuxi 采用预生成短 URL 的方案。即预先生成一批没有冲突的短 URL 字符串,当外部请求输入长 URL 需要生成短 URL 的时候,直接从预先生成好的短 URL 字符串池中获取一个即可。

预生成短 URL 的算法可以采用随机数来实现,6 个字符,每个字符都用随机数产生(用0~63 的随机数产生一个 Base64 编码字符)

为了避免随机数产生的短 URL 冲突,需要在预生成的时候检查该 URL 是否已经存在(用布隆过滤器检查)。因为预生成短 URL 是

离线的,所以这时不会有性能方面的问题。事实上,Fuxi 在上线之前就已经生成全部需要的 144 亿条短 URL 并存储在文件系统中(预估需要短 URL120 亿,Fuxi 预生成的时候进行了 20% 的冗余,即 144 亿。)

代码实现如下:

import java.util.Random;
public class RandomBase64 {
    public static void main(String[] args) {
        StringBuilder shortUrl = new StringBuilder();
        Random random = new Random();
        for (int i = 0; i < 6; i++) {
            int randomNumber = random.nextInt(64);
            shortUrl.append(Base64Encoder(randomNumber));
        }
        System.out.println("生成的短URL: " + shortUrl.toString());
    }
    
    public static char Base64Encoder(int number) {
        char[] base64Chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".toCharArray();
        return base64Chars[number];
    }
}

4、Fuxi 的整体部署模型

Fuxi 的业务逻辑比较简单,相对比较有挑战的就是高并发的读请求如何处理、预生成的短URL 如何存储以及访问。高并发访问主要通过负载均衡与分布式缓存解决,而海量数据存储则通过 HDFS 以及 HBase 来完成。具体架构图如下。

image-20231128164131002

系统调用可以分成两种情况,一种是用户请求生成短 URL 的过程;另一种是用户访问短URL,通过 Fuxi 跳转到长 URL 的过程。

对于用户请求生成短 RUL 的过程,在短 URL 系统 Fuxi 上线前,已经通过随机数算法预生成 144 亿条短 URL 并将其存储在 HDFS 文件系统中。系统上线运行后,应用程序请求生成短 URL 的时候(即输入长 URL,请求返回短 URL),请求通过负载均衡服务器被发送到短 URL 服务器集群,短 URL 服务器再通过负载均衡服务器调用短 URL 预加载服务器集群。

短 URL 预加载服务器此前已经从短 URL 预生成文件服务器(HDFS)中加载了一批短URL 存放在自己的内存中,这时,只需要从内存中返回一个短 URL 即可,同时将短 URL与长 URL 的映射关系存储在 HBase 数据库中,时序图如下。

image-20231128220922513

对于用户通过客户端请求访问短 URL 的过程(即输入短 URL,请求返回长 URL),请求通过负载均衡服务器发送到短 URL 服务器集群,短 URL 服务器首先到缓存服务器中查找是否有该短 URL,如果有,立即返回对应的长 URL,短 URL 生成服务器构造重定向响应返回给客户端应用。

如果缓存没有用户请求访问的短 URL,短 URL 服务器将访问 HBase 短 URL 数据库服务器集群。如果数据库中存在该短 URL,短 URL 服务器会将该短 URL 写入缓存服务器集群,并构造重定向响应返回给客户端应用。如果 HBase 中没有该短 URL,短 URL 服务器将构造 404 响应返回给客户端应用,时序图如下。

image-20231128221026337

过期短 URL 清理服务器会每个月启动一次,将已经超过有效期(2 年)的 URL 数据删除,并将这些短 URL 追加写入到短 URL 预生成文件中。

为了保证系统高可用,Fuxi 的应用服务器、文件服务器、数据库服务器都采用集群部署方案,单个服务器故障不会影响 Fuxi 短 URL 的可用性。

对于 Fuxi 的高性能要求,80% 以上的访问请求将被设计为通过缓存返回。Redis 的缓存响应时间 1ms 左右,服务器端请求响应时间小于 3ms,满足 80% 请求小于 5ms 的性能目标。对于缓存没有命中的数据,通过 HBase 获取,HBase 平均响应时间 10ms,也可以满足设计目标中的性能指标。

对于 Redis 缓存内存空间估算,业界一般认为,超过 80% 请求集中在最近 6 天生成的短URL 上,Fuxi 主要缓存最近六天生成的短 URL 即可。根据需求容量估计,最近 6 天生成的短 URL 数量约 1 亿条,因此需要 Redis 缓存服务器内存空间:1亿 × 1KB = 100GB

3、详细设计

详细设计关注重定向响应码、短 URL 预生成文件及加载、用户自定义短 URL 等几个关键设计点。

1、重定向响应码

满足短 URL 重定向要求的 HTTP 重定向响应码有 301 和 302 两种,其中 301 表示永久重定向,即浏览器一旦访问过该短 URL,就将重定向的原始长 URL 缓存在本地,此后不再请求短 URL 生成器,直接根据缓存在浏览器(HTTP 客户端)的长 URL 路径进行访问。302 表示临时重定向,每次访问短 URL 都需要访问短 URL 生成器。

一般说来,使用 301 状态码可以降低 Fuxi 服务器的负载压力,但无法统计短 URL 的使用情况,而 Fuxi 的架构设计完全可以承受这些负载压力,因此 Fuxi 使用 302 状态码构造重定向响应。

2、短 URL 预生成文件及预加载

Fuxi 的短 URL 是在系统上线前全部预生成的,并存储在 HDFS 文件中。共 144 亿个短URL,每个短 URL 6 个字符,文件大小 :144亿 × 6B = 86.4GB

文件格式就是直接将 144 亿个短 URL 的 ASC 码无分割地存储在文件中,如下是存储了 3个短 URL 的文件示例:

Wdj4FbOxTw9CHtvPM1

所以如果短 URL 预加载服务器第一次启动的时候加载 1 万个短 URL,那么就从文件头读取 60K 数据,并标记当前文件偏移量 60K。下次再加载 1 万个短 URL 的时候,再从文件60K 偏移位置继续读取 60K 数据即可。

因此,Fuxi 除了需要一个在 HDFS 记录预生成短 URL 的文件外,还需要一个记录偏移量的文件,记录偏移量的文件也存储在 HDFS 中。同时,由于预加载短 URL 服务器集群部署多台服务器,会出现多台服务器同时加载相同短 URL 的情况,所以还需要利用偏移量文件对多个服务器进行互斥操作,即利用文件系统写操作锁的互斥性实现多服务器访问互斥

应用程序的文件访问流程应该是:写打开偏移量文件 -> 读偏移量 -> 读打开短 URL 文件 -> 从偏移量开始读取 60K 数据 -> 关闭短 URL 文件 -> 修改偏移量文件 -> 关闭偏移量文件。

由于写打开偏移量文件是一个互斥操作,所以第一个预加载短 URL 服务器写打开偏移量文件以后,其他预加载短 URL 服务器无法再写打开该文件,也就无法完成读 60K 短 URL 数据及修改偏移量的操作,这样就能保证这两个操作是并发安全的。

加载到预加载短 URL 服务器的 1 万个短 URL 会以链表的方式存储,每使用一个短 URL,链表头指针就向后移动一位,并设置前一个链表元素的 next 对象为 null。这样用过的短URL 对象可以被垃圾回收。

当剩余链表长度不足 2000 的时候,触发一个异步线程,从文件中加载 1 万个新的短URL,并链接到链表的尾部。

与之对应的 URL 链表类图如下。

image-20231128222635415

  • URLNode:URL 链表元素类,成员变量 uRL 即短 URL 字符串,next 指向下一个链表元素。
  • LinkedURL:URL 链表主类,成员变量 head 指向链表头指针元素,uRLAmount 表示当前链表剩余元素个数。acquireURL() 方法从链表头指针指向的元素中取出短 URL 字符串,并执行 urlAmount-- 操作。当 urlAmount < 2000 的时候,调用私有方法loadURL(),该方法调用一个线程从文件中加载 1 万个短 URL 并构造成链表添加到当前链表的尾部,并重置 uRLAmount。

3、用户自定义短 URL

Fuxi 允许用户自己定义短 URL,即在生成短 URL 的时候,由用户指定短 URL 的内容。为了避免预生成的短 URL 和用户指定的短 URL 冲突,Fuxi 限制用户自定义短 URL 的字符个数,不允许用户使用 6 个字符的自定义短 URL,且 URL 长度不得超过 20 个字符。

但是用户自定义短 URL 依然可能和其他用户自定义短 URL 冲突,所以 Fuxi 生成自定义短URL 的时候需要到数据库中检查冲突,是否指定的 URL 已经被使用,如果发生冲突,要求用户重新指定。

4、URL Base64 编码

标准 Base64 编码表如下:

image-20231128223114259

其中“+”和“/”在 URL 中会被编码为“%2B”以及“%2F”,而“%”在写入数据库的时候又和 SQL 编码规则冲突,需要进行再编码,因此直接使用标准 Base64 编码进行短URL 编码并不合适。URL 保留字符编码表如下。

image-20231128223211182

所以,我们需要针对 URL 场景对 Base64 编码进行改造,使用 URL 保留字符表以外的字符对 Base64 编码表中的 62,63 进行编码:将“+”改为“-”,将“/”改为“_”,Fuxi 最终采用的 URL Base64 编码表如下。

image-20231128223254100

4、总结

我们开头提到,Fuxi 是一个高并发(2 万 QPS)、海量存储(144 亿条数据)、还需要10ms 的高性能平均响应时间的系统。但是我们后面看到,Fuxi 的架构并不复杂。

这一方面是源于 Fuxi 的业务逻辑非常简单,只需要完成短 URL 与长 URL 的映射关系生成与获取就可以了。另一方面则是源于开源技术体系的成熟,比如一个 HDFS 集群可支持百万 TB 规模的数据存储,而我们需要的存储空间只有区区不到 100GB,都有点大材小用了。事实上,Fuxi 选择 HDFS 更多的考量是利用 HDFS 的高可用,HDFS 的自动备份策略为我们提供了高可用的数据存储解决方案

同理,高并发也是如此,2 万 QPS 看起来不小,但实际上,由于业务逻辑简单,单个数据都很小,加上大部分请求数据可以通过 Redis 缓存获取,所以实际响应时间是非常短的,10ms 的平均响应时间使得 Fuxi 真正承受的并发压力只有 200。对于这样简单的业务逻辑以及 200 这样的并发压力,我们使用配置高一点的服务器的话,只需要一台短 URL 服务器其实就可以满足了。所以,我们在短 URL 服务器之前使用负载均衡服务器,这也是更多地为高可用服务。

往期推荐

1.高并发系统:它的通用设计方法是什么?
2. 软件建模与文档:架构师怎样绘制系统架构蓝图?
3. 高并发架构设计方法:面对高并发,怎么对症下药?

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/201187.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

水离子水壁炉的科技创新与时尚家居潮流

近年来&#xff0c;水离子水壁炉作为家居装饰的新宠儿&#xff0c;正在以其独特的科技创新和时尚设计引领家居潮流。这一新型壁炉不仅注重外观美感&#xff0c;更借助先进科技实现了温馨的火焰效果&#xff0c;成为现代家居中的独特亮点。 水离子水壁炉的科技创新主要体现在其采…

【Mysql学习笔记】3 - 本章作业

1.判断 1. 这句话表示ename as name 可以不要这个as&#xff0c;同理后面的sal salary也是别名&#xff0c;而选项D的Annual Salary中间也有空格&#xff0c;程序会判断为as 但as不能连用&#xff0c;所以错误&#xff0c;选D 2.选B&#xff0c;因为null不能加上判断符号<&…

Stable Diffusion绘画系列【7】:极致东方美学

《博主简介》 小伙伴们好&#xff0c;我是阿旭。专注于人工智能AI、python、计算机视觉相关分享研究。 ✌更多学习资源&#xff0c;可关注公-仲-hao:【阿旭算法与机器学习】&#xff0c;共同学习交流~ &#x1f44d;感谢小伙伴们点赞、关注&#xff01; 《------往期经典推荐--…

leetCode 131.分割回文串 + 回溯算法 + 图解 + 笔记

131. 分割回文串 - 力扣&#xff08;LeetCode&#xff09; 给你一个字符串 s&#xff0c;请你将 s 分割成一些子串&#xff0c;使每个子串都是 回文串 。返回 s 所有可能的分割方案。回文串 是正着读和反着读都一样的字符串 示例 1&#xff1a; 输入&#xff1a;s "aa…

RabbitMQ消息模型之Work Queues

Work Queues Work Queues&#xff0c;也被称为&#xff08;Task Queues&#xff09;&#xff0c;任务模型&#xff0c;也是官网给出的第二个模型&#xff0c;使用的交换机类型是直连direct&#xff0c;也是默认的交换机类型。当消息处理比较耗时的时候&#xff0c;可能生产消息…

F. Magic Will Save the World

首先积攒了能量打了怪再积攒是没有意义的&#xff0c;可以直接积攒好&#xff0c;然后一次性进行攻击 那么怎么进行攻击了&#xff1f;可以尽量的多选怪物使用水魔法攻击剩余的再用火魔法进行攻击&#xff0c; 也就是只要存在合法的体积&#xff08;即装入背包的怪物的体积之…

Docker、Kubernetes、OCI、CRI-O、containerd、runc 之间的关系以及它们是如何一起工作的?

最近网上看到一张图片&#xff0c;能够很清晰地展现出 Docker、Kubernetes、OCI、CRI-O、containerd、runc 之间的关系以及它们是如何在一起工作的&#xff0c;如下&#xff1a; 本文可以作为之前一篇文章&#xff08;《K8s、Docker、CRI、OCI 之间的爱恨情仇》&#xff09;的…

Echarts的引入使用

ECharts文档 1.下载并引入Echarts 2.准备一个具备大小的DOM容器 3.初始化echarts实例对象 4.指定配置项和数据(option) 5.将配置项设置给echarts实例对象 最后是一个js文件 echarts的引入 1.引入echarts - js 文件 <script src"js/echarts.min.js"></scri…

【新手解答1】深入探索 C 语言:变量名、形参 + 主调函数、被调函数 + 类和对象 + 源文件(.c 文件)、头文件(.h 文件)+ 库

C语言的相关问题解答 写在最前面目录 问题1变量名与变量的关系与区别变量和数据类型形参&#xff08;形式参数&#xff09;的概念 问题2解析&#xff1a;主调函数和被调函数延伸解析&#xff1a;主调函数对于多文件程序的理解总结 问题3类和对象变量和数据类型变量是否为抽象的…

YOLOv5算法进阶改进(6)— 更换主干网络之ResNet18

前言:Hello大家好,我是小哥谈。ResNet18是ResNet系列中最简单的一个模型,由18个卷积层和全连接层组成,其中包含了多个残差块。该模型在ImageNet数据集上取得了很好的表现,成为了深度学习领域的经典模型之一。ResNet18的优点是可以解决深度神经网络中梯度消失的问题,使得性…

第一个C代码讲解

文章目录 编写C文件创建文本文件编写代码修改文件后缀切换文件路径 编译代码打开命令行使用gcc编译代码运行程序双击运行使用命令行运行 代码分析编译过程 编写C文件 编辑C代码文件的工具有很多&#xff0c;为了让大家初学的时候摆脱编译软件的干扰&#xff0c;更容易理解编译过…

hql面试题之上海某资深数仓开发工程师面试题-求不连续月份的月平均值

1.题目 A,B两组产品的月平均值&#xff0c;月平均值是当月的前三个月值的一个平均值&#xff0c;注意月份是不连续的&#xff0c;如果当月的前面的月份不存在&#xff0c;则为0。如A组2023-04的月平均值为2023年1月的数据加2023-02月的数据的平均值&#xff0c;因为没有其他月…

MES管理系统在智能工厂建设中的五个核心作用

随着制造业的数字化转型&#xff0c;智能工厂已经成为了现代工业生产的标志。而在智能工厂中&#xff0c;MES生产管理系统扮演着至关重要的角色。MES管理系统是一种用于管理和监控生产过程的软件系统&#xff0c;通过集成生产计划、资源调度、设备控制、质量管理等功能&#xf…

Cytoscape学习教程

写在前面 今天分享的内容是自己遇到问题后,咨询社群里面的同学,帮忙解决的总结。 关于Cytoscape,对于做组学或生物信息学的同学基本是陌生的,可能有的同学用这个软件作图是非常溜的,做出来的网络图也是十分的好看,“可玩性”很高,就像前面分享的aPEAR包一样aPEAR包绘制…

Python自动化测试工具selenium使用指南

概述 selenium是网页应用中最流行的自动化测试工具&#xff0c;可以用来做自动化测试或者浏览器爬虫等。官网地址为&#xff1a;selenium。相对于另外一款web自动化测试工具QTP来说有如下优点&#xff1a; 免费开源轻量级&#xff0c;不同语言只需要一个体积很小的依赖包支持…

大杀四方,华为组建智能车大联盟 | 百能云芯

最近&#xff0c;华为和一系列汽车公司合资的新公司迎来新的进展。除了与长安汽车的合作外&#xff0c;据传华为已经邀请奇瑞、赛力斯、北汽以及江淮汽车入股新公司&#xff0c;这将使华为成为中国智能汽车平台的重要主导者。 根据澎湃新闻的报道&#xff0c;知情人透露&#x…

Mysql之子查询(知识点+例题)

Mysql之子查询<知识点例题> 什么是子查询案例分析案例分析子查询的分类单行子查询子查询中的空值问题题目练习题目一题目二题目三题目四题目五补充&#xff1a;聚合函数与GROUP BY的使用关系 CASE表达式&#xff08;子查询中的运用&#xff09;简单CASE表达式搜索CASE表达…

Python将Labelme的Json标注文件进行增、删、改、查

Python将Labelme的Json标注文件进行增、删、改、查 前言前提条件相关介绍实验环境Json标注文件的增、删、改、查增代码实现输出结果 删代码实现输出结果 改代码实现输出结果 查代码实现输出结果 前言 由于本人水平有限&#xff0c;难免出现错漏&#xff0c;敬请批评改正。更多精…

ISCTF2023 部分wp

学一年了还在入门( web where_is_the_flag ISCTF{41631519-1c64-40f6-8dbb-27877a184e74} 圣杯战争 <?php // highlight_file(__FILE__); // error_reporting(0);class artifact{public $excalibuer;public $arrow;public function __toString(){echo "为Saber选择…

elementui的table合并列,三个一组

<el-table :span-method"objectSpanMethod" :cell-style"iCellStyle" :data"tableData" height"63vh" border style"width: 100%; margin-top: 6px"><el-table-column type"index" label"序号"…
最新文章