二刷红宝书 -- 你真的懂script标签吗

📖阅读本文你学习到✨

  • 灵活使用<script>标签
  • 深入理解asyncdefer、动态加载script方式
  • 行内script和外部script的比较

在日常开发中可能很少在html里面写script标签,一般都是vue react一把梭。很少看见script的身影

  • vue3 <script setup>...</script>
    • ts版本<script setup lang='ts'>...</script>
  • vue2中 <script> export default {...}</script>
  • 函数式 reactexport default function Demo(){... return <></>}

但是其实<script>大有来头,搞不好还可以提高页面加载速度,提高输入延迟 (FID) ,首次绘制(FP)等性能指标

深入理解<script>元素

script有8大属性可以供开发者可选,分别是asyncsrc, crossorigin(CORS)defer,integritycharsetlanguagetype。 其中language,charset,type很少使用,现代浏览器也会忽视这些属性,这里就不在赘述,主要围绕剩下的属性展开。

前置知识

  • 使用script的方式

    • 直接嵌入到网页中即在htmlscript
    • 引入外部script文件 即
  • 浏览器解析script的顺序是按照页面出现的顺序解释,如果使用defer或者async属性另论,下面会解释

  • script中的src属性可以是外部域文件,同时你需要明白该域是个可信来源,Integerity属性是防范这种问题的一个武器

  • 标签位置

    • 如果将script元素放在页面的head标签内 ,这就意味着要等head所有的script资源下载完全页面才能显示,在此期间浏览器窗口完全空白。
    • 解决方法是 将script引用放在body元素中的内容后面, 这样页面会在处理js代码之前完全渲染页面,白屏时间变短 用户替体验效果更好!
HTML 复制代码
<!DOCTYPE html> 
<html>
    <head>
        <title>Example HTML Page</title> 
        <script src="example1.js"></script> 
        <script src="example2.js"></script> 
    </head>
<body>
<!-- 这里是页面内容 -->
</body>
</html>

修改版如下

HTML 复制代码
<!DOCTYPE html> 
<html>
    <head>
        <title>Example HTML Page</title> 
    </head>
<body>
    <!-- 这里是页面内容 -->
    <script src="example1.js"></script> 
    <script src="example2.js"></script> 
</body>
</html>

defer

概念

可选参数,表示脚本可以延迟到文档完全被解析和显示之后再执行。只对外部脚本文件有效。

主要作用:推迟执行脚本

<script defer> 表示脚本在执行的时候不会改变页面的结构,就相当于告诉浏览器立即下载,但延迟执行

HTML 复制代码
<!DOCTYPE html>
<html>
    <head>
        <title>Example HTML Page</title>
        <script defer src="example1.js"></script> 
        <script defer src="example2.js"></script>
    </head>
    <body>
    <!-- 这里是页面内容 -->
    </body>
</html>
  • <script defer>执行时机
    • 因为加上defer属性,script会在浏览器解析完html标签之后执行
    • script不一定会按顺序执行,也不一定会在DOMContentLoaded时间之前执行
    • 最佳方案是把要推迟执行的脚本放在body元素中的内容后面
  • 作用对象
    • defer属性只对外部script文件有效

async

概念

可选参数,表示应该立即开始下载脚本,但不能阻止其他页面动作,比如下载资源或等待其他脚本加载。只对外部脚本文件有效

主要作用: 异步执行脚本

HTML 复制代码
<!DOCTYPE html>
<html>
    <head>
    <title>Example HTML Page</title>
    <script async src="example1.js"></script>
    <script async src="example2.js"></script> 
    </head>
    <body>
    <!-- 这里是页面内容 -->
    </body>
</html>
  • 执行顺序
    • 不保证俩个script次序执行
    • 但是会保证在页面的load事件之前进行,但可能会在DOMContentLoaded之前或之后
  • 告诉页面不会使用document.write

动态加载脚本

JavaScript可以使用DOM API,可以通过DOM中动态添加script元素同样可以加载制定脚本

js 复制代码
let script = document.createElement('script')
script.src = 'demo.js'
document.head.appendChild(script)
  • 需要注意的是
    • 这种加载默认是异步的,你可以按需改成同步script.async = false
    • 这种方式对浏览器预加载器是不可见的,这会影响在资源获取队列中的优先级,可能会严重影响性能。解决办法是文档头部显示声明它们
html 复制代码
<link rel='preload' href='demo.js'>

行内代码和外部文件的比较

虽然可以直接在html中写JavaScript代码,但是最佳实践是引入外部文件,但也并不是强制性的规则,推荐的理由主要是俩个:

  • 可维护性。类似于组件化思想, 把html中需要的JavaScript抽离出单独的文件,再按需引入,更容易后期维护
  • 缓存。浏览器会更具体定的设置缓存所有外部链接的JavaScript文件,这意味着如果来个页面用到相同的文件也只需要下载一次,加载速度更快

参考

《JavaScript 高级程序设计 第四版》

相关推荐
Cyber4K4 分钟前
【Python专项】进阶语法-日志分类与分析(2)
开发语言·前端·python
匀泪11 分钟前
云原生(Kubernetes存储)
前端·chrome
前端之虎陈随易25 分钟前
为什么今天还会有新语言?MoonBit 想解决什么问题?
大数据·linux·javascript·人工智能·算法·microsoft·typescript
kyriewen38 分钟前
你等的Babel编译,够喝三杯咖啡了——用Rust重写的SWC,只需眨个眼
前端·javascript·rust
搬砖码1 小时前
同源多标签页通信 4 种方案,从入门到生产环境
前端·面试
张元清1 小时前
SSR 状态管理陷阱:defineStore vs defineContextStore
前端·javascript·面试
donecoding1 小时前
nrm、corepack、npm registry 三者的爱恨情仇
前端·node.js·前端工程化
小gaigagi1 小时前
从吉客云·奇门到MySQL的完整数据流
前端
悟空瞎说1 小时前
用 Rust 开发 QML 桌面应用(第二篇)—— 日志系统完整搭建
前端
LIO1 小时前
前端开发之Git 代码仓库管理详细教程
前端·git