[{"data":1,"prerenderedAt":-1},["ShallowReactive",2],{"\u002Fresource\u002Fdocument\u002Flist?undefined":3,"\u002Fresource\u002Fdocument\u002Fquery\u002Fsdhodlihphnpcg37?undefined":462,"\u002Fresource\u002Fadvertise\u002Flist?type=all?undefined":467},{"data":4,"status":460,"success":461},[5,148,202,291,332,370,420],{"books":6,"desc":145,"id":8,"image":146,"title":147},[7,40,63,78,93,105,117],{"cateId":8,"chapters":9,"desc":36,"id":11,"time":37,"title":38,"video":39},1,[10,15,18,21,24,27,30,33],{"bookId":11,"id":12,"indexOrder":13,"name":14},24,"8egfulw98v3h680j",0,"JavaSE 笔记（一）走进Java语言",{"bookId":11,"id":16,"indexOrder":13,"name":17},"pew6po6wrou23pk3","JavaSE 笔记（二）面向过程编程",{"bookId":11,"id":19,"indexOrder":13,"name":20},"eldst1fgrbdkmfs7","JavaSE 笔记（三）面向对象基础",{"bookId":11,"id":22,"indexOrder":13,"name":23},"48zphgkpjto8cath","JavaSE 笔记（四）面向对象高级篇",{"bookId":11,"id":25,"indexOrder":13,"name":26},"6r4llai92yc15j98","JavaSE 笔记（五）泛型程序设计",{"bookId":11,"id":28,"indexOrder":13,"name":29},"k6fmxd6qabgkwm9i","JavaSE 笔记（六）集合类与IO",{"bookId":11,"id":31,"indexOrder":13,"name":32},"qrd0xfttsz32gpqg","JavaSE 笔记（七）多线程与反射",{"bookId":11,"id":34,"indexOrder":13,"name":35},"td5tgn04nqmkrryt","JavaSE 笔记（八）GUI程序开发","基于Java25全新录制的SE课程",2025,"JavaSE 核心内容","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV163GGz2E8c",{"cateId":8,"chapters":41,"desc":59,"id":8,"time":60,"title":61,"video":62},[42,44,46,49,51,53,55,57],{"bookId":8,"id":43,"indexOrder":13,"name":14},"ibeeuwsbbi00undq",{"bookId":8,"id":45,"indexOrder":13,"name":17},"dncxjecdv4wciqcp",{"bookId":8,"id":47,"indexOrder":13,"name":48},"jviyz2hsht9ete5k","JavaSE 笔记（三）面向对象基础篇",{"bookId":8,"id":50,"indexOrder":13,"name":23},"qb9i6q9fap7bg1cc",{"bookId":8,"id":52,"indexOrder":13,"name":26},"hnkrjrkm3hjzeq6s",{"bookId":8,"id":54,"indexOrder":13,"name":29},"erpm32wduoaaqmrx",{"bookId":8,"id":56,"indexOrder":13,"name":32},"lfqtvxr7azumcwja",{"bookId":8,"id":58,"indexOrder":13,"name":35},"qs7gqok56gzc6idr","2022年制作的JavaSE版本",2022,"JavaSE 22年旧版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1YP4y1o75f\u002F",{"cateId":8,"chapters":64,"desc":75,"id":66,"time":60,"title":76,"video":77},[65,69,72],{"bookId":66,"id":67,"indexOrder":13,"name":68},2,"g96k66kczovvbm1i","JVM 笔记（一）走进JVM",{"bookId":66,"id":70,"indexOrder":13,"name":71},"ydd7n3jg8unc3clg","JVM 笔记（二）内存管理",{"bookId":66,"id":73,"indexOrder":13,"name":74},"r9dq37de0kaeauoi","JVM 笔记（三）类与类加载","了解Java的底层运作机制","Java JVM 虚拟机","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Er4y1r7as\u002F",{"cateId":8,"chapters":79,"desc":90,"id":81,"time":60,"title":91,"video":92},[80,84,87],{"bookId":81,"id":82,"indexOrder":13,"name":83},3,"asncyye9ya18gfar","JUC 笔记（一）再谈多线程",{"bookId":81,"id":85,"indexOrder":13,"name":86},"5tr1sm4ho6ygpt9q","JUC 笔记（二）并发编程核心",{"bookId":81,"id":88,"indexOrder":13,"name":89},"1scf51z5300mzxkh","JUC 笔记（三）并发编程进阶","你也可以成为多线程的主宰者","Java JUC 并发编程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1JT4y1S7K8\u002F",{"cateId":8,"chapters":94,"desc":102,"id":96,"time":60,"title":103,"video":104},[95,99],{"bookId":96,"id":97,"indexOrder":13,"name":98},4,"eedesc445ygiqhil","NIO 笔记（一）基础内容",{"bookId":96,"id":100,"indexOrder":13,"name":101},"ndz9t0uunrmfmv4n","NIO 笔记（二）Netty框架专题","编写畅快的高性能网络服务器","Java NIO 网络编程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1ar4y1J7mC\u002F",{"cateId":8,"chapters":106,"desc":114,"id":108,"time":60,"title":115,"video":116},[107,111],{"bookId":108,"id":109,"indexOrder":13,"name":110},5,"9890i8ofuadpwy2b","[扩展篇] Java 9-17新特性介绍",{"bookId":108,"id":112,"indexOrder":13,"name":113},"tsrkqvb6zpmtwh0n","[扩展篇] JavaSE关键字总结 笔记","精彩仍在继续，不要停止脚步","其他内容","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1tU4y1y7Fg\u002F",{"cateId":8,"chapters":118,"desc":141,"id":120,"time":142,"title":143,"video":144},[119,123,126,129,132,135,138],{"bookId":120,"id":121,"indexOrder":13,"name":122},6,"4db9h32opv7imszh","JavaSE 笔记（一）面向过程编程",{"bookId":120,"id":124,"indexOrder":13,"name":125},"c93u3v37br7hgn1q","JavaSE 笔记（二）面向对象基础篇",{"bookId":120,"id":127,"indexOrder":13,"name":128},"yglsjde9gi1jxkcb","JavaSE 笔记（三）泛型与集合类",{"bookId":120,"id":130,"indexOrder":13,"name":131},"ilhi987n986rmvo3","JavaSE 笔记（四）异常机制",{"bookId":120,"id":133,"indexOrder":13,"name":134},"pqv38vexmenglk4k","JavaSE 笔记（五）IO",{"bookId":120,"id":136,"indexOrder":13,"name":137},"jiq41n87i9ia7ilw","JavaSE 笔记（六）多线程",{"bookId":120,"id":139,"indexOrder":13,"name":140},"wn7x2mge9ws79zps","JavaSE 笔记（七）反射","此版本为早期录制的旧版本",2021,"JavaSE 21年旧版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Gv411T7pi\u002F","包含JavaSE基础路线全部教程笔记，打下坚实的基础","https:\u002F\u002Fpic2.zhimg.com\u002F80\u002Fv2-bf1a927f037a79f4d57d9ae543430a0d_1440w.webp","JavaSE 系列笔记 ☕️",{"books":149,"desc":199,"id":66,"image":200,"title":201},[150,166,178],{"cateId":66,"chapters":151,"desc":162,"id":153,"time":163,"title":164,"video":165},[152,156,159],{"bookId":153,"id":154,"indexOrder":13,"name":155},21,"iqbc2haub31bwqtz","Lombok 极速上手",{"bookId":153,"id":157,"indexOrder":13,"name":158},"ijay2hay19kn1k031","Mybatis 快速上手",{"bookId":153,"id":160,"indexOrder":13,"name":161},"ru4ogh2waocpn4jo","Maven 快速上手","JavaWeb阶段必须扩展知识点",2024,"常用知识讲解","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1gb421J7ok\u002F",{"cateId":66,"chapters":167,"desc":175,"id":169,"time":163,"title":176,"video":177},[168,172],{"bookId":169,"id":170,"indexOrder":13,"name":171},22,"ek20yvb6huhxizx7","JavaWeb 笔记（一）计算机网络基础",{"bookId":169,"id":173,"indexOrder":13,"name":174},"pgevws6w2krkffa4","JavaWeb笔记（二）Java与数据库","全面升级的JavaWeb课程","JavaWeb 网站开发","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1kS421X7rq\u002F",{"cateId":66,"chapters":179,"desc":196,"id":181,"time":142,"title":197,"video":198},[180,184,187,190,193],{"bookId":181,"id":182,"indexOrder":13,"name":183},7,"ggwwj09j2vkfftvd","JavaWeb 笔记（一）Java网络编程",{"bookId":181,"id":185,"indexOrder":13,"name":186},"sauvq105istskjaz","JavaWeb 笔记（二）数据库基础",{"bookId":181,"id":188,"indexOrder":13,"name":189},"xgbeasmvrhxx9tn4","JavaWeb 笔记（三）Java与数据库",{"bookId":181,"id":191,"indexOrder":13,"name":192},"k7dfwua3bsezvw9q","JavaWeb 笔记（四）前端基础",{"bookId":181,"id":194,"indexOrder":13,"name":195},"ycpagby2v7j4p728","JavaWeb 笔记（五）后端开发","搭建属于自己的Web网站","JavaWeb 旧版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1CL4y1i7qR\u002F","包含JavaWeb路线全套笔记，从零开始搭建自己的网站！","https:\u002F\u002Fpic3.zhimg.com\u002F80\u002Fv2-df3b38e3012258ed70c23b586309e3f6_1440w.webp","JavaWeb 系列笔记 🚛",{"books":203,"desc":288,"id":81,"image":289,"title":290},[204,220,235,255,273],{"cateId":81,"chapters":205,"desc":216,"id":207,"time":217,"title":218,"video":219},[206,210,213],{"bookId":207,"id":208,"indexOrder":13,"name":209},8,"h7sjo5oy0l03607e","SSM笔记（一）Spring基础",{"bookId":207,"id":211,"indexOrder":13,"name":212},"eve8gq72qmdb46sg","SSM笔记（二）SpringMvc基础",{"bookId":207,"id":214,"indexOrder":13,"name":215},"63v73g0zh1qlr6fk","SSM笔记（三）SpringSecurity基础","Spring的探索之路从这里开始",2023,"JavaSSM 基础部分","[\"https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Kv4y1x7is\u002F\", \"https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Lh4y1M7kx\u002F\", \"https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1fV411M7aS\u002F\"]",{"cateId":81,"chapters":221,"desc":232,"id":223,"time":217,"title":233,"video":234},[222,226,229],{"bookId":223,"id":224,"indexOrder":13,"name":225},16,"0k66v5r6slsfuog4","SpringBoot笔记（一）核心内容",{"bookId":223,"id":227,"indexOrder":13,"name":228},"bqlrnc2yvkaxo8s1","SpringBoot笔记（二）数据交互",{"bookId":223,"id":230,"indexOrder":13,"name":231},"wci9lb9tgea866jt","SpringBoot笔记（三）前后端分离","SpringBoot全新重制版","SpringBoot 新版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1xu4y1m7UP\u002F",{"cateId":81,"chapters":236,"desc":252,"id":238,"time":60,"title":253,"video":254},[237,240,243,246,249],{"bookId":238,"id":239,"indexOrder":13,"name":225},9,"e43gl1ilygps032v",{"bookId":238,"id":241,"indexOrder":13,"name":242},"emnmd8nzfdb3hr50","SpringBoot笔记（二）Git版本控制",{"bookId":238,"id":244,"indexOrder":13,"name":245},"jjlolj5igvttvyhv","SpringBoot笔记（三）Redis数据库",{"bookId":238,"id":247,"indexOrder":13,"name":248},"skgr4ivb5curdoux","SpringBoot笔记（四）其他框架介绍",{"bookId":238,"id":250,"indexOrder":13,"name":251},"le91fqhu4dqui1k4","SpringBoot笔记（五）Linux系统","逐步走向企业级开发","SpringBoot 旧版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1UL411V7f3\u002F",{"cateId":81,"chapters":256,"desc":270,"id":258,"time":60,"title":271,"video":272},[257,261,264,267],{"bookId":258,"id":259,"indexOrder":13,"name":260},10,"oejzo0l77zeb6a7e","SpringCloud笔记（一）微服务基础",{"bookId":258,"id":262,"indexOrder":13,"name":263},"f6eya9taaelsl35p","SpringCloud笔记（二）微服务进阶",{"bookId":258,"id":265,"indexOrder":13,"name":266},"35v1hbsfcdgagdnw","SpringCloud笔记（三）微服务应用",{"bookId":258,"id":268,"indexOrder":13,"name":269},"a782u84512tyuo1m","SpringCloud笔记（四）消息队列","体验微服务架构带来的魅力","SpringCloud 进阶","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1AL4y1j7RY\u002F",{"cateId":81,"chapters":274,"desc":285,"id":276,"time":142,"title":286,"video":287},[275,278,280,282],{"bookId":276,"id":277,"indexOrder":13,"name":209},11,"efjw75u8a251qxk5",{"bookId":276,"id":279,"indexOrder":13,"name":212},"guc134xb7sl78vju",{"bookId":276,"id":281,"indexOrder":13,"name":215},"u8ekxxucowr2b1tm",{"bookId":276,"id":283,"indexOrder":13,"name":284},"vkpmw9wbej21nei6","SSM笔记（四）MySQL进阶","此教程为2021年旧版教程","JavaSSM 旧版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1xL4y1H7Tq\u002F","包含Spring全套框架笔记，从开始到Spring Boot，以及众多运维小知识。","https:\u002F\u002Fpic4.zhimg.com\u002F80\u002Fv2-28c3144421220d7c048703281bc34f63_1440w.webp","Spring 系列笔记 🍏",{"books":292,"desc":329,"id":96,"image":330,"title":331},[293,308],{"cateId":96,"chapters":294,"desc":305,"id":296,"time":60,"title":306,"video":307},[295,299,302],{"bookId":296,"id":297,"indexOrder":13,"name":298},12,"jd3e8u5cmvx5gco6","C语言（一）计算机思维导论",{"bookId":296,"id":300,"indexOrder":13,"name":301},"lqv77apvx82nkkio","C语言（二）基础语法",{"bookId":296,"id":303,"indexOrder":13,"name":304},"xb0b9t37gyv96xns","C语言（三）高级特性","包含高等院校需要教授的全部内容","C语言程序设计","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Cr4y137os\u002F",{"cateId":96,"chapters":309,"desc":326,"id":311,"time":60,"title":327,"video":328},[310,314,317,320,323],{"bookId":311,"id":312,"indexOrder":13,"name":313},13,"8a046ps2e4w6k4py","数据结构与算法（一）线性结构篇",{"bookId":311,"id":315,"indexOrder":13,"name":316},"3ma8db91f9zrnkja","数据结构与算法（二）树形结构篇",{"bookId":311,"id":318,"indexOrder":13,"name":319},"0lsjm59k7cgu4tpr","数据结构与算法（三）散列表篇",{"bookId":311,"id":321,"indexOrder":13,"name":322},"0qzy7bogo0g2pusa","数据结构与算法（四）图结构篇",{"bookId":311,"id":324,"indexOrder":13,"name":325},"6gmcxcikcilyxblj","数据结构与算法（五）排序算法篇","虽然很难，但是它是考研必学科目","数据结构与算法","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV13W4y127Ey\u002F","你的内心一直有一个坚定的声音在告诉你，一定要考上一名研究生，向着未来前进吧！","https:\u002F\u002Fpic2.zhimg.com\u002F80\u002Fv2-ac128404efb29ce1c9d1ccc61024f1d1_1440w.webp","C语言 系列笔记 🥬",{"books":333,"desc":367,"id":108,"image":368,"title":369},[334,349,358],{"cateId":108,"chapters":335,"desc":346,"id":337,"time":163,"title":347,"video":348},[336,340,343],{"bookId":337,"id":338,"indexOrder":13,"name":339},17,"urw2e6gg1lprv65w","Kotlin（一）基础语法",{"bookId":337,"id":341,"indexOrder":13,"name":342},"t7lnl87f74f3v1ju","Kotlin（二）类与对象",{"bookId":337,"id":344,"indexOrder":13,"name":345},"v1zzvki0knb1xvml","Kotlin（三）高级特性","包含Kotlin语言完整基础部分","Kotlin程序设计基础","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1P94y1c7tV\u002F",{"cateId":108,"chapters":350,"desc":355,"id":352,"time":163,"title":356,"video":357},[351],{"bookId":352,"id":353,"indexOrder":13,"name":354},18,"ovbzpe7065bye1st","Kotlin扩展（一）","包含Kotlin额外扩展知识","Kotlin扩展篇","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Hg4y1m7Ca\u002F",{"cateId":108,"chapters":359,"desc":364,"id":361,"time":163,"title":365,"video":366},[360],{"bookId":361,"id":362,"indexOrder":13,"name":363},19,"3at7ybv04dmjc0wp","Gradle基础教程","Gradle配置教程（Kotlin）","Gradle教程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Fc411x7xF\u002F","Kotlin让JVM平台焕发新的生机，让语言的表达更加优美","https:\u002F\u002Fpic2.zhimg.com\u002F80\u002Fv2-be815568f7c79c64cdaa171b0409786d_1440w.webp","Kotlin 系列笔记 ☘️",{"books":371,"desc":418,"id":120,"title":419},[372,391,403],{"cateId":120,"chapters":373,"desc":387,"id":375,"time":388,"title":389,"video":390},[374,378,381,384],{"bookId":375,"id":376,"indexOrder":13,"name":377},26,"zjf5qapwqtqiohcn","JavaScript笔记（一）基础语法",{"bookId":375,"id":379,"indexOrder":13,"name":380},"95jc6sjyjwcp9pvp","JavaScript笔记（二）核心知识",{"bookId":375,"id":382,"indexOrder":13,"name":383},"j35cdc1qz8dzq7pn","JavaScript笔记（三）进阶知识",{"bookId":375,"id":385,"indexOrder":13,"name":386},"sdhodlihphnpcg37","JavaScript笔记（四）前端基础","包含JavaScript最新语法规范讲解",2026,"JavaScript教程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1xq6gBgESU",{"cateId":120,"chapters":392,"desc":400,"id":394,"time":37,"title":401,"video":402},[393,397],{"bookId":394,"id":395,"indexOrder":13,"name":396},23,"bsisgazdftiz3o9c","HTML5笔记（一）基础内容",{"bookId":394,"id":398,"indexOrder":13,"name":399},"njol93fs34gfwuzf","HTML5笔记（二）高级内容","包含HTML基础内容和相关知识点","HTML5核心教程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1BrBiYNEWg",{"cateId":120,"chapters":404,"desc":415,"id":406,"time":37,"title":416,"video":417},[405,409,412],{"bookId":406,"id":407,"indexOrder":13,"name":408},25,"jo74ciirtg8wh90y","CSS笔记（一）基础入门",{"bookId":406,"id":410,"indexOrder":13,"name":411},"ap5ixyomoejuw4ue","CSS笔记（二）盒模型和布局",{"bookId":406,"id":413,"indexOrder":13,"name":414},"4djgk5xy1lzpiuf2","CSS笔记（三）变换和过渡","包含CSS3基础内容和相关知识点","CSS3核心教程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1sQeEzFEKi","包含Web前端学习路径全部教程笔记，打下坚实的基础","Web前端 系列笔记",{"books":421,"desc":458,"id":423,"image":368,"title":459},[422,432,450],{"cateId":423,"chapters":424,"desc":429,"id":426,"time":163,"title":430,"video":431},100,[425],{"bookId":426,"id":427,"indexOrder":13,"name":428},20,"o0ab271mkdsas87","Markdown基础语法","编写简洁而又优美的文档","Markdown教程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1eJ4m157kC",{"cateId":423,"chapters":433,"desc":447,"id":435,"time":60,"title":448,"video":449},[434,438,441,444],{"bookId":435,"id":436,"indexOrder":13,"name":437},14,"6386mh7anqt4tzyv","设计模式（一）面向对象设计原则",{"bookId":435,"id":439,"indexOrder":13,"name":440},"8ftkb38wfn6ox0ug","设计模式（二）创建型",{"bookId":435,"id":442,"indexOrder":13,"name":443},"i1msql1k8y70etey","设计模式（三）结构型",{"bookId":435,"id":445,"indexOrder":13,"name":446},"5434a3cyyjvwhs8s","设计模式（四）行为型","使你的编码水平得到质的飞跃","设计模式系列","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1u3411P7Na\u002F",{"cateId":423,"chapters":451,"desc":456,"id":453,"time":60,"title":457},[452],{"bookId":453,"id":454,"indexOrder":13,"name":455},15,"zj9uvg0sp3b0sok8","Docker 容器技术 笔记","这里包含其他中间件课程笔记","其他中间件笔记","我们对知识的探索从未停止，只有不断地学习，才能走向美好的未来！","其他笔记分类 🌽",200,true,{"data":463,"status":460,"success":461},{"bookId":375,"content":464,"id":385,"indexOrder":96,"introduction":465,"lastUpdate":466,"name":386},"![image-20260212203819527](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F12\u002FpR5r\u002Fimage-20260212203819527.png)\n\n# DOM核心知识\n\n**注意：** 本章知识仅适用于Web前端开发学习路线或全栈开发路线，如果仅使用JavaScript做后端开发，不强制要求学习。\n\n前面我们为大家介绍了JS的基础语法部分，这一章我们将继续深入Web浏览器环境，学习然后控制页面上的元素并实现各种高级操作。\n\n## 元素和属性\n\n在实际开发中，我们经常需要动态控制页面上的元素样式，比如当鼠标点击元素时，我们可以改变它的颜色，点击按钮时可以展示弹窗，用户滑动窗口时实时展示当前滚动的进度。这些效果都需要通过DOM提供的方法来实现，我们可以利用JavaScript与其进行交互，让我们的网站动起来。\n\n### DOM介绍\n\nDOM（Document Object Model，文档对象模型）可以理解为：**浏览器把 HTML 页面转换成的一棵“对象树”**。这棵树里的每一个节点，都是 JavaScript 可以操作的对象。比如下面的代码：\n\n```html\n\u003C!DOCTYPE html>\n\u003Chtml lang=\"en\">\n  \u003Chead>\n    \u003Cmeta charset=\"UTF-8\">\n    \u003Ctitle>Title\u003C\u002Ftitle>\n  \u003C\u002Fhead>\n  \u003Cbody>\n    \u003Cdiv>我是页面上的一段普通内容\u003C\u002Fdiv>\n    \u003Cp>我是一个段落\u003C\u002Fp>\n  \u003C\u002Fbody>\n\u003C\u002Fhtml>\n```\n\n浏览器会把上面的代码自动转换为下面的树形结构：\n\n![image-20260212212153383](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F12\u002FSfg2\u002Fimage-20260212212153383.png)\n\n在HTML中，这种结构叫做DOM树，我们需要将这个树形结构倒过来看，HTML元素就是这棵树的树根，而它的子元素，实际上就是这棵树上的分支。所有的元素，在浏览器加载页面后，会在内存中生成一个对应的 DOM 节点对象，此时 JavaScript 就可以直接操作这个节点。\n\n在JS中，节点分为很多种类型，一共有14种类型，但其中比较常见的是以下五种类型：\n\n1. **元素节点**（`ELEMENT_NODE`）：操作 HTML 元素的核心。\n2. **文本节点**（`TEXT_NODE`）：处理元素内的文本内容。\n3. **注释节点**（`COMMENT_NODE`）：有时用于模板标记或调试。\n4. **文档节点**（`DOCUMENT_NODE`）：全局 `document` 对象，用于访问整个页面。\n5. **文档片段节点**（`DOCUMENT_FRAGMENT_NODE`）：高效批量添加 DOM 元素。\n\n比如我们编写的元素就是一个节点，也就是 HTML 标签本身，它是页面结构的核心组成部分，下面这些都是元素节点：\n\n```html\n\u003Cdiv class=\"container\">\u003C\u002Fdiv>\n\u003Cp>Hello World\u003C\u002Fp>\n\u003Ca href=\"https:\u002F\u002Fexample.com\">链接\u003C\u002Fa>\n```\n\n而文本节点，就是包含在元素或属性中的纯文本内容，包括空格、换行等空白字符：\n\n```html\n\u003Cdiv>\n  这里是文本节点\n  \u003Cp>Hello \u003Cspan>World\u003C\u002Fspan>\u003C\u002Fp>\n\u003C\u002Fdiv>\n```\n\n此外，HTML 中的注释内容，虽然不会在页面中显示，但可以通过 DOM 访问，这些是注释节点：\n\n```html\n\u003C!-- 这是一个注释节点 -->\n```\n\n整个文档的根就是文档节点，一般情况下是`html`标签：\n\n```html\n\u003Chtml lang=\"en\">\n  ...\n\u003C\u002Fhtml>\n```\n\n几乎在HTML文件中出现的所有内容，都会变成一个JS可以控制的节点，哪怕是注释也会被算在其中。因此，**DOM 让 JavaScript 有了“操作页面的能力”**。下一节我们将介绍如何通过JS获取DOM上的节点。\n\n### 获取元素\n\n在操作页面之前，第一步永远是：**先找到元素**，通过JS查找元素有很多种方式，浏览器为我们提供了多种“查找 DOM 元素”的方式，下面我们从最基础、最容易理解的方法开始，一步步往后讲。\n\n我们需要用到`doucment`对象，它是整个DOM的总管，代表当前这个HTML文档，我们可以通过它来获取DOM上的任何内容。第一种是通过 id 获取元素（`getElementById`）这是**最常用、也是新手最先学会的方法**：\n\n```js\ndocument.getElementById(id)\n```\n\n这里的参数是 **元素的 id 名称（字符串）**，返回值是 **一个元素对象**，如果找不到元素，则返回 `null`：\n\n```html\n\u003Cdiv id=\"box\">我是一个盒子\u003C\u002Fdiv>\n```\n\n```js\nconst box = document.getElementById('box')\nconsole.log(box)\n```\n\n可以看到，这里得到了一个HTML元素对象，类型是`HTMLElement`，HTML中所有的元素类型都是`Node`的子类，继承关系和顺序如下：\n\n1. **EventTarget**：最顶层的基类，让对象拥有处理事件的能力（比如 `addEventListener`）。\n2. **Node**：基础的节点类，DOM 树中的所有内容（标签、文本、注释等）都是 Node。\n3. **Element**：更具体的节点，它专门指代“标签”，提供了处理属性（Attribute）的方法。\n4. **HTMLElement**：HTML 专用的元素，它包含了所有 HTML 标签共有的属性（比如 `id`, `className`, `style`）。\n5. **具体标签**（如 `HTMLDivElement`）：特定标签的属性（比如 `\u003Ca>` 标签独有的 `href`）\n\n我们如果直接打印它，会以HTML代码的形式在控制台展示：\n\n![image-20260213012627718](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F12\u002F8nnS\u002Fimage-20260213012627718.png)\n\n我们可以通过`nodeType`来获取它的具体类型：\n\n```js\nconsole.log(box.nodeType === Node.ELEMENT_NODE)\n```\n\n由于得到的结果是一个数字，我们直接使用`Node`构造函数中提供的常量来进行比较即可。\n\n当然，我们也可以在一个单独的JS文件中编写这段代码：\n\n```html\n\u003C!DOCTYPE html>\n\u003Chtml lang=\"en\">\n\u003Chead>\n  \u003Cmeta charset=\"UTF-8\">\n  \u003Ctitle>Title\u003C\u002Ftitle>\n  \u003Cscript src=\"js\u002Fscript.js\">\u003C\u002Fscript>\n\u003C\u002Fhead>\n```\n\n```js\n\u002F\u002Fscript.js\nconst box = document.getElementById('box')\nconsole.log(box)\n```\n\n此时我们会发现，这样好像不太行，控制台中获取到的元素是一个`null`：\n\n![image-20260213115140379](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002FV0zf\u002Fimage-20260213115140379.png)\n\n这其实是因为我们的JS引入是写在`head`标签内的，而页面内容是在`body`中，网页的加载顺序是从上往下进行的，当JS加载的时候，页面内容还未渲染完成（DOM树还未完成构建）所以就无法拿到需要的元素了。\n\n这里推荐几种做法，最简单的就是将 `\u003Cscript>` 标签移到 `\u003C\u002Fbody>` 结束标签之前：\n\n```html\n\u003Cbody>\n  \u003Cdiv>我是页面上的一段普通内容\u003C\u002Fdiv>\n  \u003Cdiv id=\"box\">我是一个盒子\u003C\u002Fdiv>\n  \u003C!-- 放在最后执行，此时前面的元素都已经完成加载 -->\n  \u003Cscript src=\"js\u002Fscript.js\">\u003C\u002Fscript>\n\u003C\u002Fbody>\n```\n\n还有一种方法是给 `\u003Cscript>` 添加 `defer` 属性，这个属性可以使得JS延迟加载，确保脚本执行时能访问到所有 DOM 元素：\n\n```html\n\u003Chead>\n  \u003Cmeta charset=\"UTF-8\">\n  \u003Ctitle>Title\u003C\u002Ftitle>\n  \u003Cscript defer src=\"js\u002Fscript.js\">\u003C\u002Fscript>\n\u003C\u002Fhead>\n```\n\n我们也可以使用`getElementsByTagName`根据标签名字一次性获取所有元素：\n\n```js\ndocument.getElementsByTagName(\"div\")\n```\n\n![image-20260213123903844](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002FahJ0\u002Fimage-20260213123903844.png)\n\n这里得到的是一个`HTMLCollection`，它是一个类数组（array-like）集合对象，包含了一组有序的 DOM 元素，使用方法非常简单：\n\n```js\nconst elements = document.getElementsByTagName(\"div\")\nconsole.log(elements.item(0))   \u002F\u002F使用item访问指定下标的元素\nconsole.log(elements[1])   \u002F\u002F也可以和数组一样玩\n```\n\n![image-20260213141603054](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002F9kvD\u002Fimage-20260213141603054.png)\n\n此外，`HTMLCollection`还提供了一个`namedItem`用于快速查找集合中`name`或`id`属性为指定值的元素：\n\n```js\nconsole.log(elements.namedItem(\"box\"))\n```\n\n我们也可以使用`getElementsByClassName`通过 `class` 名称来获取元素：\n\n```js\nconst elements = document.getElementsByClassName(\"item\")\nconsole.log(elements)\n```\n\n```html\n\u003Cdiv class=\"item\">A\u003C\u002Fdiv>\n\u003Cdiv class=\"item\">B\u003C\u002Fdiv>\n```\n\n我们还可以根据`name`属性来获取元素，这个方法在现代开发中**用得不多**，主要用于表单：\n\n```html\n\u003Cinput type=\"text\" name=\"username\">\n\u003Cinput type=\"password\" name=\"password\">\n```\n\n```js\nconst inputs = document.getElementsByName('username')\n```\n\n除了根据名字属性获取外，`document`上直接提供了根元素的获取，`documentElement`直接代表页面根元素，也就是`html`元素：\n\n```js\ndocument.documentElement\n```\n\n![image-20260213153648979](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002FeF6o\u002Fimage-20260213153648979.png)\n\n此外，`body`属性也是直接代表`body`这个元素：\n\n![image-20260213160213836](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002FtFt9\u002Fimage-20260213160213836.png)\n\n最后我们来介绍一个最重要的，也是开发中最常用的一种元素获取方式，`querySelector`可以使用 CSS 选择器获取单个元素：\n\n```js\ndocument.querySelector(\"#box\")\n```\n\n要判断我们的选择器是否使用正确，也可以使用`matches`进行判断：\n\n```js\nbox.matches(\"#box\")   \u002F\u002F当CSS选择器匹配时，返回真\n```\n\n利用这个方法，可以替代前面讲到的全部方法，只要CSS选择器编写合理，就能直接选择需要的元素，非常方便。不过需要注意的是，这个方法只会返回匹配到的第一个元素，如果需要一次性获取所有满足CSS选择器的元素，可以使用`querySelectorAll`方法：\n\n```js\ndocument.querySelectorAll(\".item\")\n```\n\n它会返回一个`NodeList`对象，和我们前面介绍的`HTMLCollection`不同的是，**HTMLCollection** 仅包含 **Element（元素节点）**比如 `\u003Cdiv>`、`\u003Cp>` 等标签。而 **NodeList** 包含所有 **Node（节点）**元素，所以还包括**文本节点**（换行、空格）、**注释节点**以及属性节点，因为CSS选择器选择的范围更加广泛，所以它应该包含更多种类。它提供了`forEach`用于快速遍历子节点：\n\n```js\ndocument.querySelectorAll(\".item\").forEach((value, key, parent) => {\n    console.log(key, value, parent)   \u002F\u002Fkey就是顺序下标，value就是节点本身，parent就是当前这个节点列表\n})\n```\n\n此外，还有一个非常重要的是，`HTMLCollection`是动态的，一旦DOM发生变化，这个对象会跟着变化，而NodeList是获取时产生的静态快照，我们可以尝试在打印对象之后删除页面上的一个元素：\n\n![image-20260213152017011](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002F1qaO\u002Fimage-20260213152017011.png)\n\n此时`HTMLCollection`的长度会变化成`1`，而`NodeList`无感。\n\n以上讲解的方法，除了在`document`上使用，我们也可以在任意一个元素上使用，来实现快速查找子元素：\n\n```js\nconst box = document.getElementById('box')\nbox.querySelector(\"p\")   \u002F\u002F直接在这个box元素上查找内部符合条件的子元素\n```\n\n此外，我们还可以使用：\n\n```js\nelement.closest('.container')   \u002F\u002F用于向上查找符合选择器的祖先元素\n```\n\n查找到元素之后，我们就可以开始愉快的操控它了。\n\n### HTML属性\n\n在前面我们已经学会了如何**获取元素**，但仅仅拿到元素还不够，真正让页面“动起来”的关键，是**读取和修改元素的属性**。**属性（Attribute）**，简单理解就是写在 HTML 标签上的那些“信息”，比如：\n\n```html\n\u003Cdiv id=\"box\" class=\"item\" title=\"提示文本\">\u003C\u002Fdiv>\n```\n\n这里的 `id`、`class`、`title`，都是元素的属性，这是我们在HTML中为大家介绍的。需要获取HTML标签上的属性，我们可以直接使用`getAttribute`方法：\n\n```js\nconsole.log(element.getAttribute(\"id\"))\n```\n\n注意返回值**永远是字符串**，因为属性都是直接写的值，如果属性不存在，会返回 `null`。除了获取之外，我们也可以使用`setAttribute`来修改某个属性：\n\n```js\nelement.setAttribute(\"class\", \"666\")\n```\n\n![image-20260214170321600](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002F6Mwu\u002Fimage-20260214170321600.png)\n\n如果这个属性不存在，那么将会新增这个属性：\n\n```js\nbox.setAttribute(\"data-v-25565\", \"666\")\n```\n\n![image-20260214171513288](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002FgMc7\u002Fimage-20260214171513288.png)\n\n不过，对于一些非标准的HTML标签属性，我们更建议大家使用后续DOM属性中的`dataset`来实现。\n\n对于一些布尔属性来说，设置不需要填写特别的内容，如果要使用直接设置值为空即可，如果不需要则移除属性，比如：\n\n```html\n\u003Cinput disabled=\"disabled\">\n```\n\n```js\ninput.removeAttribute('disabled') \u002F\u002F 表示 false\ninput.setAttribute('disabled', '') \u002F\u002F 表示 true\n```\n\n这里需要注意的是，**HTML 属性名本身不区分大小写**，所以即使我们写一个大写的也会变成小写形式，覆盖也会按照小写的形式去进行覆盖：\n\n```js\nbox.setAttribute(\"CLASS\", \"666\")\n```\n\n![image-20260214171915265](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002FKda2\u002Fimage-20260214171915265.png)\n\n不过，这里还是建议大家在 **JavaScript 中调用方法时，属性名统一推荐使用小写**，这是业界的通用约定，避免出现兼容性和可读性问题。\n\n有时候我们并不关心属性的值，只关心它**有没有被写在标签上**，这时可以使用`hasAttribute`来判断：\n\n```js\nelement.hasAttribute('title')\n```\n\n我们也可以使用 `removeAttribute` 删除某个属性：\n\n```js\nelement.removeAttribute('title')\n```\n\n如果属性本身不存在，**不会报错**，但也不会有任何效果。\n\n### DOM属性\n\n在上一节中我们讲了 **HTML 属性（Attribute）**，它们是**写在标签上的信息**。而这一节要讲的 **DOM 属性（Property）**，是**浏览器在内存中为每一个元素创建的 JavaScript 对象属性**。所以，HTML 属性是“写给浏览器看的”，DOM 属性是“写给 JavaScript 用的”。\n\n当浏览器加载 HTML 页面时，会做两件事：\n\n1. 解析 HTML 标签上的属性（Attribute）\n2. 基于这些属性，**生成一个对应的 DOM 对象**\n\n比如一个`input`标签：\n\n```html\n\u003Cinput type=\"text\" value=\"hello\">\n```\n\n浏览器会创建一个 `HTMLInputElement` 对象，其中包含大量 DOM 属性：\n\n```js\ninput.value\ninput.type\ninput.disabled\ninput.checked\ninput.placeholder\n```\n\n这些属性并不一定全部写在 HTML 中，但 **只要元素类型支持，就一定存在于 DOM 对象上**。想要直接观察DOM元素的各个属性，可以直接打开开发者工具，然后再元素界面选择需要观察的元素，接着在下面找到属性：\n\n![image-20260124000847227](https:\u002F\u002Fs2.loli.net\u002F2026\u002F01\u002F24\u002FVTxgbqYJ6FhGLBE.png)\n\nDOM 属性的访问方式非常直观，**就像普通 JS 对象一样**，比如我们想要修改元素的`id`属性，直接对其进行修改即可，修改后会立即生效到元素上：\n\n```js\nconst box = document.getElementById('box')\nbox.id = 'crazy'\n```\n\n![image-20260214210437028](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002FnlS9\u002Fimage-20260214210437028.png)\n\n对于一些非标准HTML属性，我们也可以使用`dataset`来实现：\n\n```js\nbox.dataset.role = 'guest'\n```\n\n![image-20260215153512552](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F15\u002F3sqR\u002Fimage-20260215153512552.png)\n\n此外，它还可以实现自动解析驼峰的效果：\n\n```js\nconsole.log(box.dataset.userId)  \u002F\u002F获取的其实是data-user-id\n```\n\n不过，和前面介绍的 HTML 属性的一个区别是，DOM 属性的值不一定是字符串。比如一些布尔属性，它们的值就是布尔类型的：\n\n```js\nconsole.log(box.draggable)  \u002F\u002Ffalse\n```\n\n这里我们着重介绍两个比较常用的DOM属性，首先是用于设置`class`的`className`和`classList`属性：\n\n```js\nconsole.log(box.className)   \u002F\u002F字符串结果\nbox.className = \"crazy item\"\n```\n\n修改`className`后，元素的`class`属性也会一起跟着变化，但是如果我们需要追加新的类，还需要手动进行拼接，这显然是非常麻烦的，所以我们更推荐大家使用下面的`classList`属性来修改类，它使用起来更加方便：\n\n```js\nconsole.log(box.classList)   \u002F\u002F得到一个类列表\n```\n\n这里得到的是一个`DOMTokenList`对象，它可以实现对所有以空格分割的属性进行快捷编辑，我们可以直接通过它来进行类的增加和删除，直接使用`add`就可以插入新的类了，就像使用数组那样：\n\n```js\nbox.classList.add('crazy', 'thursday', 'vivo', '50yuan')\n```\n\n![image-20260214215036006](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002FJj7s\u002Fimage-20260214215036006.png)\n\n我们也可以使用`remove`来对属性进行删除：\n\n```js\nbox.classList.remove(\"item\")\n```\n\n![image-20260214215306162](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002FKkw0\u002Fimage-20260214215306162.png)\n\n其他方法也很好用：\n\n```js\nbox.classList.toggle('active')   \u002F\u002F切换类，如果有则删除，没有就新增\nbox.classList.toggle('active', true)   \u002F\u002F切换类，强制设置\nbox.classList.contains('active')   \u002F\u002F判断是否包含类\nbox.classList.replace(\"item\", \"crazy\")  \u002F\u002F把一个类替换成另外一个类\n```\n\n我们接着来看如何更改内联样式，使用`style`即可：\n\n```html\n\u003Cdiv id=\"box\" style=\"background: pink\">我是文本\u003C\u002Fdiv>\n```\n\n```js\nconsole.log(box.style)\n```\n\n`style`属性中包含了目前几乎所有可用的CSS属性，需要调整哪个属性，只需要直接对其进行赋值即可。比如我们现在需要修改文本的颜色：\n\n```js\nbox.style.color = 'white'   \u002F\u002F等价于 color: white; 白色\n```\n\n![image-20260214225933760](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002FeQp7\u002Fimage-20260214225933760.png)\n\n此时页面上的元素`style`属性也会跟着发生变化，每个属性的设置都是独立的，不会影响其他属性。\n\n当然，DOM属性不仅包含HTML标签上的属性，还有一些元素自己本身的属性，比如前面介绍的`innerText`、`nodeType`等，其实都是DOM属性。有关其他DOM属性，我们会在后续课程中逐步进行讲解，大家在使用时也可以在MDN文档上自行查询需要的属性。\n\n### 元素内容\n\n在前面我们已经学习了如何获取元素，以及元素的属性，这一节我们接着来介绍几个比较特殊的属性，它们可以操控元素的内容。在所有的元素对象中，`innerHTML` 用来 **获取或设置元素内部的 HTML 结构**：\n\n```html\n\u003Cdiv id=\"box\">\n  \u003Cspan>hello\u003C\u002Fspan>\n\u003C\u002Fdiv>\n```\n\n```js\nconst box = document.getElementById('box')\nconsole.log(box.innerHTML)\n\u002F\u002F \u003Cspan>hello\u003C\u002Fspan>\n```\n\n可以看到，`innerHTML`代表的是元素内部的HTML文档结构，打印出来的值也是一个HTML格式的代码。同样的，既然可以获取，我们也可以使用这个属性来修改内部的HTML代码：\n\n```js\nbox.innerHTML = '\u003Cp>新的内容\u003C\u002Fp>'\n```\n\n![image-20260213164153780](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002F9qYs\u002Fimage-20260213164153780.png)\n\n虽然这种做法在修改HTML结构上简单粗暴，但是它存在诸多问题，其中最主要的就是安全和性能问题：\n\n1. 当你修改 `innerHTML` 时，浏览器并不会只更新你改动的那一小部分，浏览器必须销毁该元素下所有的旧 DOM 节点，重新解析 HTML 字符串，并创建新的 DOM 对象，大规模的 DOM 变更会触发昂贵的浏览器重排（Reflow）和重绘（Repaint），在循环中使用 `innerHTML += '...'` 更是性能自杀。\n2. 如果你直接把用户输入的数据赋给 `innerHTML`，就相当于给黑客开了后门，因为客户可以利用这种方式在你的网页上直接插入`javascript`标签来执行恶意JS代码。\n3. `innerHTML` 对格式要求极严。如果你不小心传入了未闭合的标签（比如 `\u003Cdiv>文字`），浏览器会尝试自动修复，但这往往会导致最终生成的 DOM 结构和你预期的完全不同，甚至引发布局崩坏。\n4. 如果你用 `innerHTML` 覆盖了一个容器，即便你只是改了一行文字，原本绑定在容器内部子元素上的所有 **Event Listeners**（事件监听器）都会随之消失（后续章节会介绍事件监听机制）\n\n因此，如果实在需要增加删除或是修改元素，我们更推荐使用后续章节中讲到的方法来操作，而不是直接修改HTML内容。如果仅仅只是修改页面上的文字内容，建议使用下面的`textContent`方式。\n\n除了`innerHTML`之外，还有`innerText`，它可以用来获取或设置 **“人眼能看到的文本”**。\n\n```html\n\u003Cdiv id=\"box\">Hello World\u003C\u002Fdiv>\n```\n\n或是这种嵌套写法，都只会解析可视文本部分：\n\n```html\n\u003Cdiv id=\"box\">\n  Hello \u003Cspan>World\u003C\u002Fspan>\n\u003C\u002Fdiv>\n```\n\n```js\nconsole.log(box.innerText)   \u002F\u002FHello World\n```\n\n注意，如果文本内容是不可见的，那么这里得到的结果也会是一个空字符串：\n\n```html\n\u003Cdiv id=\"box\" style=\"visibility: hidden\">Hello World\u003C\u002Fdiv>\n```\n\n最后，还有一个`textContent`，它的功能和`innerText`差不多，也可以用来获取或设置 **纯文本内容**，但是它不会受到可视性影响：\n\n```js\nconsole.log(box.textContent)   \u002F\u002F即使不可见依然可以得到正确结果\n```\n\n此外，针对于HTML中的换行也会一起保留下来，而`innerText`是渲染之后的实际展示文本，不会保留换行：\n\n```html\n  \u003Cdiv id=\"box\" style=\"visibility: hidden\">Hello\n    World\u003C\u002Fdiv>\n```\n\n![image-20260213172145204](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002FhOk8\u002Fimage-20260213172145204.png)\n\n相比`innerText`，它获取内容的性能更高，因为前者需要计算布局（Reflow）来确定文本是否可见。但是，一些特殊标签里面的内容也会被`textContent`拿到：\n\n```html\n\u003Cdiv id=\"box\">\n  \u003Cscript>\n      for (let i = 0; i \u003C 6; i++) {\n          console.log(\"孩子们，这并不好笑\")\n      }\n  \u003C\u002Fscript>\n\u003C\u002Fdiv>\n```\n\n![image-20260213172358089](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002FrdA6\u002Fimage-20260213172358089.png)\n\n而`innerText`则会绕过这些实际不可见的内容。无论使用`textContent`还是`innerText`，我们都可以对内容进行修改，但是注意，这里设置的是纯文本内容，也就是普通文本元素：\n\n```js\nbox.textContent = \"卡布奇诺今犹在，不见当年倒茶人\"\n```\n\n```js\nbox.innerText = \"卡布奇诺今犹在，不见当年倒茶人\"\n```\n\n![image-20260213172609962](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002F7Eei\u002Fimage-20260213172609962.png)\n\n注意，由于这里仅仅只是文本内容设置，所以即使我们设置一段HTML代码，也会被自动转义为文本形式的内容，变成普通文本元素：\n\n```js\nbox.innerText = \"\u003Cdiv>卡布奇诺今犹在，不见当年倒茶人\u003C\u002Fdiv>\"\n```\n\n![image-20260213173110970](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002Fr2kR\u002Fimage-20260213173110970.png)\n\n由于`textContent`不考虑实际渲染样式，所以在外面修改内容时浏览器不需要重新计算布局，在性能上相比`innerText`更加稳定，如果要频繁对页面内容进行修改，建议优先考虑`textContent`。\n\n除了使用`inner`获取内容外，我们还可以使用`outer`来获取包含元素本身在内的全部内容：\n\n```js\nconsole.log(box.outerHTML)\n```\n\n![image-20260215143630779](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F15\u002FAs5r\u002Fimage-20260215143630779.png)\n\n注意，当我们替换`outerHTML`内容时，会连带整个标签一起变化。\n\n还有`outerText`，但是这个属性默认情况下和`innerText`其实是一致的，不过，如果我们尝试修改它，它会连带着整个标签一起变成普通文本：\n\n```js\nbox.outerText = \"牛逼啊\"\n```\n\n![image-20260215145045049](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F15\u002F3nZr\u002Fimage-20260215145045049.png)\n\n### 创建和操作元素\n\n前面我们介绍了如何**修改内容**，只需要利用`innerHTML \u002F innerText \u002F textContent`属性即可。但是实际上，直接对`innerHTML`进行修改会导致一些性能问题，这一节我们就介绍一下更多关于 **DOM 的增删改操作**。\n\n想在页面中新增一个元素，**第一步一定是创建它**，`document`对象为我们提供了一个`createElement`方法可以快速创建一个新的元素：\n\n```js\nconst div = document.createElement('div')\ndiv.textContent = '我是新创建的元素'   \u002F\u002F和普通元素一样，可以正常设置属性\n```\n\n不过，这仅仅只是在内存中创建了一个新的元素，此时在页面上还看不到任何变化。创建完元素之后，**一定要插入到 DOM 树中**，我们可以使用`appendChild`方法来将它添加到某个元素的内部，作为子元素存在：\n\n```js\nconst box = document.getElementById('box')\nconst div = document.createElement('div')\ndiv.textContent = '我是新创建的元素'\n\u002F\u002F调用父元素的appendChild方法为其添加这个新创建的元素到内部\nbox.appendChild(div)\n```\n\n![image-20260213192117331](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002Fe4bT\u002Fimage-20260213192117331.png)\n\n注意，`appendChild`只会在原有基础上新增子元素，如果原本就存在子元素，则会自动添加到原本存在元素的后面：\n\n![image-20260213192700748](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002FrZ5b\u002Fimage-20260213192700748.png)\n\n还有一个比较有意思的是，如果这个元素不是我们创建的，而是原本在页面上就存在的，那么将其添加到新的位置后，相当于是从原来的位置移动到这个新的位置上，而不是复制一个新的元素插入：\n\n```js\nconst box = document.getElementById('box')\nconst item = document.querySelector('.item')\n\u002F\u002F实际上是将这个元素移动到box的下面\nbox.appendChild(item)\n```\n\n除了原本的`appendChild`，我们也可以使用`append`方法来实现元素插入，效果是完全一样的，但是它支持填入多个元素，按照顺序一次性插入：\n\n```js\nconst box = document.getElementById('box')\nconst div = document.createElement('div')\ndiv.textContent = '我是新创建的元素'\nconst item = document.querySelector('.item')\nbox.append(div, item)\n```\n\n此外，除了插入元素之外，`append`还可以直接插入一个字符串，这个字符串直接作为文本元素存入。\n\n```js\nconst box = document.getElementById('box')\nbox.append(\"我是文本内容\")\n```\n\n![image-20260213193511262](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002Fzy1V\u002Fimage-20260213193511262.png)\n\n这里除了创建简单元素之外，实际上`document`还为我们提供了很多不同的节点类型创建：\n\n```js\ndocument.createTextNode('hello')   \u002F\u002F可以实现创建一个文本节点\ndocument.createComment(\"这是注释\")   \u002F\u002F创建注释\n```\n\n这里需要注意的是，虽然`append`可以实现节点插入，但是依然会出现前面提到的性能问题，如果我们需要一次性插入多个节点，建议使用`DocumentFragment`，它就像是一个临时的小DOM树，也可以增删DOM元素，我们可以先把要插入的元素放入其中，之后一次性进行插入：\n\n```js\nconst fragment = document.createDocumentFragment();  \u002F\u002F创建一个临时dom树\n\u002F\u002F添加元素到fragment中\nbox.append(fragment)\n```\n\n这也可以极大地优化批量插入元素的情况，相比`for`循环每次插入都会导致元素重新计算发生重排，这种方式会大大减少重排次数，提高性能。\n\n当然，既然可以在尾部插入元素，我们也可以在首部插入元素，新元素会出现在 **第一个子节点的位置**：\n\n```js\nconst box = document.getElementById('box')\nbox.prepend(div, item, \"我是文本内容\")   \u002F\u002F效果和上面一样，但是在元素内部的前面插入\n```\n\n我们还可以实现精确位置插入，使用`insertBefore`可以实现在指定元素的前面插入，比如我们现在想要插入到这个`p`子标签的前面：\n\n```html\n\u003Cdiv id=\"box\">\n  \u003Ca>你干嘛\u003C\u002Fa>\n  \u003Cp>哎哟\u003C\u002Fp>\n\u003C\u002Fdiv>\n```\n\n```js\nconst box = document.getElementById('box')\n\nconst div = document.createElement('div')\ndiv.textContent = '我是新创建的元素'\n\u002F\u002F除了直接使用document来查询元素之外，我们还可以对着任意一个元素进行查询\nconst p = box.querySelector('p')\n\nbox.insertBefore(div, p)  \u002F\u002F这里的意思将div插入到p之前\n```\n\n不过，这种方式只能往某个元素的前面插入，用起来不是很方便。这里更建议大家使用更加万能的元素插入操作`insertAdjacentElement`，它的参数可以自由控制插入点，相比前面几种更加方便：\n\n```js\nconst box = document.getElementById('box')\nconst div = document.createElement('div')\ndiv.textContent = '我是新创建的元素'\nconst p = box.querySelector('p')\n\u002F\u002F第一个参数用于控制插入点,beforebegin 就是在它本身之前\nbox.insertAdjacentElement('beforebegin', div)\n```\n\n![image-20260214000419704](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F13\u002F9Zlt\u002Fimage-20260214000419704.png)\n\n| position        | 位置说明                     | 示意           |\n| --------------- | ---------------------------- | -------------- |\n| `\"beforebegin\"` | 在 target **前面**（当兄弟） | target 之前    |\n| `\"afterbegin\"`  | 在 target **内部最前**       | 第一个子元素   |\n| `\"beforeend\"`   | 在 target **内部最后**       | 最后一个子元素 |\n| `\"afterend\"`    | 在 target **后面**（当兄弟） | target 之后    |\n\n如果只是单纯插入一个文本元素，那么直接使用`insertAdjacentText`，它可以实现普通文本插入：\n\n```js\nbox.insertAdjacentText('beforebegin', \"我是文本内容\")\n```\n\n当然，如果你希望插入一段HTML代码，也可以使用`insertAdjacentText`：\n\n```js\nbox.insertAdjacentText('beforebegin', \"我是文本内容\")\n```\n\n除此之外，还有`insertAdjacentHTML`方法可以实现对HTML代码的插入，但是注意，这种方式和`innerHTML`一样，存在一些性能问题和XSS攻击问题。\n\n介绍完元素的插入，我们接着来看元素的删除，删除非常简单，只需要直接对着元素自己调用`remove`方法即可：\n\n```js\nconst box = document.getElementById('box')\nbox.remove()   \u002F\u002F移除自己\n```\n\n当然，我们也可以通过`removeChild`来移除子元素，但是这里需要传入子元素（那我干嘛不直接对着子元素`remove`呢）\n\n```js\nconst box = document.getElementById('box')\nconst p = box.querySelector('p')\nbox.removeChild(p)\n```\n\n然后是元素的替换操作，使用`replaceChild`即可替换某个指定元素：\n\n```js\nconst box = document.getElementById('box')\nconst p = box.querySelector('p')\nconst a = document.createElement('a')\na.textContent = '点我啊哥哥'\n\u002F\u002F第一个参数是新的元素，第二个参数是原本需要被替换的\nbox.replaceChild(a, p)\n```\n\n我们还可以使用`replaceChildren`来实现对所有子元素的替换：\n\n```js\nbox.replaceChildren(a, \"我是普通文本元素\")\n```\n\n`replaceChildren`会移除所有子元素，并将给到的元素替换进去。\n\n除了对子元素进行替换之外我们也可以直接对某个元素进行替换，使用`replaceWith`替换：\n\n```js\nconst box = document.getElementById('box')\nconst a = document.createElement('a')\na.textContent = '点我啊哥哥'\n\u002F\u002F直接替换调用元素\nbox.replaceWith(a)\n\u002F\u002F也可以一次性传入多个元素，包括文本元素也是可以的，注意不会进行HTML转换\nbox.replaceWith(a, \"我是普通文本元素\")\n```\n\n### 元素关系\n\n在真实开发中，我们**几乎不可能只操作一个元素**，更多情况是：批量修改列表样式，遍历表格、菜单、卡片，找某个元素的父 \u002F 子 \u002F 兄弟节点。实现这一切的关键就是在 DOM 树中，根据关系找到其他节点。\n\n常见的关系有 3 种：**父节点**、**子节点**、**兄弟节点**\n\n首先我们来看看如何通过元素找到自己的父元素，使用`parentElement`即可直接得到父元素：\n\n```js\nconst box = document.getElementById('box')\nconsole.log(box.parentElement)\n```\n\n![image-20260214124924131](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002FnpS7\u002Fimage-20260214124924131.png)\n\n由于这里返回的是`HTMLElement`对象，所以我们还可以继续使用`parentElement`向上找：\n\n```js\nconsole.log(box.parentElement)  \u002F\u002Fbody\nconsole.log(box.parentElement.parentElement)   \u002F\u002Fhtml\nconsole.log(box.parentElement.parentElement.parentElement)   \u002F\u002Fnull\n```\n\n当寻找到DOM的最顶层时，无法再继续获得父元素了。除了`parentElement`外，我们也可以使用`parentNode`来获取父节点，但是相比`parentElement`，它支持更多类型，不仅仅是Element对象，包括Document也是能拿到的：\n\n```js\nconsole.log(box.parentNode)   \u002F\u002Fbody\nconsole.log(box.parentNode.parentNode)   \u002F\u002Fhtml\nconsole.log(box.parentNode.parentNode.parentNode)  \u002F\u002Fdocument对象\n```\n\n我们接着来看子元素的获取，使用`childNodes`来得到所有子节点，注意这里得到的是Node，也就是说会包含多种类型的节点，包括文本和注释：\n\n```html\n\u003Cdiv id=\"box\">\n  \u003C!--  我是个很搞笑的注释  -->\n  \u003Ca>你干嘛\u003C\u002Fa>\n  \u003Cp>哎哟\u003C\u002Fp>\n\u003C\u002Fdiv>\n```\n\n```js\nconst box = document.getElementById('box')\nconsole.log(box.childNodes)\n```\n\n![image-20260214163140004](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002Ffp9W\u002Fimage-20260214163140004.png)\n\n可以看到这里得到了7个节点，不对啊，怎么会有7个呢？算上注释这里明明只有3个节点啊，这其实是因为空白部分也被算作文本节点导致的：\n\n![image-20260214163300393](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002F1osX\u002Fimage-20260214163300393.png)\n\n如果需要精确获取HTML元素，不包含无关的文本和注释，我们可以使用`children`属性：\n\n```js\nconsole.log(box.children)\nconsole.log(box.childElementCount)   \u002F\u002F使用childElementCount还能快速得到子元素数量\n```\n\n![image-20260214163352501](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002FXtz0\u002Fimage-20260214163352501.png)\n\n得到的结果是一个`HTMLCollection`，只包含HTML元素。有时候为了方便，我们还可以快速获取第一个或最后一个子元素：\n\n```js\nconsole.log(box.firstChild)   \u002F\u002F第一个子元素，拿到的是Node对象，包含文本和注释\nconsole.log(box.lastChild)    \u002F\u002F最后一个子元素，拿到的是Node对象，包含文本和注释\nconsole.log(box.firstElementChild)   \u002F\u002F第一个子元素，拿到的是Element对象\nconsole.log(box.lastElementChild)    \u002F\u002F最后一个子元素，拿到的是Element对象\n```\n\n除了查找子元素外，我们也可以实现对兄弟元素的查找，`Sibling`代表兄弟元素，和上面一样，我们也可以获取包含文本或注释的或是纯HTML元素的：\n\n```js\nconsole.log(box.previousSibling);  \u002F\u002F上一个兄弟元素，拿到的是Node对象，包含文本和注释\nconsole.log(box.nextSibling);  \u002F\u002F下一个兄弟元素，拿到的是Node对象，包含文本和注释\nconsole.log(box.previousElementSibling);  \u002F\u002F上一个兄弟元素，拿到的是Element对象\nconsole.log(box.nextElementSibling);   \u002F\u002F上一个兄弟元素，拿到的是Element对象\n```\n\n综上，建议各位小伙伴尽量使用返回结果是`Element`的属性，实际开发中它们更加常用。\n\n### 元素位置和滚动\n\n在真实开发中，我们经常会遇到这些需求：判断一个元素**在页面的什么位置**、获取元素**距离窗口顶部\u002F左侧的距离**、判断元素是否**出现在可视区域**等等，这些能力，都和 **元素位置（Position）** 与 **滚动（Scroll）** 有关。\n\n`offset` 系列属性，是 **最容易理解、也是新手最常用的一组**，它反映的是元素在**页面布局中的位置和尺寸**，常用的有 4 个：\n\n```js\nelement.offsetWidth   \u002F\u002F元素宽度\nelement.offsetHeight   \u002F\u002F元素高度\n```\n\n这里的元素高度是按照`border-box`进行计算的，也是就包含了边框、内边距、内容在内的总宽度，不包含外边距，它可以反应盒子本身占据的空间。\n\n```js\nelement.offsetLeft   \u002F\u002F元素x坐标\nelement.offsetTop    \u002F\u002F元素y坐标\n```\n\n`left`和`top`坐标表示**元素左上角**相对于 **offsetParent** 的距离。`offsetParent` 指的是离当前元素最近的、设置了定位（position 不为 static）的祖先元素，如果一路往上都没找到，那么默认是 `body`（其实就是CSS中介绍的定位方式）我们可以直接打印`offsetParent`查看这个定位父元素是谁：\n\n```js\nconsole.log(box.offsetParent)   \u002F\u002Fbody\n```\n\n我们也可以稍加修改：\n\n```html\n\u003Cdiv style=\"position: relative\">\n  \u003Cdiv id=\"box\">我是文本\u003C\u002Fdiv>\n\u003C\u002Fdiv>\n```\n\n![image-20260215022931547](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002F5Otg\u002Fimage-20260215022931547.png)\n\n注意，以上提到的属性都是由CSS进行控制的，为只读属性，无法手动进行修改。\n\n接着是`client` 系列，它们描述的是 **元素内部可视区域**，和滚动条强相关。\n\n```js\nelement.clientWidth  \u002F\u002F元素宽度\nelement.clientHeight  \u002F\u002F元素高度\n```\n\n这里的元素高度是按照`padding-box`进行计算的，也就是只包含了内边距、内容在内的总宽度，不包含边框和外边距，包括滚动条占据的宽度也不包含。\n\n```html\n\u003Cstyle>\n  #box {\n    overflow: auto;\n    height: 100px;\n    width: 200px;\n    border: 2px solid black;\n  }\n\n  .inner-box {\n    background: #f0f0f0;\n    height: 2000px;\n  }\n\u003C\u002Fstyle>\n```\n\n```html\n\u003Cdiv id=\"box\">\n  \u003Cdiv class=\"inner-box\">\u003C\u002Fdiv>\n\u003C\u002Fdiv>\n```\n\n```js\nconsole.log(box.offsetWidth)   \u002F\u002F204，整个盒子大小\nconsole.log(box.clientWidth)   \u002F\u002F185，实际内容区域大小\n```\n\n同样的，clientLeft \u002F clientTop可以表示`padding-box`的坐标，但是注意这个坐标是相对的盒子的边框的距离，一般情况下就是盒子边框的大小：\n\n```js\nconsole.log(box.clientLeft)   \u002F\u002F2，距离左边框大小\nconsole.log(box.clientTop)   \u002F\u002F2，距离上边框大小\n```\n\n这个属性一般很少单独用，知道它是 **border 的厚度** 就够了，同样的，这里提到的4个属性无法被手动修改，由CSS控制的只读属性。\n\n我们接着来介绍滚动状态下，内容的这四个属性又会如何变化，首先是`scrollWidth \u002F scrollHeight`属性，它代表实际的滚动区域宽度和高度：\n\n```js\nconsole.log(box.scrollWidth)  \u002F\u002F实际的滚动区域宽度，也就是总的滚动宽度\nconsole.log(box.scrollHeight)   \u002F\u002F实际滚动区域高度，也就是总的滚动高度\n```\n\n而`scrollLeft \u002F scrollTop`表示的则是当前滚动的左边位置和顶部位置：\n\n```js\nconsole.log(box.scrollTop)   \u002F\u002F内容滚动顶部位置\nconsole.log(box.scrollLeft)   \u002F\u002F内容滚动左侧位置\n```\n\n![image-20260215025953866](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002FBdr3\u002Fimage-20260215025953866.png)\n\n和前面不同，针对于`scrollLeft \u002F scrollTop`属性，我们可以手动为其设置一个值，使得滚动位置可以动态调整：\n\n![image-20260215030330493](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F14\u002FHwo7\u002Fimage-20260215030330493.png)\n\n手动赋值后，滚动条会自动跳到我们指定的位置上。不过，这里更推荐大家使用专门的滚动控制方法`scrollTo`或是`scroll`，这两个方法效果完全一样，只是名字不同，文档更推荐使用`scroll`方法：\n\n```js\n\u002F\u002F使用数字直接控制在x轴和y轴上滚动到哪个位置，px为单位\nbox.scroll(0, 0)\n\u002F\u002F这里需要传入一个对象，其中top和left属性控制滚动目标位置（可以缺一）\n\u002F\u002Fbehavior控制滚动行为，目前只有立即滚动和平滑滚动\nbox.scroll({ top: 0, behavior: 'smooth' })\n```\n\n如果希望滚动过程更加流畅丝滑，建议使用平滑滚动效果，不同浏览器会有不同的展现方式，但是都大差不差。\n\n出了让滚动条直接跳到指定位置，我们也可以控制滚动条相对滚动，使用`scrollBy`来实现：\n\n```js\nbox.scrollBy(0, 1000)\n```\n\n这里会使得滚动条向`y`轴方向向下滚动1000个像素点。\n\n此外，还有一个滚动方法，`scrollIntoView`可以实现滚动到指定元素位置，使得指定元素在可视区域中：\n\n```html\n\u003Cdiv style=\"height: 2000px\">我是页面上的一段普通内容\u003C\u002Fdiv>\n\u003Cdiv id=\"box\">我是内容\u003C\u002Fdiv>\n```\n\n```js\nconst box = document.getElementById('box')\nbox.scrollIntoView()\n```\n\n此外，除了获取元素在页面上的绝对位置之外，我们也可以获取元素位于视口上的信息，它会按照元素实际距离视口位置进行取值，当滚动位置发生变化的时候，元素视口位置也会跟着发生变化：\n\n```js\nconst box = document.getElementById('box')\nconsole.log(box.getBoundingClientRect())\n```\n\n![image-20260215113852456](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F15\u002Ft4jW\u002Fimage-20260215113852456.png)\n\n这里的`top`指的是元素顶部和视口顶部距离，当元素滚动到视口顶部时，`top`值也会变成0，而如果已经滚到上面去了，那么`top`会变成负数，`left`也是同理。`y`和`x`值和这里的`top`和`left`是一样的，不多做介绍了。\n\n除了上面和左边，`bottom`也可以为我们提供元素底部与视口底部的距离（等价于 `top + height`），`right`同理，和`left`是相对应的。\n\n这里的`width`和`height`默认情况下和之前的`offsetWidth`一样，表示元素的总宽度（包括 padding、border，不包括 margin）但是，如果元素应用了 CSS 变换（如 `scale`、`rotate`），返回的尺寸和位置也会是变换后的最终结果：\n\n```html\n\u003Cdiv id=\"box\" style=\"width:fit-content;scale: 1.5\">我是内容\u003C\u002Fdiv>\n```\n\n```js\nconst box = document.getElementById('box')\nconsole.log(box.offsetWidth)   \u002F\u002F元素变换前本身占据空间的尺寸 64px\nconsole.log(box.getBoundingClientRect())   \u002F\u002Fwidth = 96px\n```\n\n可以看到，这里发生了CSS变换，虽然元素本身占据的位置大小不变，但是实际展示的大小不一样。这对于需要获取元素实际展示尺寸的情况，就非常好用。\n\n## 事件\n\n在讲 DOM 事件之前，我们先回顾一下 **DOM（Document Object Model）** 是什么。当浏览器解析 HTML 时，会把页面结构转换成一棵“节点树”，这棵树就是 DOM。而 **DOM 事件（DOM Event）**，就是当用户或浏览器对页面做出某种行为时，浏览器通知 JavaScript 的一种机制。\n\n事件就是页面上发生的动作，例如：用户点击按钮、鼠标移动、键盘按下、页面加载完成、表单提交。\n\n### 事件处理\n\n理解事件时，通常可以拆解为三个部分：\n\n- **事件源（Event Target）：** 谁触发了事件？（例如：一个 `\u003Cbutton>` 按钮）\n- **事件类型（Event Type）：** 发生了什么？（例如：`click` 点击、`keydown` 按键）\n- **事件处理程序（Event Handler）：** 发生了之后要做什么？（通常是一个 JavaScript 函数）\n\n比如，我们现在需要处理按钮的点击事件，那么事件源就是一个按钮元素，事件类型就是点击事件，在明确事件源和事件类型后，我们就可以针对这个事件进行处理，当发生这个事件时，就执行我们自定义的JS代码。\n\n处理事件一共有三种方式，我们先来看最简单的一种，我们可以直接在标签上进行编写：\n\n```html\n\u003Cbutton onclick=\"alert('你点击了按钮')\">我是一个大按钮\u003C\u002Fbutton>\n```\n\n这种方式叫做 **内联事件**，一般来说，事件属性名称都是以`on`开头的，比如这里的点击事件，就是`onclick`，当用户点击元素时就会触发这个事件，浏览器为我们准备了多种多样的事件，以便我们可以在各种情况下都能处理用户的交互行为。\n\n在`onclick`属性的内部，需要填写我们想要执行的JS代码，和之前一样，不能出现语法错误，同时，由于直接在行内编写，出现多行JS代码时，需要使用`;`隔开：\n\n```html\n\u003Cbutton onclick=\"console.log('我是测试');alert('你点击了按钮')\">我是一个大按钮\u003C\u002Fbutton>\n```\n\n虽然这种方式写起来非常直观，但是只适合单行JS代码编写，如果有很多行写起来会非常麻烦，建议如果存在多行的情况下，最好封装成函数的形式使用：\n\n```js\nfunction test() {\n    console.log('我是测试')\n    alert('你点击了按钮')\n}\n```\n\n```html\n\u003Cbutton onclick=\"test()\">我是一个大按钮\u003C\u002Fbutton>\n```\n\n除此之外，我们也可以使用DOM属性来进行事件处理：\n\n```js\nconst btn = document.getElementById('test');\n\u002F\u002F直接为元素的onclick属性赋值，即可将函数绑定到此事件上\nbtn.onclick = () => {\n  console.log('我是测试');\n  alert('你点击了按钮');\n}\n```\n\n这种方式和前面的写法效果完全一样，并且我们可以在赋值（绑定）的函数中自由编写代码。\n\n不过，虽然这种方式可以更方便的绑定事件处理逻辑，但是这个事件的处理逻辑只能存在一个，重复赋值只会覆盖原本的事件处理逻辑，如果我们需要为某个时间添加多个处理操作，就需要用到更加高级的事件监听器了：\n\n```js\n\u002F\u002F添加事件监听器来实现事件处理\nbtn.addEventListener('click', () => {\n  console.log('我是测试');\n  alert('你点击了按钮');\n})\n```\n\n我们可以为指定事件添加事件监听器，当触发事件时，事件监听器中编写的函数也会自动执行，这是现代开发的标准方式。同时，事件监听器可以无限制添加，当触发时，会按照添加的顺序依次执行：\n\n```js\nbtn.addEventListener('click', () => {\n  console.log('我是测试1');\n})\nbtn.addEventListener('click', () => {\n    console.log('我是测试2');\n})\nbtn.addEventListener('click', () => {\n    console.log('我是测试3');\n})\n```\n\n![image-20260224164303807](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F24\u002F7Bgn\u002Fimage-20260224164303807.png)\n\n同样的，如果我们不需要某个事件监听器了，也可以移除它：\n\n```js\nconst handler = () => {\n    console.log(\"我是测试\")\n}\n\nbtn.addEventListener('click', handler)\nbtn.removeEventListener('click', handler)   \u002F\u002F注意移除的必须是同一个函数对象\n```\n\n有些时候，我们可能只希望事件监听器只执行一次，也可以进行参数配置：\n\n```js\nbtn.addEventListener('click', handler, { once: true })  \n\u002F\u002F最后一次参数是监听器配置，once表示只监听一次\n```\n\n以上三种事件处理方式都可以实现事件处理，各位小伙伴可以根据自己的实际需求进行选择。\n\n需要注意的是，在事件监听中，非箭头函数的`this`的指向并不是全局对象（箭头函数依然是按照之前的规则绑定`this`）而是当前元素本身：\n\n```js\nfunction handler() {\n    console.log(this)   \u002F\u002F在调用时，由于是在事件中执行，this指向的就是事件源\n}\nbtn.addEventListener('click', handler)\n```\n\n所以在事件里，如果你需要用到当前元素对象，可以直接使用`this`，但是注意箭头函数形式是无效的。\n\n除了简单监听事件之外，我们还可以使用`event`对象，每次事件触发，浏览器都会自动传入一个事件对象，除了在标签属性上无法使用外，其他地方都可以直接作为参数使用：\n\n```js\n\u002F\u002F直接修改onclick可以使用\nbtn.onclick = event => console.log(event)\n\n\u002F\u002F事件监听器可以使用\nbtn.addEventListener('click', (event) => {\n    console.log(event)\n})\n```\n\n这个 `event` 包含了本次事件的多种信息，比如触发的元素、鼠标位置、按键信息、是否按下`ctrl`、事件阶段、是否可取消等。比如点击事件，它包含了：\n\n```js\nevent.target        \u002F\u002F 真正触发事件的元素\nevent.currentTarget \u002F\u002F 当前绑定事件的元素（至于为什么要分两个，下面接着介绍）\nevent.type          \u002F\u002F 事件类型\nevent.clientX       \u002F\u002F 鼠标X坐标\nevent.clientY       \u002F\u002F 鼠标Y坐标\n```\n\n有关不同事件对象的详细信息，我们会在后续课程中逐步介绍。\n\n### 事件流\n\n事件流是事件章节最重要的核心知识，实际上浏览器在处理事件时，并不是简单的“点谁就执行谁”。想象一下，你点击了页面上的一个按钮，你不仅点击了按钮本身，同时也点击了它的父容器、甚至整个页面。\n\n为了解决“到底谁先响应”的问题，浏览器采用了三阶段的流动模型。根据 W3C 标准，一个事件的生命周期按顺序分为以下三个阶段：\n\n1. 捕获阶段 (Capture Phase)\n\n   事件从最顶层的窗体对象（`Window`）开始，逐级向下传播，直到到达触发事件的具体元素。\n\n   * **目的：** 在事件到达目标之前拦截它。\n   * **顺序：** `Window` -> `Document` -> `\u003Chtml>` -> `\u003Cbody>` -> ... -> 目标元素的父级。\n\n2. 目标阶段 (Target Phase)\n\n   事件终于到达了真正触发它的那个元素（即 `event.target`）。\n\n   - 在这个阶段，事件被触发并执行绑定的监听函数。\n\n3. 冒泡阶段 (Bubbling Phase)\n\n   事件从目标元素开始，逐级向上传播，直到回到最顶层的 `Window` 对象。\n\n   - **意义：** 这是最常用的阶段，也是 JavaScript 默认监听的阶段。\n   - **顺序：** 目标元素 -> 父级 -> ... -> `\u003Cbody>` -> `\u003Chtml>` -> `Document` -> `Window`。\n\n比如下面的结构：\n\n```html\n\u003Cbody>\n  \u003Cdiv>我是页面上的一段普通内容\u003C\u002Fdiv>\n  \u003Cdiv>\n    \u003Cbutton>我是一个大按钮\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Fbody>\n```\n\n![image-20260224174319642](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F24\u002F3Cei\u002Fimage-20260224174319642.png)\n\n事件会从整个文档DOM树的最底层逐步传播到目标元素上，当找到目标元素时，开始按照冒泡顺序依次触发事件：\n\n```html\n\u003Cbody onclick=\"console.log('3')\">\n  \u003Cdiv>我是页面上的一段普通内容\u003C\u002Fdiv>\n  \u003Cdiv onclick=\"console.log('2')\">\n    \u003Cbutton onclick=\"console.log('1')\">我是一个大按钮\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Fbody>\n```\n\n![image-20260224175023351](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F24\u002Fw5Ab\u002Fimage-20260224175023351.png)\n\n可以看到，最内层点击的目标元素就是冒泡的最底部，从这个元素开始，逐步向外层元素触发`onclick`事件，所以事件触发并非是点击某个元素本身触发，包含此元素的盒子、页面都会一起触发点击事件，虽然从层级关系来说最上面展示的应该是按钮，但是点击按钮的同时也点击了压在下面的外层盒子。\n\n这种机制很好地实现了想要一次性监听外层盒子的事件触发，而非局限于某个内部元素的效果。\n\n理解了事件流，就能掌握前端开发中最重要的性能优化手段——**事件委托**，假如一个父容器（比如 `\u003Cul>`）下面有 1000 个子元素（`\u003Cli>`），你不需要给每个 `\u003Cli>` 都绑定点击事件。相反，你可以利用 **冒泡机制**，只在 `\u003Cul>` 上绑定一个监听器：\n\n```html\n\u003Cul>\n  \u003Cli>我是列表项1\u003C\u002Fli>\n  \u003Cli>我是列表项2\u003C\u002Fli>\n  \u003Cli>我是列表项3\u003C\u002Fli>\n  \u003Cli>我是列表项4\u003C\u002Fli>\n\u003C\u002Ful>\n```\n\n```js\ndocument.querySelector('ul').addEventListener('click', evt => {\n    console.log(evt.target)  \u002F\u002F直接通过target获取被点的li标签\n  \tconsole.log(evt.currentTarget)  \u002F\u002F获取当前事件触发元素ul\n})\n```\n\n当点击`ul`范围内的任何一个元素时，都会通过冒泡机制触发`ul`的点击事件，我们只需要通过`target`属性即可拿到具体点击的是哪一个元素了。\n\n除了正常执行事件流之外，我们也可以通过 `addEventListener` 方法来决定在哪一个阶段“拦截”事件：\n\n* **`capture` 为 `false`（默认值）：** 监听函数在 **冒泡阶段** 执行。\n* **`capture` 为 `true`：** 监听函数在 **捕获阶段** 执行。\n\n```html\n\u003Cbody onclick=\"console.log('3')\">\n  \u003Cdiv>我是页面上的一段普通内容\u003C\u002Fdiv>\n  \u003Cdiv id=\"test\">\n    \u003Cbutton onclick=\"console.log('1')\">我是一个大按钮\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Fbody>\n```\n\n```js\nbtn.addEventListener('click', (event) => {\n    console.log(\"2\")\n}, {   \u002F\u002F第二个参数填写监听选项\n    capture: true\n})\n```\n\n![image-20260224184509168](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F24\u002F6Yxi\u002Fimage-20260224184509168.png)\n\n可以看到，当`capture`开启之后，事件的处理会提前到捕获阶段执行。\n\n有时候，我们可能希望点击子元素后不触发父元素的任何逻辑，这时你需要手动“截断”流，也就是阻止事件继续向后传播或是冒泡。我们可以使用`stopPropagation`方法：\n\n```js\nbtn.addEventListener('click', (event) => {\n    event.stopPropagation()   \u002F\u002F停止事件继续传播\n    console.log(\"2\")\n})\n```\n\n![image-20260224190037562](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F24\u002FM1sp\u002Fimage-20260224190037562.png)\n\n可以看到，事件没有继续向上进行冒泡，父元素的事件处理不再执行，不过，由于阻止的是事件冒泡，所以如果我们将事件处理时机提前到`capture`，那么事件依然可以被处理。\n\n需要注意的是，这里仅仅是阻止继续向下传播，但是当前元素绑定的其他同类型监听器仍会执行。如果需要彻底断绝后续所有监听器的事件处理，我们可以使用`stopImmediatePropagation`方法，它会立即结束所有事件冒泡并阻断后续监听器执行。\n\n### 事件默认行为\n\n这一节我们接着介绍事件相关的控制操作。首先是事件默认行为，在默认情况下，随着一些事件的触发，某些特殊的标签会产生一些默认行为，比如`a`标签在点击时会发生跳转：\n\n```html\n\u003Ca href=\"https:\u002F\u002Fwww.baidu.com\" target=\"_blank\" id=\"test\">我是一个大按钮\u003C\u002Fa>\n```\n\n我们也可以通过事件监听来阻止其默认行为：\n\n```js\na.addEventListener('click', event => {\n    console.log(\"1\")\n    event.preventDefault()   \u002F\u002F阻止事件默认行为\n})\n```\n\n此时再次点击链接就无法进行跳转了。利用这种机制，我们还可以让用户在右键页面时无法弹出菜单，防止偷代码：\n\n```js\n\u002F\u002Fcontextmenu是右键点击弹出菜单事件\ndocument.documentElement.addEventListener('contextmenu', event => {\n    event.preventDefault()\n})\n```\n\n需要注意的是，并非所有事件都可以被阻止默认行为，我们可以通过`cancelable`属性来进行判断：\n\n```js\nconsole.log(event.cancelable)\n```\n\n它表示当前这个事件，是否“允许被取消默认行为”。常见可取消事件有：`click`、`submit`、`keydown`、`beforeunload`、`touchstart`、`wheel`。\n\n在创建监听器时有一个非常有意思的参数，`passive`表示我们在Listener中永远不会调用`preventDefault`方法，但是如果我们又使用了这个方法的话，浏览器只会打印一个错误信息，而不会真的去执行`preventDefault`方法：\n\n```js\na.addEventListener('click', event => {\n    console.log(\"1\")\n    event.preventDefault()\n}, {\n    passive: true\n})\n```\n\n![image-20260224223934192](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F24\u002FoE1f\u002Fimage-20260224223934192.png)\n\n可以看到`preventDefault`并没有阻止事件行为触发，而是产生了一个错误信息到控制台（注意这里不是抛错误，不会导致后续代码执行失败）某些事件在一些浏览器上会默认将`passive`设置为`true`来保证一些性能上的优化。\n\n### 鼠标事件\n\n鼠标事件是前端开发中**最常用的一类事件**，几乎所有交互都离不开它。\n\n常见鼠标事件如下：\n\n| 事件名        | 说明               |\n| ------------- | ------------------ |\n| `click`       | 单击               |\n| `dblclick`    | 双击               |\n| `mousedown`   | 鼠标按下           |\n| `mouseup`     | 鼠标抬起           |\n| `mousemove`   | 鼠标移动           |\n| `mouseenter`  | 鼠标进入（不冒泡） |\n| `mouseleave`  | 鼠标离开（不冒泡） |\n| `mouseover`   | 鼠标进入（会冒泡） |\n| `mouseout`    | 鼠标离开（会冒泡） |\n| `contextmenu` | 右键菜单           |\n\n首先介绍一下`click`点击事件，最常见的点击事件：\n\n```js\nbtn.addEventListener('click', function () {\n  console.log('点击了')\n})\n```\n\n注意，一次`click` = 按下 + 抬起 都发生在同一元素，如果按下在 A，抬起在 B，不算`click`。如果需要精细化控制，可以考虑`mousedown \u002F mouseup`事件，它可以实现更底层的监听，分别针对鼠标按下和释放：\n\n```js\nbtn.addEventListener('mousedown', () => {\n  console.log('按下')\n})\n\nbtn.addEventListener('mouseup', () => {\n  console.log('抬起')\n})\n```\n\n利用这种监听，就可以实现例如拖拽功能、长按检测、自定义按钮效果等操作。\n\n`mousemove`事件当鼠标在元素上移动时会持续触发：\n\n```js\nbtn.addEventListener('mousemove', (e) => {\n  console.log(e.clientX, e.clientY)\n})\n```\n\n然后是`mouseenter\u002Fmouseleave`，当鼠标进入\u002F离开元素区域时，会触发一次：\n\n```js\nbtn.addEventListener('mouseenter', () => {\n  console.log('mouseenter')\n})\n```\n\n与其类似的还有一个`mouseover\u002Fmouseout`事件，它们也可以在鼠标进入\u002F离开元素区域时，触发一次，但是区别是，上面的`mouseenter\u002Fmouseleave`不会出现冒泡，而下面的会出现冒泡的现象。同时，`mouseover\u002Fmouseout`在进入子元素时也会触发，所以，实际开发中更推荐使用 `mouseenter`，它能更加准确反应鼠标进入元素。\n\n在以上提到的所有鼠标事件中，常用的属性如下：\n\n```js\n  console.log(e.clientX) \u002F\u002F 相对视口X坐标\n  console.log(e.clientY)\n  console.log(e.pageX)   \u002F\u002F 相对页面X坐标\n  console.log(e.pageY)\n  console.log(e.button)  \u002F\u002F 按下的是哪个鼠标键\n```\n\n我们的鼠标一共有三个按键，包含左中右：\n\n| 值   | 按键 |\n| ---- | ---- |\n| 0    | 左键 |\n| 1    | 中键 |\n| 2    | 右键 |\n\n### 键盘事件\n\n键盘事件主要用于：表单输入、游戏控制和快捷键等功能。常见键盘事件包含：\n\n| 事件名     | 说明                                       |\n| ---------- | ------------------------------------------ |\n| `keydown`  | 键按下                                     |\n| `keyup`    | 键抬起                                     |\n| `keypress` | 也是键按下，但不包含Ctrl这类控制键，不推荐 |\n\n比如我们想要检测某个键是否被按下：\n\n```js\n\u002F\u002F检测按键按下一般可以直接对着文档对象添加事件监听\ndocument.addEventListener('keydown', e => {\n  console.log(e.key)   \u002F\u002F通过key属性来获取被按下的键\n})\n```\n\n需要注意的是，如果我们对着某个子元素添加按键监听，那么是不会有效果的：\n\n```html\n\u003Ca href=\"https:\u002F\u002Fwww.baidu.com\" target=\"_blank\" id=\"test\">我是一个\u003Cp>大按钮\u003C\u002Fp>\u003C\u002Fa>\n```\n\n```js\na.addEventListener('keydown', function (e) {\n    console.log(e.key)\n})\n```\n\n这是因为，只有能成为**焦点元素 (focusable element)** 的 DOM 节点，才能响应键盘事件，比如，`\u003Cinput>`\u002F`\u003Cbutton>`\u002F`\u003Ca>`\u002F`\u003Cselect>` 等表单 \u002F 交互元素，就可以成为焦点元素。像 `\u003Cdiv>\u002F\u003Cspan>\u002F\u003Cp>\u002F\u003Cli>` 这类普通块级 \u002F 行内子元素，则无法成为焦点元素。\n\n我们可以试试看`input`标签：\n\n```html\n\u003Cinput id=\"test\">\n```\n\n![image-20260225101019073](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F25\u002FKp7k\u002Fimage-20260225101019073.png)\n\n当我们处于输入状态时，`input`标签就是聚焦状态，此时可以正确响应键盘事件。当然，除了通过上面的方式去聚焦之外，我们也可以使用JS代码来主动聚焦：\n\n```js\nconst a = document.getElementById('test')\na.focus()\n```\n\n或是使用Tab键进行选择：\n\n![image-20260225101956304](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F25\u002FeJ7r\u002Fimage-20260225101956304.png)\n\n当然，如果是Tab键无法选中的元素，比如`div`这类，我们可以给需要监听按键的子元素添加一个核心属性，使其具备焦点能力：\n\n```html\n\u003Cdiv class=\"key-listener\" tabindex=\"0\">点击我后按任意键试试\u003C\u002Fdiv>\n```\n\n```js\ndocument.querySelector('.key-listener').addEventListener('keydown', e => {\n    console.log(e.key)\n})\n```\n\n这里用到了`tabindex`属性，它能让普通元素具备焦点能力，现在使用Tab就可以快速聚焦了（`tabindex=\"-1\"` 表示可通过 JS 手动聚焦，但不能通过 Tab 键选中，如果是正整数越大表示优先级越高）\n\n接着是一些常用的键盘事件的属性：\n\n```js\ne.key        \u002F\u002F 按下的字符\ne.code       \u002F\u002F 键盘物理位置\ne.ctrlKey    \u002F\u002F 是否按住ctrl\ne.shiftKey\ne.altKey\ne.metaKey   \u002F\u002FMac键位专用\n```\n\n利用这些属性，我们可以通过判断来实现组合按键监听效果：\n\n```js\ndocument.addEventListener('keydown', e => {\n  \t\u002F\u002F判断是否按下ctrl键（MacOS下command就是meta键）\n    if((e.metaKey || e.ctrlKey) && e.key === 'c') {\n        alert('你按下了Ctrl + C')\n    }\n})\n```\n\n结合前面介绍的阻止默认行为，我们还可以防止用户复制网页内容：\n\n```js\ndocument.addEventListener('keydown', e => {\n    if((e.metaKey || e.ctrlKey) && e.key === 'c') {\n        console.log(\"用户复制内容被阻止\")\n        e.preventDefault()\n    }\n})\n```\n\n### 表单事件\n\n在实际开发中，**表单是最常见的交互区域**，一个表单中包含多种输入框和按钮。登录、注册、搜索、留言、提交订单……几乎所有业务都离不开表单。常见表单事件如下：\n\n| 事件名   | 说明           |\n| -------- | -------------- |\n| `submit` | 表单提交       |\n| `input`  | 输入时触发     |\n| `change` | 失去焦点后触发 |\n| `focus`  | 获得焦点       |\n| `blur`   | 失去焦点       |\n| `reset`  | 表单重置       |\n\n首先是针对于表单中的输入框，有一个`input`事件，当输入框内容发生改变时立即触发（实时触发）\n\n```html\n\u003Cform id=\"loginForm\">\n  \u003Clabel>\n    \u003Cinput type=\"text\" id=\"username\" placeholder=\"请输入用户名\">\n  \u003C\u002Flabel>\n  \u003Clabel>\n    \u003Cinput type=\"password\" id=\"password\" placeholder=\"请输入密码\">\n  \u003C\u002Flabel>\n  \u003Cbutton>登录\u003C\u002Fbutton>\n\u003C\u002Fform>\n```\n\n```js\nconst input = document.getElementById('username')\n\n\u002F\u002F监听input事件\ninput.addEventListener('input', () => {\n    console.log('当前输入内容：', input.value)   \u002F\u002F使用value属性来获取输入框中输入的内容\n})\n```\n\n每输入一个字符都会触发`input`事件，包括删除、粘贴等，像实时校验用户输入内容是否合法就可以采用这种方式，比如判断用户输入的用户名是否是邮箱格式：\n\n```js\ninput.addEventListener('input', () => {\n  \t\u002F\u002F使用正则表达式进行判断\n    const emailRegex = \u002F^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$\u002F\n    const isEmail = emailRegex.test(input.value)\n    console.log('是否是邮箱:', isEmail)\n})\n```\n\n这就是非常典型的一种表单校验案例。当然，还有一个`change`事件和 `input` 很像，但有一个关键区别，就是只有在内容变化且失去焦点后才触发。\n\n针对于使用了中文输入法的情况，有时候非常恶心：\n\n![image-20260225160152582](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F25\u002FuSe7\u002Fimage-20260225160152582.png)\n\n由于中文输入法也是先输入每个字符再完成拼写，所以就会出现这种输入法过程中的文本输入也被监听到的情况，因此我们还可以更细致地进行事件监听：\n\n1. compositionstart  -  输入法开始\n2. compositionupdate  -  输入法正在打字\n3. compositionend  -  输入法结束\n\n```js\n\u002F\u002F只监听输入法结束时的事件\ninput.addEventListener('compositionend', () => {\n    console.log(input.value)   \u002F\u002F得到的就是输入法完成的结果\n})\n```\n\n利用这种机制，我们就可以进行更加精确判断了：\n\n```js\nlet composition = false\n\nconst showText = () => {\n    if(!composition) {\n        console.log(input.value)\n    }\n}\n\ninput.addEventListener('compositionstart', () => composition = true)\ninput.addEventListener('compositionend', () => {\n    composition = false\n    showText()   \u002F\u002F因为compositionend不会一起触发input事件，所以完成输入法之后也得手动查看一次\n})\ninput.addEventListener('input', showText)\n```\n\n除了普通的文本输入框之外，选择框也可以在选择之后触发`input`事件：\n\n```html\n\u003Clabel>\n  \u003Cselect id=\"gender\">\n    \u003Coption>男\u003C\u002Foption>\n    \u003Coption>女\u003C\u002Foption>\n  \u003C\u002Fselect>\n\u003C\u002Flabel>\n```\n\n```js\nconst select = document.getElementById('gender')\nselect.addEventListener('input', evt => {\n    console.log(select.value)\n})\n```\n\n接着是焦点获取相关的事件，在之前我们提到，部分元素是可以获得焦点的，键盘事件只有在取得焦点时才会正确响应，当元素聚焦或失焦时，也可以通过对应的事件来进行处理，`focus` 和 `blur` 就是聚焦和失焦时的事件：\n\n```js\ninput.addEventListener('focus', function () {\n  console.log('获得焦点')\n})\n\ninput.addEventListener('blur', function () {\n  console.log('失去焦点')\n})\n```\n\n同样的，利用这种机制我们也可以在用户输入完成之后进行表单校验。\n\n接着是非常重要的表单提交事件，在HTML阶段我们学习过，当点击表单内的按钮时，会自动提交表单内容，发起一个表单请求，默认为GET请求方式，对应的事件就是表单提交事件：\n\n```js\nconst form = document.getElementById('loginForm')\n\nform.addEventListener('submit', function (e) {\n  console.log('表单提交了')\n})\n```\n\n在默认情况下，表单提交会刷新页面并发送请求，我们也可以阻止其默认行为来防止页面刷新，并通过自行发起请求的方式来进行登录，有关发起XHR请求相关内容我们会在前端路线后续课程中介绍。\n\n### 编辑事件\n\n编辑事件主要用于可编辑内容区域，比如现在输入框中有一段文本：\n\n```html\n\u003Ctextarea style=\"width: 300px;height: 200px\" id=\"text\">\u003C\u002Ftextarea>\n```\n\n```js\nconst input = document.getElementById('text')\ninput.value = `1、祝有爱者有爱，无爱者自由。\n2、不要靠近我，熟悉我，关心我，然后离开我。\n3、人在刚认识的时候最好，虚伪且礼貌。\n4、有空多爱自己，别人很忙。\n5、世界上最危险的东西就是希望。\n6、或许换个时间，有些人真的很合适。\n7、我渴望你是救赎，也恐惧你是深渊。\n8、想离开的人从来都不缺理由。\n9、我对你的热情，在无数个夜晚里消耗殆尽。\n10、也许最稳定的关系，就是没有关系。`\ninput.addEventListener('select', evt => {\n    console.log(evt)\n})\n```\n\n当我们选中其中一段文本之后，就会触发`select`事件：\n\n![image-20260225172405128](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F25\u002FxHa6\u002Fimage-20260225172405128.png)\n\n我们通过文本框的`selectionStart`和`selectionEnd`属性即可获得选择的起始和结束位置（就是选择的文本内容子串起始和结束下标）\n\n```js\ninput.addEventListener('select', evt => {\n    console.log(\"选择起始位置\", input.selectionStart)\n    console.log(\"选择结束位置\", input.selectionEnd)\n    console.log(\"选择内容:\", input.value.substring(input.selectionStart, input.selectionEnd))\n})\n```\n\n只不过，这个事件只能在一些可以编辑的元素中触发，如果是不可编辑的元素，是无法触发的：\n\n```html\n\u003Cdiv id=\"text\">\n  苹果正式官宣：2026年3月4日北京时间22:00\n  举办「Apple Experience」特别活动\n  纽约、伦敦、上海三地同步直播\n  全场重点：iPhone 17e、全新平价Mac、M5全家桶\n\u003C\u002Fdiv>\n```\n\n不过，还有一个事件`selectstart`可以在任意情况下触发，即使不是输入框，只要用户在页面上开始选择都会触发这个事件：\n\n```js\ndiv.addEventListener('selectstart', evt => {\n    console.log(evt)\n})\n```\n\n利用这个事件，我们可以实现阻止用户选择页面文本的效果：\n\n```js\ndiv.addEventListener('selectstart', evt => {\n    console.log(\"用户尝试选择文本\")\n    evt.preventDefault()   \u002F\u002F阻止选择开始\n})\n```\n\n在前面我们为大家介绍了如何利用键盘事件来防止用户使用Ctrl+C复制页面内容，不过，虽然这种方式简单粗暴，但是如果用户有其他复制方式，比如右键菜单复制，依然可以实现内容拷贝，要彻底防止用户复制内容，我们可以使用`copy`事件来处理：\n\n```js\ndiv.addEventListener('copy', evt => {\n    console.log(\"用户尝试复制文本\")\n    evt.preventDefault()\n})\n```\n\n### 页面事件\n\n前面我们讲了鼠标、键盘、表单等事件，它们大多是“用户行为触发”的。而页面级事件用于控制整个页面生命周期，通常是：\n\n- 页面加载完成\n- 页面关闭\n- 页面尺寸变化\n- 页面滚动\n- 网络状态变化\n- 页面可见性变化\n\n这些事件往往和整个页面生命周期有关，页面事件主要绑定在：`window` 对象和`document` 对象上。\n\n当 **DOM 树构建完成** 时会触发`DOMContentLoaded`事件，这个阶段仅仅只是HTML文档树解析完成，不等待图片加载，不等待 CSS 加载：\n\n```js\ndocument.addEventListener('DOMContentLoaded', () => {\n    const div = document.getElementById('text')\n    console.log('DOM结构已经构建完成')\n    console.log(div)   \u002F\u002F此时已经能够拿到DOM元素了\n})\n```\n\n由于不等待CSS加载，有可能这个时候元素的最终渲染尺寸还没有确定，诸如`getBoundingClientRect`这类方法获取的结果并不一定是准确的。\n\n接着就是`load`事件，它会在 **整个页面加载完成** 才会触发，包括DOM、图片、CSS以及一些其他的外部资源在内：\n\n```js\nwindow.addEventListener('load', () => {\n    console.log('页面完全加载完成')\n})\n```\n\n所以，我们前面提到的JS执行时DOM还未构建完成，导致无法获取到元素的问题，也可以通过事件监听的形式来完成，我们只需要在`load`事件中进行处理即可保证所有元素一定加载完成，这也算一种处理方式。\n\n除了页面加载之外，页面在卸载时也会触发相关事件，当用户即将离开页面时，会触发`beforeunload`，比如关闭标签页，页面跳转，刷新页面等情况。`unload`就是页面完全卸载之后触发了。\n\n接着是页面尺寸变化，当浏览器窗口大小改变时会触发`resize`事件：\n\n```js\nwindow.addEventListener('resize', evt => {\n    console.log(evt)\n})\n```\n\n这在响应式布局控制中非常常用。\n\n还有页面滚动时，会触发`scroll`事件：\n\n```js\nwindow.addEventListener('scroll', evt => {\n    console.log(evt)\n})\n```\n\n在后续练习中，我们会利用滚动事件来实现真正的视差滚动效果。\n\n此外，页面可见性变化时也能监听，比如我们想看当前页面是否被切到后台了，使用`visibilitychange`事件来监听：\n\n```js\nwindow.addEventListener('visibilitychange', evt => {\n    console.log(document.visibilityState)   \u002F\u002F通过visibilityState属性查看是否可见\n})\n```\n\n当切换标签页或是最小化窗口时，事件就会触发。\n\n我们甚至还可以监听页面是否发生错误，比如有JS发生了报错：\n\n```js\nwindow.addEventListener('error', function (e) {\n    console.log('发生错误：', e.message)\n})\n\nconsole.log(a)\n```\n\n![image-20260225182451688](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F25\u002Fo6Ka\u002Fimage-20260225182451688.png)\n\n当然不仅仅是JS加载错误，比如图片加载错误也是可以监听到的：\n\n```html\n\u003Cimg id=\"img\" src=\"js.jpg\">\n```\n\n```js\nconst img = document.querySelector('img')  \u002F\u002F针对图片元素进行错误监听\n\nimg.addEventListener('error', function () {\n    console.log('图片加载失败')\n})\n```\n\n## 常用WebAPI\n\n这一部分我们接着来介绍常用的WebAPI，在前端开发中，浏览器提供了一些强大的 API 供 JavaScript 与浏览器进行交互，我们将依次介绍一些常用的API以便大家使用。\n\n### 窗口对象\n\n`window` 是 JavaScript 中的一个全局对象，它是所有对象的顶层，它表示浏览器窗口（或框架），是浏览器环境的核心。它不仅是浏览器窗口的表示，还提供了访问浏览器相关功能的接口。同时，`window` 也是所有浏览器环境中全局对象的引用，我们之前没有使用`new`执行的`this`其实就是指向的`window`对象。\n\n* `window` 是全局作用域的顶级对象，在浏览器中定义了一个“全局环境”。\n* 当你在浏览器中执行代码时，你访问的全局变量（比如 `document`、`console`）都属于 `window` 对象的一部分。\n\n在`window` 对象中有很多重要的属性：\n\n* **`window.document`**：返回当前文档对象。通常用于操作 HTML 文档结构。\n* **`window.location`**：返回当前页面的 URL 信息，并允许修改 URL 来重定向页面。\n* **`window.navigator`**：返回浏览器的信息，如版本、平台、语言等。\n* **`window.console`**：提供与开发者工具相关的功能，如 `console.log()`，`console.error()` 等。\n* **`window.history`**：允许访问浏览器的历史记录，并可以执行前进、后退操作。\n* **`window.innerWidth` \u002F `window.innerHeight`**：返回浏览器视口的宽度和高度。\n* **`window.localStorage` \u002F `window.sessionStorage`**：提供本地存储（`localStorage`）和会话存储（`sessionStorage`）的接口，允许在浏览器中存储数据。\n\n需要注意的是，在浏览器中，所有的全局变量和函数都是 `window` 对象的属性，实际上`var`声明（`let`和`const`不受影响）的全局变量 `x` 等同于 `window.x`，`function`声明的全局函数 `foo()` 等同于 `window.foo()`：\n\n```js\nfunction test() {\n    console.log(\"牛逼\")\n}\n```\n\n![image-20260226162140134](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002FyT6e\u002Fimage-20260226162140134.png)\n\n实际上我们之前使用的很多全局函数都是`window`提供的。\n\n首先我们来介绍一下窗口尺寸相关的属性，在 JS 中，窗口尺寸主要分为两类：**包含滚动条的整体窗口尺寸** 和 **可视区域尺寸**。**可视区域尺寸**就是指浏览器窗口中实际能看到内容的区域大小（不包含浏览器的地址栏、工具栏、开发者工具页面，也不包含滚动条）\n\n* `window.innerWidth`：获取浏览器可视区域的**宽度**（单位：px）\n* `window.innerHeight`：获取浏览器可视区域的**高度**（单位：px）\n\n示例代码：\n\n```js\nconst viewportWidth = window.innerWidth;\nconst viewportHeight = window.innerHeight;\nconsole.log(`可视区域宽度：${viewportWidth}px，高度：${viewportHeight}px`);\n```\n\n我们也可以使用**整体窗口尺寸**，它会完整包含整个浏览器窗口大小：\n\n* `window.outerWidth`：整个浏览器窗口的**宽度**（包含浏览器边框、滚动条、工具栏等）\n* `window.outerHeight`：整个浏览器窗口的**高度**（包含浏览器标题栏、状态栏等）\n\n示例代码：\n\n```js\nconst windowWidth = window.outerWidth;\nconst windowHeight = window.outerHeight;\nconsole.log(`浏览器窗口整体宽度：${windowWidth}px，高度：${windowHeight}px`);\n```\n\n除了获取窗口尺寸之外，我们也可以获取滚动状态，当页面出现滚动条时，我们可以通过：\n\n```js\nwindow.scrollY  \u002F\u002F纵向滚动位置\nwindow.scrollX  \u002F\u002F横向滚动位置\n```\n\n来获取整个页面的滚动位置，不过需要注意的是这两个值不能通过直接修改来改变页面的滚动位置，需要和之前使用元素的内部滚动一样，调用`scroll`相关方法：\n\n```js\n\u002F\u002F 滚动到指定位置\nwindow.scroll({ top: 0 })\n```\n\n接着是`window`对象的一些方法，可以用来控制浏览器窗口或执行其他操作：\n\n* **`window.alert()`**：弹出一个警告框，显示指定的消息。\n* **`window.confirm()`**：弹出一个确认框，返回用户的选择（`true` 或 `false`）。\n* **`window.prompt()`**：弹出一个输入框，允许用户输入内容。\n* **`window.open()`**：打开一个新的浏览器窗口或标签页。\n* **`window.close()`**：关闭当前窗口。\n\n```js\n\u002F\u002F因为是全局对象，调用其中的函数可以直接写成alert\nwindow.alert(\"This is an alert!\");  \u002F\u002F 弹出警告框\nlet userConfirmed = window.confirm(\"Do you confirm this action?\");\n```\n\n有关`window`对象中提供的其他属性，我们会在本节后续内容中继续介绍。\n\n### 浏览导航和历史\n\n`location` 是 `window` 对象的一个属性，它指向一个包含当前文档 URL 信息的对象，你可以通过它读取或修改浏览器地址栏的 URL，从而实现页面跳转、刷新等操作。\n\n首先是`href`属性，它代表浏览器当前访问的完整 URL：\n\n```js\nconsole.log(location.href)\n\u002F\u002F当前浏览器地址栏的地址 http:\u002F\u002Flocalhost:63342\u002FHelloHtml\u002Findex.html\n```\n\n当然完整地址可以拆分成几个部分单独获取，假设 URL 是 `https:\u002F\u002Fwww.example.com:8080\u002Fpath\u002Fpage.html?name=test&age=18#section1`：\n\n| 属性名     | 作用                           | 示例                           |\n| :--------- | :----------------------------- | :----------------------------- |\n| `protocol` | 协议（含冒号）                 | `https:`                       |\n| `host`     | 主机名 + 端口号                | `www.example.com:8080`         |\n| `hostname` | 主机名                         | `www.example.com`              |\n| `port`     | 端口号                         | `8080`                         |\n| `pathname` | 路径（含斜杠）                 | `\u002Fpath\u002Fpage.html`              |\n| `search`   | 查询参数（含问号）             | `?name=test&age=18`            |\n| `hash`     | 锚点（含井号）                 | `#section1`                    |\n| `origin`   | 协议 + 主机名 + 端口号（只读） | `https:\u002F\u002Fwww.example.com:8080` |\n\n针对于`href`属性，我们可以对其进行修改来实现页面跳转：\n\n```js\nlocation.href = 'https:\u002F\u002Fwww.baidu.com'\n```\n\n注意修改`href`必须携带前面的协议，如果不携带会被认为是在当前站点进行跳转，会以路径的形式进行拼接，注意如果是跳转某个子页面，一定要区分最前面加`\u002F`和不加的区别，前者会从根路径开始，后者会以当前路径继续深入。\n\n> 通过这种方式跳转页面都会导致页面强制刷新，即使跳转到相同路径的页面也会刷新。\n\n除了使用这种方式跳转页面之外，我们也可以使用`assign`方法，同样可以实现跳转到指定 URL 的效果：\n\n```js\nlocation.assign('https:\u002F\u002Fwww.baidu.com')\n```\n\n此外，还有一个与`assign`效果相同的`replace`方法，但是它们的区别在于历史记录，我们会在后面接着介绍。\n\n除了页面跳转之外，我们也可以手动刷新页面：\n\n```js\nlocation.reload()   \u002F\u002F简单刷新页面\nlocation.reload(true)   \u002F\u002F强制刷新页面(忽略缓存，重新加载)\n```\n\n我们接着来看历史记录，几乎所有现代化浏览器都有历史记录功能：\n\n![image-20260226182015508](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002FL8ii\u002Fimage-20260226182015508.png)\n\n历史记录中保存了所有你访问过的网站，你可以直接跳转到自己曾经访问过的站点，我们也可以在页面的顶部点击左右箭头符号来实现向前或向后跳转：\n\n![image-20260226183408716](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002FoV9b\u002Fimage-20260226183408716.png)\n\n当然，如果只是在某一个标签页内，前进后退的范围仅限于当前这个标签所访问过的站点或地址。在浏览器中也是使用的一个类似于栈结构的容器进行存放：\n\n![image-20260226225444538](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002FSf5y\u002Fimage-20260226225444538.png)\n\n`history` 也是 `window` 对象的属性，它管理浏览器的会话历史记录（即你访问过的页面列表），可以实现 “前进”“后退”“新增 \u002F 替换历史记录” 等操作，是前端路由（如 HistoryRouter）的核心：\n\n```js\nconsole.log(history.length)  \u002F\u002F获取栈长度\n```\n\n长度就是当前标签页访问过的所有可以回退的页面数量，也就是历史记录的数量。除此之外，还有`scrollRestoration`属性，它保存的是滚动恢复策略：\n\n```js\nconsole.log(history.scrollRestoration)\n```\n\n它的作用是控制浏览器在历史记录之间导航时（后退 \u002F 前进）是否自动恢复页面的滚动位置，默认情况下值为`auto`，浏览器自动恢复到上一次的滚动位置，我们也可以设置为`manual`来让浏览器不自动恢复滚动位置，需手动控制。\n\n我们接着来看看历史记录的核心方法：\n\n| 方法名      | 作用                                                         | 示例                          |\n| :---------- | :----------------------------------------------------------- | :---------------------------- |\n| `back()`    | 回退到上一条历史记录（等价于点击浏览器后退按钮）             | `history.back()`              |\n| `forward()` | 前进到下一条历史记录（等价于点击浏览器前进按钮）             | `history.forward()`           |\n| `go(n)`     | 跳转到指定偏移量的历史记录：- `n=1`：前进 1 步- `n=-1`：后退 1 步- `n=0`：刷新当前页 | `history.go(-2)` \u002F\u002F 后退 2 步 |\n\n```js\nhistory.back() \u002F\u002F 后退一步\nhistory.forward() \u002F\u002F 前进一步\nhistory.go(-2) \u002F\u002F 后退两步\nhistory.go(1) \u002F\u002F 前进一步\nhistory.go(0) \u002F\u002F 刷新当前页\n```\n\n注意历史记录的跳转（无论是JS还是浏览器直接点击返回前进按钮）可能会导致页面刷新，当两个历史记录的页面路径或地址不同时，都会导致页面刷新。\n\n我们在前面提到的`assign`和`replace`方法，最大的区别就在于是否存在历史记录，如果是`replace`相当于直接将当前页面替换成了一个新的地址，也就是说会把当前页面的历史记录信息给覆盖，而`assign`相当于是进行一次新的调跳转，跳转之后之前的页面会作为历史记录存放。\n\n除了进行页面跳转会增加历史记录，我们也可以手动插入一个新的历史记录：\n\n```js\nhistory.pushState(null, '', 'index.html')\n```\n\n这里需要传入3个参数：\n\n* **`state` (对象)**：一个与新历史记录条目相关的状态对象。每当用户导航到该状态时，`popstate` 事件就会被触发，你可以从事件中或是`history.state`读取这个对象。如果不需要，传 `null` 即可。\n* **`title` (字符串)**：目前大多数浏览器都会忽略这个参数，通常传空字符串 `\"\"`。\n* **`url` (字符串)**：新的历史记录条目地址。**注意：** 为了安全，这个 URL 必须与当前页面同源（同一个域名）\n\n执行后，和之前一样，上一个页面的路径会作为历史，我们依然可以点击返回变回之前的页面，接着页面链接路径会替换为`pushState`传入的`url`参数，但不会刷新页面。这对于一些SPA（单页面应用）的实现，比如 VueRouter 就非常关键，如果我们通过`location.href`地址栏的 URL 会导致页面刷新。 而使用 `pushState`，你可以改变 URL，让用户觉得“跳转了页面”，但实际上网页内容是通过 JavaScript 动态更新的，整个过程非常平滑，没有白屏。\n\n与其类似的还有一个`replaceState`，不过和之前的`replace`方法一样，它会直接替换当前的历史记录，而不是保存之前并插入一个新的，所以这种方式是无法回退之前路径的。\n\n### 浏览器信息\n\n`navigator` 用于获取**当前浏览器和设备环境信息**，简单来说就是浏览器的“自我介绍”。你可以通过它知道：浏览器类型、版本、操作系统、是否联网、用户使用的语言、摄像头\u002F麦克风权限、地理位置、剪贴板等等。\n\n很多常用的WebAPI属性都可以通过`navigator`来获取，我们先从浏览器信息开始介绍。其中最主要的就是`userAgent`属性，可以获取浏览器的 **用户代理字符串**：\n\n```js\nconsole.log(navigator.userAgent)\n```\n\n![image-20260226232916113](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002FdgM3\u002Fimage-20260226232916113.png)\n\n可以看到这个结果里面包含了很多信息，包括当前使用的浏览器名称、操作系统版本等，只不过这种方式只能获取一个大概的信息，但实际版本和名称并不准确，并且部分浏览器是可以随意修改UA信息的，所以**不推荐用于精确判断**。要简单判断下是否移动端还是没问题的：\n\n```js\nif (\u002FMobile\u002F.test(navigator.userAgent)) {\n  console.log(\"移动端\")\n} else {\n  console.log(\"桌面端\")\n}\n```\n\n我们接着来看下一个属性，`language`用于获取浏览器语言，比如要判断是否中文：\n\n```js\nif (navigator.language.startsWith(\"zh\")) {\n  console.log(\"中文环境\")\n}\n```\n\n`onLine`可以判断是否联网：\n\n```js\nconsole.log(navigator.onLine)\n```\n\n大家可以试试关闭电脑WIFI或是拔掉主机的网线，观察网络状态变化，还可以结合事件监听：\n\n```js\nwindow.addEventListener(\"online\", () => {\n  console.log(\"网络已恢复\")\n})\n\nwindow.addEventListener(\"offline\", () => {\n  console.log(\"网络已断开\")\n})\n```\n\n我们还可以通过`geolocation`获取用户位置，不过此操作需要用户授权，所以我们只能以回调形式传入获取之后的处理逻辑：\n\n```js\nnavigator.geolocation.getCurrentPosition(\n  position => {   \u002F\u002F用户允许执行，则进一步执行回调\n    console.log(position.coords.latitude)  \u002F\u002F通过coords获取经纬度\n    console.log(position.coords.longitude)\n    console.log(position.coords.accuracy)   \u002F\u002F坐标精确度\n  },\n  error => {\n    console.log(\"获取失败\")   \u002F\u002F如果用户不允许，则执行错误回调\n  }\n)\n```\n\n![image-20260226234407496](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002F7Nct\u002Fimage-20260226234407496.png)\n\n这里我们可以点击访问该网站时允许（就和手机App授权是一样的）来让浏览器可以访问当前的地理信息。\n\n接着是剪贴板，浏览器可以实现在用户不主动执行Ctrl+V的情况下，直接读取用户的剪贴板内容，不过和上面一样，由于剪贴板里面可能涉及到用户隐私信息，所以需要用户手动授权才能使用：\n\n```js\nnavigator.clipboard.readText().then(text => {\n    console.log(text)\n})\n```\n\n![image-20260226235048260](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002FVu3v\u002Fimage-20260226235048260.png)\n\n这里`readText()`会返回一个Promise对象，当用户允许剪贴板之后，Promise才会正常resolve，如果用户拒绝了读取剪贴板的请求，会直接失败：\n\n![image-20260226235225136](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002FyWy7\u002Fimage-20260226235225136.png)\n\n除了读取剪贴板的内容之外，我们也可以对剪贴板内容进行修改：\n\n```js\nnavigator.clipboard.writeText(\"Hello\")\n```\n\n最后是摄像头和麦克风，我们可以通过`navigator.mediaDevices`来获取媒体设备：\n\n```js\nnavigator.mediaDevices.getUserMedia({ \n    video: true,   \u002F\u002F如果需要获取摄像头，就开启\n    audio: true    \u002F\u002F如果需要获取麦克风，就开启\n}).then(stream => {\n    console.log(\"获取成功\")\n}).catch(err => {\n    console.log(\"获取失败\")  \u002F\u002F用户不授权或是根本没有对应的媒体设备就失败\n})\n```\n\n由于也会涉及到用户隐私，所以这里依然需要用户手动授权，否则会导致获取失败：\n\n![image-20260227000454298](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002FMw8a\u002Fimage-20260227000454298.png)\n\n如果成功获取到，这里会得到一个`stream`对象，它代表数据的输入流，大家可以理解为视频数据像水流一样不断涌入，我们可以把这个流绑定到一个`video`元素上：\n\n```html\n\u003Cvideo id=\"video\" width=\"640\" height=\"480\" autoplay playsinline>\u003C\u002Fvideo>\n```\n\n```js\nnavigator.mediaDevices.getUserMedia({\n    video: { width: 640, height: 480 },  \u002F\u002F视频可以进一步设置摄像头录制的宽度和高度以及其他属性\n    audio: true\n}).then(stream => {\n    const video = document.getElementById('video')\n    video.srcObject = stream  \u002F\u002F将video元素的srcObject属性设置为这里的流即可\n})\n```\n\n现在很多的一些视频会议都可以靠`mediaDevices`来实现。除了摄像头录制之外，我们也可以对屏幕进行录制，这里我们直接使用另一个方法`getDisplayMedia`来实现，注意屏幕录制也需要单独的权限：\n\n![image-20260227001743410](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002Flb2S\u002Fimage-20260227001743410.png)\n\n```html\n\u003Cdiv style=\"position: relative;width: fit-content\">\n  \u003Cvideo style=\"position: absolute; bottom: 20px; right: 20px;border: black 2px solid;border-radius: 10px\"\n         id=\"user\" width=\"320\" height=\"240\" autoplay playsinline>\u003C\u002Fvideo>\n  \u003Cvideo id=\"screen\" width=\"1280\" height=\"720\" autoplay playsinline>\u003C\u002Fvideo>\n\u003C\u002Fdiv>\n```\n\n```js\nnavigator.mediaDevices.getDisplayMedia({\n    video: {\n        cursor: 'always', \u002F\u002F 显示鼠标光标\n        displaySurface: 'monitor' \u002F\u002F 可选：monitor\u002Fwindow\u002Fbrowser\n    },\n    audio: false  \u002F\u002F 屏幕音频就不获取了\n}).then(stream => {\n    const video = document.getElementById('screen')\n    video.srcObject = stream\n})\n```\n\n这样就可以实现会议室分享屏幕加在线聊天效果了。\n\n### 文档对象\n\n我们在前面的学习中认识了`document`对象，它代表我们整个页面文档，通过它，你可以用代码访问、修改甚至删除网页上的任何内容。这一部分，我们来深入了解一下它的更多内容。\n\n首先就是它包含了页面的诸多元数据，如 `title`, `URL`, `characterSet`等：\n\n```js\nconsole.log(document.title)  \u002F\u002F网页标题\nconsole.log(document.characterSet)   \u002F\u002F编码格式 \n```\n\n我们可以尝试修改网页的编码格式，此时`document`的对应属性也会跟着发生变化：\n\n![image-20260227003430220](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002FUzr0\u002Fimage-20260227003430220.png)\n\n如果我们想要获取页面的地址，也可以通过URL属性来获得：\n\n```js\nconsole.log(document.URL)\n```\n\n![image-20260227003716723](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002FhUr8\u002Fimage-20260227003716723.png)\n\n这个属性和`location.href`类似，但是这里的属性是只读无法修改的。\n\n前面我们介绍了如何向页面添加元素，除此之外，我们还可以直接向页面打印文本内容，会以文本节点的形式进行插入：\n\n```js\ndocument.writeln(\"Hello World!\")\n```\n\n![image-20260227004324417](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F26\u002FmS9w\u002Fimage-20260227004324417.png)\n\n前面我们学习了如何获取元素，我们可以选择通过元素的`id`或`class`属性来快速拿到对应的元素，除此之外，我们还可以一次性获取所有指定类型的元素：\n\n```js\nconsole.log(document.links)  \u002F\u002F获取所有的链接\nconsole.log(document.images)  \u002F\u002F获取所有的图片\nconsole.log(document.forms)  \u002F\u002F获取所有的表单  \n```\n\n### 数据存储\n\n浏览器本身是“无状态”的，当页面一刷新，我们创建的所有变量、函数、数据都会烟消云散，为了保证用户每次访问网站时能够保留之前使用的一些数据，我们需要一种机制，把数据“保存下来”。\n\n浏览器常见的数据存储方式有以下方式：\n\n| 存储方式       | 大小      | 生命周期                 | 是否发给服务器 | 适用场景 |\n| -------------- | --------- | ------------------------ | -------------- | -------- |\n| Cookie         | 小（4KB） | 可设置                   | ✅ 会自动发送   | 登录状态 |\n| localStorage   | 5~10MB    | 永久                     | ❌ 不会         | 本地缓存 |\n| sessionStorage | 5MB       | 会话级，标签页关闭就消失 | ❌ 不会         | 临时状态 |\n| IndexedDB      | 很大      | 永久                     | ❌ 不会         | 大型数据 |\n| Cache Storage  | 很大      | 永久                     | ❌ 不会         | PWA缓存  |\n| Memory         | 小        | 页面刷新就消失           | ❌ 不会         | 临时变量 |\n\nMemory就是我们一直以来使用的数据存储方式，所有的数据都存储在内存中，页面关闭之后自动释放，不留下任何痕迹。这一节我们来介绍其他五种数据存储方式。\n\n首先是最简单的`LocalStorage`，它是 **Web Storage API** 的一部分，用来在浏览器中**持久化存储数据**。所有存储在其中的数据都不会随着页面的刷新而消失，关闭浏览器后仍然存在。\n\n在`LocalStorage`中，数据是以键值对形式存储的，类似于Map，我们可以直接使用：\n\n```js\n\u002F\u002F注意键值对需要以字符串形式存放\nlocalStorage.setItem(\"AAA\", \"建材王总\")\n```\n\n由于键值对只能以字符串形式存放，所以这里建议将对象转换为JSON格式进行存储，直接存储对象会导致类型转换，变成`\"[object Object]\"`。\n\n添加键值对后，我们可以直接在开发者工具中查看存放了哪些数据：\n\n![image-20260227112034591](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F27\u002F3tdZ\u002Fimage-20260227112034591.png)\n\n存储在`LocalStorage`的数据是永久保存的，即使我们刷新网页，下次依然可以访问到。我们可以使用`getItem`来获取之前保存的数据：\n\n```js\nconsole.log(localStorage.getItem(\"AAA\"))\n```\n\n其他方法如下：\n\n```js\nlocalStorage.key(0)   \u002F\u002F按下标位置获取Key\nlocalStorage.removeItem(\"AAA\")   \u002F\u002F移除键值对\nlocalStorage.length   \u002F\u002F键值对数量\nlocalStorage.clear()   \u002F\u002F清空所有数据\n```\n\n不过需要注意的是，`LocalStorage`采用的是同源策略，如果我们跳转到其他站点上，在之前站点存储的数据是不通用的，每个站点的数据独立存放：\n\n![image-20260227113451949](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F27\u002FWzx3\u002Fimage-20260227113451949.png)\n\n所以各位小伙伴无需担心其他网站污染了我们自己存储的数据，因为都是独立的。接着我们要介绍一个和`LocalStorage`非常类似的存储方案：`SessionStorage`，它的用法和前者非常类似，也是键值对：\n\n```js\nsessionStorage.setItem(\"AAA\", \"建材王总\")\n```\n\n与`LocalStorage`的区别是，它是存储在会话存储空间中的：\n\n![image-20260227113959061](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F27\u002FlVf6\u002Fimage-20260227113959061.png)\n\n会话存储空间和本地存储空间不同，当我们关闭当前浏览器标签页时，会话会自动结束，同时也会清理会话空间存储的数据。注意，只要当前标签页不关闭，即使我们刷新网页也不会导致`SessionStorage`被清理，因为会话没有结束，所以一定要注意它和`Memory`数据的区别。\n\n还有就是`SessionStorage`也遵循同源策略，但仅限于同标签页内：\n\n1. 即使在同一个会话下跳转到其他页面也是无法读取的\n2. 由于是会话级别的数据，所以不同标签页之间即使是同一个站点也是无法读取的\n\n这种存储更适合一些表单临时数据、支付流程存储等场景。\n\n接着我们来介绍一下**Cookie**，它是浏览器提供的一种**小型文本数据存储机制**，用于在客户端（浏览器）保存少量数据，并在之后的请求中自动携带给服务器。\n\nCookie的格式是：\n\n```\nname=value; expires=时间; path=\u002F; domain=xxx; secure; HttpOnly; SameSite\n```\n\n它包含自己的名称、过期时间、路径、所属的域、安全性、仅网络可用、同源策略。\n\n要添加Cookie有两种方式：\n\n* 服务器通过 Set-Cookie 响应头来为客户端添加新的Cookie\n* 通过`cookieStore`手动添加新的Cookie\n\n由于我们还未解除到服务端环境，所以这里我们仅演示第二种方式，使用`set`方法即可：\n\n```js\ncookieStore.set({\n    name: 'test',   \u002F\u002F传入需要设置的Cookie名字\n    value: '我是数据'   \u002F\u002FCookie的数据\n})\n```\n\n![image-20260227121328343](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F27\u002FDhj2\u002Fimage-20260227121328343.png)\n\n可以看到，这里我们成功插入了一个新的Cookie，名称和值就是我们刚刚设定好的内容。需要注意的是，设置Cookie和之前的`LocalStorage`不同，它不是同步进行的，这里会得到一个Promise对象，当设置Cookie真正完成之后，才会结果。\n\n我们可以尝试刷新网页重新请求页面，可以看到请求头会携带我们添加的自定义Cookie：\n\n![image-20260227121839241](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F27\u002Fn5Ti\u002Fimage-20260227121839241.png)\n\n像一些用于表明身份的令牌数据，就可以通过Cookie来进行传递，所以你也可以把Cookie理解为是浏览器帮服务器“记住用户身份”的小纸条。当然，如果存在多条Cookie信息，那么会以分号分隔：\n\n![image-20260227134548611](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F27\u002F9Aat\u002Fimage-20260227134548611.png)\n\n我们也可以通过`docuemnt.cookie`来获取成品字符串：\n\n```js\nconsole.log(document.cookie)\n```\n\n默认情况下，我们创建的Cookie是存在过期时间的，如果什么都不设置那么过期时间就是会话周期内，这和前面介绍的`SessionStorage`是一样的。当Cookie过期之后，就会自动被移除，我们也可以手动为Cookie设置过期时间：\n\n```js\ncookieStore.set({\n    name: 'test',\n    value: '我是数据',\n    expires: Date.now() + 3600 * 1000  \u002F\u002F当前时间加上1个小时\n})\n```\n\n![image-20260227132413418](https:\u002F\u002Ffiles.seeusercontent.com\u002F2026\u002F02\u002F27\u002Fkb6O\u002Fimage-20260227132413418.png)\n\n这里简单介绍一下后续几个参数：\n\n* domain：当前Cookie可用的域名，注意只能是当前站点或子站点的域名，不能随意指定\n* path：当前Cookie生效的路径起始位置，默认是`\u002F`表示所有路径都适用\n* secure：表示只有在安全环境下才能使用，比如localhost或是https站点\n* HttpOnly：表示只用作请求，不能被JS随意读取或修改\n* SameSite：控制跨站请求是否携带Cookie信息，默认是严格模式，完全禁止跨站携带（比如在某个网站流量数据，结果后台恶意发起一条银行网站的转账请求，如果携带Cookie将会导致账户盗刷）\n\n不过，虽然前面介绍这三种存储方案都各有好处，但是都存在一个很大的问题，就是容量有限制。接下介绍一下**IndexedDB**，它也是浏览器提供的一种**本地数据库存储方案**，用于在客户端存储大量结构化数据。它比 `localStorage`、`sessionStorage` 更强大，它支持存储大量数据（MB ~ GB 级）就像后端的数据库一样，可以装下很多东西。\n\n还有一种是专用于PWA渐进式应用程序页面缓存的`CacheStorage`，由于目前暂时接触不到，所以这里暂时不做详细介绍。\n\n有关`IndexedDB`和`CacheStorage`的内容，我们会放到后续的《前端工程化》课程中进行详细介绍。\n\n## 本章练习\n\n### 实现真视差滚动\n\n### 苹果官网完善","本章知识仅适用于Web前端开发学习路线或全栈开发路线，如果仅使用JavaScript做后端开发，不强制要求学习。前面我们为大家介绍了JS的基础语法部分，这一章我们将继续深入Web浏览器环境，学习然后控制页面上的元素并实现各种高级操作。","2026-02-27 22:43:15",{"data":468,"status":460,"success":461},[469,474],{"id":8,"image":470,"link":471,"name":472,"type":473},"\u002Fimage\u002Fadv\u002Frainyun-2025-06.webp","https:\u002F\u002Fwww.rainyun.com\u002Fitbaima_","雨云优惠购","cloud",{"id":66,"image":475,"link":476,"name":477,"type":478},"\u002Fimage\u002Fadv\u002Fsimcard-2025-11.webp","https:\u002F\u002Fmall.itbaima.cn","号卡优惠","simcard"]