Skip to content

一. 认识BOM


  • BOM浏览器对象模型Browser Object Model

    • 简称 BOM,由浏览器提供的用于处理除文档(document)之外的所有内容的其他对象
    • 比如navigatorlocationhistory等对象
  • JS有一个非常重要的运行环境就是浏览器

    • 而且浏览器本身又作为一个应用程序需要对其本身进行操作
    • 所以通常浏览器会有对应的对象模型(BOMBrowser Object Model
    • 我们可以将BOM看成是连接JS脚本与浏览器窗口的桥梁
  • BOM主要包括以下的对象模型(接口):

    接口含义
    window包括全局属性、方法,控制浏览器窗口相关的属性、方法
    location浏览器连接到的对象的位置(URL
    history操作浏览器的会话历史记录
    navigator用户代理(浏览器)的状态和标识
    screen屏幕窗口信息

二. 全局对象window


1. window对象

  • window对象在浏览器中可以从两个视角来看待:
    • 视角一:全局对象
      • ECMAScript中其实是有一个全局对象的,这个全局对象在Node中是global
      • 在浏览器中就是window对象
    • 视角二:浏览器窗口对象
      • 作为浏览器窗口时,提供了对浏览器操作的相关的api
  • 当然,这两个视角存在大量重叠的地方,所以不需要刻意去区分它们:
    • 事实上对浏览器和Node中全局对象名称不一样的情况,目前已经指定了对应的标准,称之为**globalThis**,并且大多数现代浏览器都支持它
    • 放在window对象上的所有属性都可以被访问
    • 使用var定义的全局变量会被添加到window对象中
    • window默认给我们提供了全局的函数和类:setTimeoutMathDateObject

2. window对象的作用

  • 事实上window对象上肩负的重担是非常大的:

    • 第一:包含大量的属性,localStorageconsolelocationhistoryscreenXscrollX等等(大概60+个属性)
    • 第二:包含大量的方法,alertclosescrollToopen等等(大概40+个属性)
    • 第三:包含大量的事件,focusblurloadhashchange等等(大概30+个属性)
    • 第四:包含从EventTarget继承过来的方法,addEventListenerremoveEventListenerdispatchEvent方法
  • 那么这些大量的属性、方法、事件在哪里查看呢?

  • 查看MDN文档时,我们会发现有很多不同的符号,是什么意思:

    image-20220623173714903
    • 删除符号:表示这个API已经废弃,不推荐继续使用了
    • 点踩符号:表示这个API不属于W3C规范,某些浏览器有实现(所以兼容性的问题)
    • 实验符号:该API是实验性特性,以后可能会修改,并且存在兼容性问题

3. window常见的属性

js
// 浏览器高度
console.log(window.outerHeight)
console.log(window.innerHeight)
console.log('screenX: ', window.screenX) // 分屏扩展模式下,默认相对的是屏幕1左上角计算的,可以修改主屏幕的
console.log('screenY: ', window.screenY)

// 监听
window.addEventListener('scroll', event => {
  console.log(window.scrollY)
  console.log(window.scrollX)
})

4. window常见的方法

html
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
  </head>
  <body>

    <button id="close">打开百度并1s后关闭</button>
    <button id="scroll">滚动</button></button>

    <br><br><br><br><br><br><br><br><br><br>
    <br><br><br><br><br><br><br><br><br><br>
    <br><br><br><br><br><br><br><br><br><br>
    <br><br><br><br><br><br><br><br><br><br>
    <br><br><br><br><br><br><br><br><br><br>
    <br><br><br><br><br><br><br><br><br><br> 

  </body>
<script>
  var openBtn = document.querySelector('close')
  openBtn.onclick = function() {
    win1 = window.open('https://www.baidu.com', '_target') // '_self不能关闭'
    setTimeout(() => {
      win1.close()
    }, 1000);
  }

  var scrollBtn = document.querySelector('#scroll')
  scrollBtn.onclick = function() {
    window.scrollTo({
      top: 1000,
      behavior: "smooth" // 平滑滑动
    })
  }
</script>
</html>

5. window常见的事件

js
window.onfocus = function() {
  console.log('获取焦点')
}

window.onblur = function() {
  console.log('失去焦点')
}

window.onhashchange = function() {
  console.log('hash值发生改变')
}

三. location、URLSearchParams


1. location常见属性

  • location对象用于表示window上当前链接到的URL信息

  • 常见属性有哪些?

    属性含义示例:http://127.0.0.1:5501/test3呵呵.html?a=1&b=2#1
    href完整URLhttp://127.0.0.1:5501/test3呵呵.html?a=1&b=2#1
    (完整url,中文会编码处理)
    protocol当前协议 http:
    host主机地址(带端口号)127.0.0.1:5501
    hostname主机名(不带端口)127.0.0.1
    port端口号5501
    pathname路径/test3%E5%91%B5%E5%91%B5.html(路径,中文会编码处理)
    search查询字符串?a=1&b=2
    hash哈希值#1
    usernameURL中的username(很多浏览器已经禁用)

2. location常见方法

  • 我们会发现location其实是URL的一个抽象实现:

    image-20220623192520357
  • location有如下常用的方法:

    方法作用
    assign()赋值一个新的URL,并且跳转到该URL
    replace()替换当前的网页URL(不会在浏览器中留下之前的记录)
    reload()重新加载页面。
    Firefox 对于该方法支持一个非标准的boolean参数,
    当值为 true 时,将强制Firefox 从服务器加载页面资源
    html
    <!DOCTYPE html>
    <html lang="en">
      <head>
        <meta charset="UTF-8">
        <meta http-equiv="X-UA-Compatible" content="IE=edge">
        <meta name="viewport" content="width=device-width, initial-scale=1.0">
        <title>Document</title>
      </head>
      <body>
    
        <button>打开新的网页</button>
        <button>替换新的网页</button>
        <button>重新加载网页</button>
    
        <script>
          var btns = document.querySelectorAll('button')
          console.log(btns)
          btns[0].onclick = function() {
            location.assign('http://baidu.com')
          }
          btns[1].onclick = function() {
            location.replace('http://baidu.com')
          }
          btns[2].onclick = function() {
            location.reload()
          }
        </script>
      </body>
    </html>

3. URLSearchParams

  • URLSearchParams定义了一些实用的方法来处理URL的查询字符串

    • 可以将一个字符串转化为URLSearchParams类型
    • 也可以将一个URLSearchParams类型转成字符串
    • URLSearchParams构造函数创建出来的实例对象,是可以使用for..of
    js
    var urlSearch = new URLSearchParams('name=later&age=18')
    
    console.log(urlSearch.get('name')) // later
    
    console.log(urlSearch.toString()) // 'name=later&age=18'
    
    urlSearch.set('age', '23')
    
    console.log(urlSearch.get('age')) // 23
    
    urlSearch.append('height', '1.88')
    
    console.log(urlSearch.get('height')) // 1.88
    
    console.log(urlSearch.has('name')) // true
  • URLSearchParams常见的方法有如下:

    方法作用
    get()获取搜索参数的值
    set()设置一个搜索参数和值
    append()追加一个搜索参数和值
    has()判断是否有某个搜索参数
  • 中文会使用encodeURIComponentdecodeURIComponent进行编码和解码

四. history


  • history对象允许访问浏览器曾经的会话历史记录

  • 有两个默认属性:

    • length:会话中的记录条数
    • state:当前保留的状态值
  • 有五个方法:

    方法作用
    back()返回上一页,等同于history.go(-1)
    forward()前进下一页,等同于history.go(1)
    go()加载历史中的某一页,可以输入一个参数,指定前进或后退几页
    pushState()打开一个指定地址,且不会重新加载页面
    replaceState()打开一个新地址,并且使用replace,不会在历史记录中产生新的记录
    (直接替换当前页)
    JS
    // 前端路由核心: 修改了URL, 但是页面不刷新
    // 1> 修改hash值
    // 2> 修改history
    
    // 1.history对应的属性
    console.log(history.length)
    console.log(history.state)
    
    // 2.修改history
    var btnEl = document.querySelector("button")
    btnEl.onclick = function() {
      // history.pushState({ name: "why", age: 18 }, "", "/why")
      history.replaceState({ name: "why", age: 18 }, "", "/why")
    }
    
    var backBtnEl = document.querySelector(".back")
    backBtnEl.onclick = function() {
      // history.back()
      // history.forward()
      // 类似于上面的两个方法, 只是可以传入层级
      // history.go(-2)
    }

五. navigator、screen


1. navigator

  • navigator对象表示用户代理(浏览器或运行环境)的状态和标识等信息

    image-20220624172919619

2. screen

  • screen主要记录的是浏览器窗口外面的客户端显示器的信息

    • 比如屏幕的逻辑像素:screen.widthscreen.height

      image-20220624182351047

六. JSON


1. json的由来

  • 在目前的开发中,JSON是一种非常重要的数据格式,它并不是编程语言,而是一种可以在服务器和客户端之间传输的数据格式

  • JSON的全称是JavaScript Object NotationJS对象符号):

    • JSON是由Douglas Crockford构想和设计的一种轻量级资料交换格式,算是JS的一个子集
    • 但是虽然JSON被提出来的时候是主要应用JS中,但是目前已经独立于编程语言,可以在各个编程语言中使用
    • 很多编程语言都实现了将JSON转成对应模型的方式
  • 其他的传输格式:

    • XML:在早期的网络传输中主要是使用XML来进行数据交换的,但是这种格式在解析、传输等各方面都弱于JSON,所以目前已经很少在被使用了
    • Protobuf:另外一个在网络传输中目前已经越来越多使用的传输格式是protobuf,但是直到2021年的3.x版本才支持JS,所以目前在前端使用的较少
  • 目前JSON被使用的场景也越来越多:

    • 网络数据的传输JSON数据
    • 项目的某些配置文件
    • 非关系型数据库(NoSQL)将json作为存储格式
  • 小程序的app.json

    image-20220624184150048

2. json的基本语法

  • JSON的顶层支持三种类型的值

    • 简单值:数字Number、字符串String不支持单引号)、布尔Boolean、空值null

    • 对象值:由keyvalue组成,key是字符串类型,并且必须添加双引号,值可以是简单值、对象值、数组值

    • 数组值:数组的值可以是简单值、对象值、数组值

    • json中不能写注释

      • jsconfig.jsontsconfig.js这两个文件vscode有做特殊处理,所以其中可以写注释

      • 示例:

        json
        123
        json
        [
          "123",
          123
        ]
        json
        {
          "name": "why",
          "age": 123,
          "friend": {
            "name": "kobe",
            "age": null
          }
        }

