处理请求 url 参数

需求分析

还记得我们上节课遗留了一个问题,再来看这个例子:

  1. axios({
  2. method: 'get',
  3. url: '/base/get',
  4. params: {
  5. a: 1,
  6. b: 2
  7. }
  8. })

我们希望最终请求的 url/base/get?a=1&b=2,这样服务端就可以通过请求的 url 解析到我们传来的参数数据了。实际上就是把 params 对象的 key 和 value 拼接到 url 上。

再来看几个更复杂的例子。

参数值为数组

  1. axios({
  2. method: 'get',
  3. url: '/base/get',
  4. params: {
  5. foo: ['bar', 'baz']
  6. }
  7. })

最终请求的 url/base/get?foo[]=bar&foo[]=baz'

参数值为对象

  1. axios({
  2. method: 'get',
  3. url: '/base/get',
  4. params: {
  5. foo: {
  6. bar: 'baz'
  7. }
  8. }
  9. })

最终请求的 url/base/get?foo=%7B%22bar%22:%22baz%22%7Dfoo 后面拼接的是 {"bar":"baz"} encode 后的结果。

参数值为 Date 类型

  1. const date = new Date()
  2. axios({
  3. method: 'get',
  4. url: '/base/get',
  5. params: {
  6. date
  7. }
  8. })

最终请求的 url/base/get?date=2019-04-01T05:55:39.030Zdate 后面拼接的是 date.toISOString() 的结果。

特殊字符支持

对于字符 @:$,`、[],我们是允许出现在url` 中的,不希望被 encode。

  1. axios({
  2. method: 'get',
  3. url: '/base/get',
  4. params: {
  5. foo: '@:$, '
  6. }
  7. })

最终请求的 url/base/get?foo=@:$+,注意,我们会把空格 ` 转换成+`。

空值忽略

对于值为 null 或者 undefined 的属性,我们是不会添加到 url 参数中的。

  1. axios({
  2. method: 'get',
  3. url: '/base/get',
  4. params: {
  5. foo: 'bar',
  6. baz: null
  7. }
  8. })

最终请求的 url/base/get?foo=bar

丢弃 url 中的哈希标记

  1. axios({
  2. method: 'get',
  3. url: '/base/get#hash',
  4. params: {
  5. foo: 'bar'
  6. }
  7. })

最终请求的 url/base/get?foo=bar

保留 url 中已存在的参数

  1. axios({
  2. method: 'get',
  3. url: '/base/get?foo=bar',
  4. params: {
  5. bar: 'baz'
  6. }
  7. })

最终请求的 url/base/get?foo=bar&bar=baz

buildURL 函数实现

根据我们之前的需求分析,我们要实现一个工具函数,把 params 拼接到 url 上。我们希望把项目中的一些工具函数、辅助方法独立管理,于是我们创建一个 helpers 目录,在这个目录下创建 url.ts 文件,未来会把处理 url 相关的工具函数都放在该文件中。

helpers/url.ts

  1. import { isDate, isObject } from './util'
  2. function encode (val: string): string {
  3. return encodeURIComponent(val)
  4. .replace(/%40/g, '@')
  5. .replace(/%3A/gi, ':')
  6. .replace(/%24/g, '$')
  7. .replace(/%2C/gi, ',')
  8. .replace(/%20/g, '+')
  9. .replace(/%5B/gi, '[')
  10. .replace(/%5D/gi, ']')
  11. }
  12. export function bulidURL (url: string, params?: any) {
  13. if (!params) {
  14. return url
  15. }
  16. const parts: string[] = []
  17. Object.keys(params).forEach((key) => {
  18. let val = params[key]
  19. if (val === null || typeof val === 'undefined') {
  20. return
  21. }
  22. let values: string[]
  23. if (Array.isArray(val)) {
  24. values = val
  25. key += '[]'
  26. } else {
  27. values = [val]
  28. }
  29. values.forEach((val) => {
  30. if (isDate(val)) {
  31. val = val.toISOString()
  32. } else if (isObject(val)) {
  33. val = JSON.stringify(val)
  34. }
  35. parts.push(`${encode(key)}=${encode(val)}`)
  36. })
  37. })
  38. let serializedParams = parts.join('&')
  39. if (serializedParams) {
  40. const markIndex = url.indexOf('#')
  41. if (markIndex !== -1) {
  42. url = url.slice(0, markIndex)
  43. }
  44. url += (url.indexOf('?') === -1 ? '?' : '&') + serializedParams
  45. }
  46. return url
  47. }

