JavaScript 中 URL 和 URLSearchParams 的陷阱

javascript 中 url 和 urlsearchparams 的陷阱

一切都始于一个错误

在 javascript 和 node.js 中使用 url 应该很简单,但是我们项目中最近的一个错误让我陷入了 url 和 urlsearchparams api 中微妙怪癖的兔子洞。这篇文章将探讨这些怪癖,它们如何在您的代码中引起问题,以及您可以采取哪些措施来避免它们。

问题:使用 axios 处理 url

我们在生成 url 并向其添加哈希签名时遇到了此问题。查询参数的百分比编码不一致,导致意外行为和错误的哈希签名。

很明显,url 和 urlsearchparams 对象之间的交互需要格外小心。

陷阱#1:url.search 与 urlsearchparams.tostring()

第一个惊喜是 url.search 和 urlsearchparams.tostring() 之间的区别

立即学习“Java免费学习笔记(深入)”;

使用 .searchparams 修改 url 时要小心,因为根据 whatwg 规范,urlsearchparams 对象使用不同的规则来确定要对哪些字符进行百分比编码。例如,url 对象不会对 ascii 波形符 (~) 字符进行百分号编码,而 urlsearchparams 将始终对其进行编码。

// example 1const url = new url("https://example.com?param=foo bar");console.log(url.search); // prints param=foo%20barconsole.log(url.searchparams.tostring()); // prints ?param=foo+bar// example 2const myurl = new url('https://example.org/abc?foo=~bar');console.log(myurl.search);  // prints ?foo=~bar// modify the url via searchparams...myurl.searchparams.sort();console.log(myurl.search);  // prints ?foo=%7ebar

在我们的项目中,我们需要显式地重新分配 url.search = url.searchparams.tostring() 以确保查询字符串的编码一致。

陷阱#2:加号困境

另一个问题是 urlsearchparams 如何处理 字符。默认情况下,urlsearchparams 将 解释为空格,这在编码二进制数据或 base64 字符串时可能会导致数据损坏。

const params = new urlsearchparams("bin=e+axqb+a");console.log(params.get("bin")); // "e axqb a"

一种解决方案是在将值附加到 urlsearchparams 之前使用encodeuricomponent:

params.append("bin", encodeuricomponent("e+axqb+a"));

更多详细信息请参阅 mdn 文档。

陷阱 #3:urlsearchparams.get 与 urlsearchparams.tostring()

比较 urlsearchparams.get 和 urlsearchparams.tostring 的输出时会出现另一个微妙之处。例如:

const params = new urlsearchparams("?key=value&key=other");console.log(params.get("key")); // "value" (first occurrence)console.log(params.tostring()); // "key=value&key=other" (all occurrences serialized)

在多值场景中,get 仅返回第一个值,而 tostring 则序列化所有值。

我们代码库中的修复

在我们的项目中,我们通过显式重新分配搜索属性解决了该问题:

url.search = url.searchparams.tostring();url.searchparams.set(  "hash",  cryptography.createsha256hmacbase64urlsafe(url.href, secret_key ?? ""));

这确保了在添加哈希值之前所有查询参数都已正确编码。

node.js 查询字符串模块

whatwg urlsearchparams 接口和 querystring 模块具有类似的用途,但 querystring 模块的用途更通用,因为它允许自定义分隔符(& 和 =)。另一方面,urlsearchparams api 纯粹是为 url 查询字符串而设计的。

querystring 比 urlsearchparams 性能更高,但不是标准化 api。当性能不重要或需要与浏览器代码兼容时,请使用 urlsearchparams。

与 querystring 模块不同,使用 urlsearchparams 时,不允许数组值形式的重复键。数组使用 array.tostring() 进行字符串化,它只是用逗号连接所有数组元素。

const params = new urlsearchparams({  user: 'abc',  query: ['first', 'second'],});console.log(params.getall('query'));// prints [ 'first,second' ]console.log(params.tostring());// prints 'user=abc&query=first%2csecond'

使用querystring模块,查询字符串’foo=bar&abc=xyz&abc=123’被解析为:

{  "foo": "bar",  "abc": ["xyz", "123"]}

要点

小心 urlsearchparams 处理特殊字符(例如 ~)和空格。必要时使用encodeuricomponent。

了解 url.search、urlsearchparams.get 和 urlsearchparams.tostring 之间的区别,以避免意外行为。