3. json的序列化

  • 某些情况下我们希望将JS中的复杂类型转化为json格式的字符串,这样方便对其进行处理

    • 比如我们希望将一个对象保存到localStorage

    • 但是如果我们直接存放一个对象,这个对象会被转化为[object Object]格式的字符串,并不是我们想要的结果

      image-20220624215028611
  • 在**ES5中引用了JSON全局对象**,该对象有两个常用的方法:

    • stringify(序列化):
      • JS类型转成对应的JSON字符串
      • undefined、任意的函数以及symbol值,在序列化过程中会被忽略(出现在非数组对象的属性值中时)或者被转换成null(出现在数组中时)。函数、undefined被单独转换时,会返回undefined,如JSON.stringify(function(){} || undefined)
    • parse(反序列化):
      • 解析JSON字符串,转回对应的JS类型
  • 那么上面的代码我们可以通过如下的方法来使用:

js
var obj = {
  name : 'later'
}

// localStorage.setItem('obj', obj)
localStorage.setItem('obj', JSON.stringify(obj)) // 序列化对象转成JSON字符串进行存取

var _obj = localStorage.getItem('obj')
console.log(_obj, typeof _obj) // {"name":"later"} string
console.log(JSON.parse(_obj), typeof JSON.parse(_obj)) // {name: 'later'} 'object' 反序列化JSON字符串转化为JS类型