helpers/util.ts

  1. const toString = Object.prototype.toString
  2. export function isDate (val: any): val is Date {
  3. return toString.call(val) === '[object Date]'
  4. }
  5. export function isObject (val: any): val is Object {
  6. return val !== null && typeof val === 'object'
  7. }

实现 url 参数处理逻辑

我们已经实现了 buildURL 函数,接下来我们来利用它实现 url 参数的处理逻辑。

index.ts 文件中添加如下代码:

  1. function axios (config: AxiosRequestConfig): void {
  2. processConfig(config)
  3. xhr(config)
  4. }
  5. function processConfig (config: AxiosRequestConfig): void {
  6. config.url = transformUrl(config)
  7. }
  8. function transformUrl (config: AxiosRequestConfig): string {
  9. const { url, params } = config
  10. return bulidURL(url, params)
  11. }

在执行 xhr 函数前,我们先执行 processConfig 方法,对 config 中的数据做处理,除了对 urlparams 处理之外,未来还会处理其它属性。

processConfig 函数内部,我们通过执行 transformUrl 函数修改了 config.url,该函数内部调用了 buildURL

那么至此,我们对 url 参数处理逻辑就实现完了,接下来我们就开始编写 demo 了。

demo 编写

examples 目录下创建 base 目录,在 base 目录下创建 index.html:

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="utf-8">
  5. <title>Base example</title>
  6. </head>
  7. <body>
  8. <script src="/__build__/base.js"></script>
  9. </body>
  10. </html>

接着创建 app.ts 作为入口文件:

  1. import axios from '../../src/index'
  2. axios({
  3. method: 'get',
  4. url: '/base/get',
  5. params: {
  6. foo: ['bar', 'baz']
  7. }
  8. })
  9. axios({
  10. method: 'get',
  11. url: '/base/get',
  12. params: {
  13. foo: {
  14. bar: 'baz'
  15. }
  16. }
  17. })
  18. const date = new Date()
  19. axios({
  20. method: 'get',
  21. url: '/base/get',
  22. params: {
  23. date
  24. }
  25. })
  26. axios({
  27. method: 'get',
  28. url: '/base/get',
  29. params: {
  30. foo: '@:$, '
  31. }
  32. })
  33. axios({
  34. method: 'get',
  35. url: '/base/get',
  36. params: {
  37. foo: 'bar',
  38. baz: null
  39. }
  40. })
  41. axios({
  42. method: 'get',
  43. url: '/base/get#hash',
  44. params: {
  45. foo: 'bar'
  46. }
  47. })
  48. axios({
  49. method: 'get',
  50. url: '/base/get?foo=bar',
  51. params: {
  52. bar: 'baz'
  53. }
  54. })

接着在 server.js 添加新的接口路由:

  1. router.get('/base/get', function(req, res) {
  2. res.json(req.query)
  3. })

然后在命令行运行 npm run dev,接着打开 chrome 浏览器,访问 http://localhost:8080/ 即可访问我们的 demo 了,我们点到 Base 目录下,通过开发者工具的 network 部分我们可以看到成功发送的多条请求,并可以观察它们最终请求的 url,已经如期添加了请求参数。

那么至此我们的请求 url 参数处理编写完了,下一小节我们会对 request body 数据做处理。