在 node.js 中,如果要将重复的查询参数键解析为数组,请使用查询字符串模块。

以上就是JavaScript 中 URL 和 URLSearchParams 的陷阱的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1498928.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月19日 20:53:47
下一篇 2025年12月19日 20:53:58

相关推荐

  • 高效处理Axios响应数据:避免Map生成Undefined值的最佳实践

    本文旨在解决JavaScript中Array.prototype.map方法在条件不满足时返回undefined的常见问题,尤其是在处理Axios响应并需要基于另一组数据进行筛选和转换的场景。我们将深入探讨如何利用Set、filter和map组合,高效且准确地从复杂数据结构中提取所需信息,避免生成冗…

    2025年12月20日
    000
  • JS如何实现WeakMap?弱引用的使用

    WeakMap通过弱引用键解决内存泄漏问题,适用于关联对象私有数据、缓存和DOM状态管理,其键必须为对象且不可遍历,与Map的强引用和通用性形成对比,适合需生命周期同步的场景。 WeakMap在JavaScript中是一个非常巧妙的工具,它允许你以一种特殊的方式存储键值对,即键是弱引用。这意味着当一…

    2025年12月20日
    000
  • 什么是贪心算法?贪心算法的适用条件

    贪心算法的核心思想是在每一步选择中都采取当前状态下最优的决策,期望通过一系列局部最优解最终得到全局最优解,其与动态规划的最大区别在于贪心算法不具备回溯机制,决策一旦做出不可更改,而动态规划通过保存子问题的解并综合考虑所有可能路径来保证全局最优;判断贪心算法是否适用的关键是问题必须同时满足贪心选择性质…

    2025年12月20日
    000
  • 什么是单向数据流?数据流的管理

    单向数据流通过State、Action、View三者协作,确保数据从Action到Store再到View的单向流动,提升应用的可预测性与可维护性,解决了双向绑定导致的数据混乱问题,适用于大型应用开发。 单向数据流,简单来说,就是数据只能在一个方向上流动,不能反向流动。这带来了一种可预测性和易于调试的…

    2025年12月20日
    000
  • 贪心算法是什么?贪心算法的适用场景

    贪心算法并不总能得到全局最优解,因为它仅基于当前状态做出局部最优选择,而不考虑未来影响或回溯调整;其适用前提是问题具备贪心选择性质和最优子结构性质,如分数背包、霍夫曼编码、最小生成树(prim、kruskal)和dijkstra最短路径等;与动态规划不同,贪心算法不可逆且不存储子问题解,因此判断其适…

    2025年12月20日
    000
  • 什么是Web Worker?多线程的实现

    Web Worker通过后台线程执行耗时任务,避免阻塞主线程,提升页面响应速度。 Web Worker 允许你在后台线程中运行 JavaScript 代码,而不会阻塞主线程,从而提升 Web 应用的性能和响应速度。它本质上是浏览器提供的一种多线程解决方案,但与传统的多线程模型有所不同,它基于消息传递…

    2025年12月20日
    000
  • JS如何实现useRef?Ref的持久化

    useRef能持久化是因为它返回的对象在组件实例的生命周期内始终保持同一引用,React通过将该对象绑定到组件的内部节点(如Fiber节点)实现跨渲染的持久存储,每次调用useRef都返回同一实例,确保.current值在多次渲染间不变且修改不触发重渲染。 useRef 在JavaScript(尤其…

    2025年12月20日
    000
  • js如何检测原型是否被密封

    检测 javascript 原型是否被密封最直接的方法是使用 object.issealed(),它会返回一个布尔值表示对象是否被密封;2. 密封对象后不能添加或删除属性,但可以修改现有属性值,而冻结对象(object.freeze())则完全禁止修改;3. 密封操作不影响原型链上的属性查找,实例仍…

    2025年12月20日 好文分享
    000
  • js 如何将数字转为字符串

    答案:JavaScript中数字转字符串常用String()、toString()、模板字面量和空字符串拼接;String()可处理null/undefined更安全,toString()支持进制转换但不适用null/undefined,模板字面量可读性好,隐式转换意图不明确;浮点数精度问题可用to…

    2025年12月20日
    000
  • js如何操作地理定位

    首先,javascript通过浏览器的geolocation api获取用户经纬度,前提是用户授权;1. 检查浏览器是否支持地理定位:使用”geolocation” in navigator判断,不支持则提示用户;2. 调用navigator.geolocation.getc…

    2025年12月20日 好文分享
    000
  • 什么是编辑距离?动态规划计算编辑距离

    编辑距离是衡量两字符串差异的最小操作数,通过动态规划构建矩阵计算,广泛应用于拼写检查、DNA比对等领域,可采用空间优化、剪枝等方法提升性能,其与莱文斯坦距离为同一概念。 编辑距离,简单来说,就是衡量两个字符串差异程度的一种方法。它告诉你,要把字符串A变成字符串B,最少需要多少次“增、删、改”操作。而…

    2025年12月20日
    000
  • js 如何移除数组的某个元素

    移除 javascript 数组中的某个元素,核心方法有两种:1. 使用 splice() 方法可直接修改原数组,适用于已知索引且需在原数组上操作的场景;2. 使用 filter() 方法可创建新数组,适用于根据条件移除元素或需保持原数组不变的场景。若要移除所有指定值的元素,推荐使用 filter(…

    2025年12月20日
    000
  • js怎么遍历对象的原型链

    遍历javascript原型链的核心方法是使用object.getprototypeof()从对象开始逐级获取原型,直到null为止;2. 实际应用包括调试继承关系、检查对象能力、实现高级框架功能;3. object.getprototypeof()是标准api,__proto__是非标准且不推荐使…

    2025年12月20日 好文分享
    000
  • JS中如何实现字典结构?字典的常见操作

    JavaScript中实现字典结构主要用对象和Map,对象适合字符串键和简单场景,Map支持任意类型键、保持插入顺序且性能更优,遍历时对象常用Object.entries(),Map推荐for…of,选择取决于键类型、顺序需求及性能要求。 在JavaScript里,实现字典结构的核心思路…

    2025年12月20日
    000
  • js 怎样用debounce创建防抖函数

    防抖函数的作用是确保事件在停止触发一段时间后才执行回调,避免频繁触发导致性能问题,1. 通过延迟执行并重新计时来减少函数调用次数;2. 适用于输入搜索、窗口调整等场景;3. 与节流的区别在于防抖只在停止触发后执行一次,而节流固定频率执行;4. 可通过添加leading和trailing选项优化;5.…

    2025年12月20日
    000
  • 什么是CommonJS?模块化的规范

    CommonJS在Node.js中扮演了基石角色,它通过require和module.exports实现了服务器端JavaScript的模块化,解决了命名空间污染和依赖管理问题,促进了npm生态的繁荣;其同步加载机制适合本地文件系统,使代码组织更清晰、可维护,而与ES Modules相比,Commo…

    2025年12月20日
    000
  • js 如何用last获取数组的最后一个元素

    javascript数组没有内置last()方法,最常用获取最后一个元素的方法是通过索引myarray[myarray.length – 1];2. es2022引入的at(-1)方法提供更直观的负索引访问,语法更简洁且可读性更强;3. array[array.length &#8211…

    2025年12月20日
    000
  • 最长公共子序列是什么?LCS的求解方法

    最长公共子序列(LCS)通过动态规划求解,利用dpi表示两字符串前i和前j个字符的LCS长度,当字符匹配时dpi=1+dpi-1,否则dpi=max(dpi-1, dpi),最终dpm即为所求长度,该方法避免重复计算,时间复杂度O(mn),适用于diff工具、生物信息学序列比对等场景,且可通过回溯d…

    2025年12月20日
    000
  • JS定时器怎么使用

    JS定时器通过setTimeout和setInterval实现,前者延迟执行一次,后者周期性重复执行,需用clearTimeout和clearInterval清除,避免内存泄漏和回调堆积。 JS定时器主要用于在指定的时间间隔后执行一段代码,或者重复执行一段代码。 setTimeout 和 setIn…

    2025年12月20日
    000
  • JavaScript事件循环中哪些操作会产生微任务

    微任务主要由promise回调、mutationobserver和queuemicrotask产生。1.promise的.then()、.catch()、.finally()会在状态变化后将回调放入微任务队列;2.mutationobserver用于监听dom变化,其回调作为微任务批量处理以优化性能…

    2025年12月20日 好文分享
    000

发表回复

登录后才能评论
关注微信