4. Stringify的参数replace

  • JSON.stringify() 方法将一个JS对象或值转换为JSON 字符串:

    • 如果指定了一个replacer 函数,则可以选择性地替换值,函数必须返回value
    • 如果指定的replacer 是数组,则可选择性地仅包含数组指定的属性
    js
    var obj1 = JSON.stringify(obj)
    console.log(obj1) // {"name":"later","age":18}
    
    var obj2 = JSON.stringify(obj, ['name'])
    console.log(obj2) // {"name":"later"}
    
    var obj3 = JSON.stringify(obj, (key, value) => {
      if (key == "name") {
        return "111"
      }
      return value 
    })
    console.log(obj3) // {"name":"111","age":18}

5. Stringify的参数space

  • space 参数用来控制结果字符串里面的间距

    image-20220624221800791

6. toJSON方法

  • 如果对象本身包含toJSON方法,那么会直接使用toJSON方法返回的结果:
js
var obj = {
  name: "later",
  toJSON: function() {
    return "123"
  }
}

var objJSONString = JSON.stringify(obj)
console.log(objJSONString) // '123'

7. JSON的反序列化

  • JSON.parse() 方法用来解析JSON字符串,构造由字符串描述的JS值或对象

    • 提供可选的reviver函数用以在返回之前对所得到的对象执行变换(操作)

      js
      var obj = {
        name: "later",
        age: 18
      }
      
      var objJSONString = JSON.stringify(obj)
      console.log(objJSONString) // {"name":"why","age":18}
      
      var newObj = JSON.parse(objJSONString, function(key, value) {
        if (key === "age") {
          return value + 2
        }
        return value
      })
      console.log(newObj) // {name: 'later', age: 20}
  • JSON的方法可以帮我们实现对象的深拷贝:

    • 但是目前我们还没有了解什么是对象的拷贝、浅拷贝、深拷贝的概念
    • 我们会在JS高级中学习

