手写 URL 解析器,面试官到底想考什么?

前言

最近一直在看一些面经,发现大部分面试官问八股提到URL的话,十有八九就是问从URL到页面渲染,但是我看到一个比较有意思的手写题:解析URL。对于没见过的面试题,我都是非常感兴趣的,下面来分享并说说我是怎么做的。

有关从URL到页面渲染的请看我的文章:网页都知道要双向握手才加载!从 URL 到页面渲染,单向喜欢连 DNS 都解析不通

一、先搞懂:URL 到底长啥样?

在动手写代码前,咱们先拆解下目标 URL 的结构,就像拆快递一样,一层一层来。后面的是我自己随便编的,我们只需输出结果,不影响:

JavaScript 复制代码
const url = 'https://www.baidu.com/order/home?user=Tom&id=123&city=上海&id=56';

它的结构可以拆成 4 部分:

  • 协议(protocol) :https ------ 告诉浏览器用什么方式传输数据
  • 主机名(hostname)www.baidu.com ------ 目标服务器的 "家门地址"
  • 路径(path) :/order/home ------ 服务器上具体的 "房间号"
  • 查询参数(query) :? 后面的所有内容 ------ 传给服务器的 "小纸条",重点是这里有重复的 id 参数,需要转成数组!

我们的目标就是把它转成规整的对象:

JavaScript 复制代码
// 期望输出的结果
const output = {
    protocol: 'https',
    hostname: 'www.baidu.com',
    path: '/order/home',
    query: {
        user: 'Tom',
        id: [123, 56], // 重复参数自动转数组
        city: '上海'
    }
}

二、动手开写!原生 JS 手写 URL 解析器

当时我就没想那么多,可能是暴力解题用多了。这核心思路不就一个:字符串拆分(split)+ 循环组装,不用任何框架,纯原生 JS 不就能搞定吗?

JavaScript 复制代码
// 待解析的URL
const url = 'https://www.baidu.com/order/home?user=Tom&id=123&city=上海&id=56';

// URL解析函数
function urlParser(url) {
    // 1. 拆分协议:以"://"为分割符,前面就是协议https
    const protocalArr = url.split('://');
    const protocol = protocalArr[0];

    // 2. 拆分主机名:协议后面的字符串,以"/"分割,第一段就是主机名
    const hostname = protocalArr[1].split('/')[0];

    // 3. 拆分路径:主机名后面、?前面的部分就是路径
    const path = protocalArr[1].split(hostname)[1].split('?')[0];

    // 4. 拆分查询参数:?后面的内容用&分割,得到每一个参数项
    const queryArr = protocalArr[1].split(hostname)[1].split('?')[1].split('&');
    const queryObj = {}; // 用来存放最终的参数对象

    // 循环处理每一个参数
    queryArr.forEach(item => {
        // 每个参数用=分割,左边是key,右边是value
        const [key, value] = item.split('=');
        
        // 重点:处理重复参数(比如id出现两次)
        if (queryObj[key]) {
            // 如果已经是数组,直接push新值
            if (Array.isArray(queryObj[key])) {
                queryObj[key].push(value);
            } else {
                // 如果不是数组,转成数组再存新值
                queryObj[key] = [queryObj[key], value];
            }
        } else {
            // 无重复,直接存原值
            queryObj[key] = value;
        }
    })
    
    // 返回最终解析好的对象
    return {
        protocol,
        hostname,
        path: path,
        query: queryObj
    }
}

// 调用函数,打印结果
console.log(urlParser(url));

结语

一道很简单的手写题,无非就是麻烦一点。能出这道题我觉得大概也是想考察这个逻辑思维的能力,好了,今天的分享就到这!

相关推荐
NiceCloud喜云2 小时前
Opus 4.8 的 Effort Control 怎么选:Low 到 Max 五档策略
android·java·大数据·前端·c++·python·spring
wordbaby3 小时前
React Native + RNOH:跨页面数据回传的最佳实践与避坑指南
前端·react native
GISer_Jing3 小时前
Three.js着色器编译机制深度解析
javascript·webgl·着色器
丷丩3 小时前
MapLibre GL JS第22课:查看本地GeoJSON
前端·javascript·map·mapbox·maplibre gl js
jiayong233 小时前
面试中遇到不熟悉问题的应对策略深度解析
面试·职场和发展
油炸自行车3 小时前
Claude Code 错误:API Error: 400 Failed to deserialize the JSON body into the
开发语言·javascript·json·trae·claude code·api error 400
Front思4 小时前
AI前端工程师需要具备能力+
前端·人工智能·ai
JAVA社区4 小时前
Java高级全套教程(十)—— SpringCloudAlibaba超详细实战详解
java·开发语言·spring cloud·面试·职场和发展
ZC跨境爬虫6 小时前
跟着 MDN 学CSS day_29:(掌握文本与字体样式的核心艺术)
前端·css·ui·html·tensorflow
李子琪。7 小时前
网络空间安全深度实战:CSRF 漏洞原理剖析与基于 Token 的纵深防御体系构建(全栈实验报告)
前端·安全·csrf