注意:

  • JSON.parse() 不允许用逗号作为结尾

    js
    // both will throw a SyntaxError
    JSON.parse("[1, 2, 3, 4, ]");
    JSON.parse('{"foo" : 1, }');

七. Storage


1. 认识Storage

  • WebStorage主要提供了一种机制,可以让浏览器提供一种比cookie更直观的keyvalue存储方式:
    • 如果你想要操作一个域名的本地存储,可以使用:

      • localStorage:本地存储,提供的是一种永久性的存储方法,在关闭掉网页重新打开时,存储的内容依然保留
    • 如果想要操作一个域名的会话存储,可以使用:

      • sessionStorage:会话存储,提供的是本次会话的存储,在关闭掉会话时,存储的内容会被清除

        image-20220624223600580

2. localStorage和sessionStorage的区别

  • 验证一:关闭网页后重新打开,localStorage会保留,而sessionStorage会被删除
  • 验证二:在页面内实现跳转,localStorage会保留,sessionStorage也会保留
  • 验证三:打开新的网页,localStorage会保留,sessionStorage不会被保留

3. Storage常见的属性和方法

  • Storage有如下的属性和方法:

  • 属性:

    • Storage.length:只读属性
      • 返回一个整数,表示存储在Storage对象中的数据项数量
  • 方法:

    方法作用
    Storage.key()该方法接受一个数值n作为参数,返回存储中的第nkey名称
    Storage.getItem()该方法接受一个key作为参数,并且返回key对应的value
    Storage.setItem()该方法接受一个keyvalue,并且把keyvalue添加到存储中
    如果key已有值存储,则更新其对应的值
    Storage.removeItem()该方法接受一个key作为参数,并把该key从存储中删除
    Storage.clear()该方法的作用是清空存储中的所有key

4. 监听 localStorage 的改变

  • 当前页面使用的 localstorage 被其他页面修改时会触发 StorageEvent 事件

    注意:

    • 事件在同一个域下的不同页面之间触发,即在 A 页面注册了 storge 的监听处理,只有在跟 A 同域名下的 B 页面操作 storage 对象,A 页面才会被触发 storage 事件
    • sessionStorage 无效,只支持 localStorage
  • A 页面

    html
    // A 页面
    <!DOCTYPE html>
    <html lang="zh-CN">
    <head>
      <meta charset="UTF-8">
      <meta http-equiv="X-UA-Compatible" content="IE=edge">
      <meta name="viewport" content="width=device-width, initial-scale=1.0">
      <title>Document</title>
    </head>
    <body>
      <button class="set-storage">设置storage</button>
      <button class="update-storage">修改storage</button>
    
      <script>
        const btnEl = document.querySelector('.set-storage') 
        var obj = {
          name: 'obj'
        } 
        btnEl.onclick = function() {
          localStorage.setItem('name', JSON.stringify(obj))
        }
        const btnEl2 = document.querySelector('.update-storage')  
        btnEl2.onclick = function() {
          localStorage.setItem('name', 'aaa')
        }
    
        // 监听本页面的localStorage改变
        // 重写localStorage.setItem方法, 抛出自定义事件
        var rawSetItem = window.localStorage.setItem
        localStorage.setItem = function(key, value) {
          var myStorageSetItem = new Event('myStorageSetItem')
          myStorageSetItem._key = key
          myStorageSetItem._oldValue = localStorage.getItem(key)
          myStorageSetItem._newValue = value
          dispatchEvent(myStorageSetItem)
          rawSetItem.call(this, key, value)
        }
        addEventListener('myStorageSetItem', function(e) {
          console.log('e: ', e)
        })
      </script>
    </body>
    </html>
  • B 页面

    html
    <!DOCTYPE html>
    <html lang="zh-CN">
    <head>
      <meta charset="UTF-8">
      <meta http-equiv="X-UA-Compatible" content="IE=edge">
      <meta name="viewport" content="width=device-width, initial-scale=1.0">
      <title>Document</title>
    </head>
    <body>
      <script>
        // 必须是给window对象上注册,document无效
        addEventListener('storage', e => {
          console.log('e: ', e)
          console.log('JSON.parse(e.newValue): ', JSON.parse(e.newValue))
          console.log('e.newValue: ', e.newValue)
          console.log(JSON.parse(e.newValue) || null)
          console.log('e.storageArea: ', e.storageArea)
          console.log('e.url: ', e.url)
        })
      </script>
    </body>
    </html>
  • https://developer.mozilla.org/zh-CN/docs/Web/API/StorageEvent

Released under the MIT License.