JavaScript高级

简介: 本文系统讲解前端DOM、BOM、事件机制与Ajax核心知识:涵盖DOM树结构、常用操作API、属性与样式的区别,BOM浏览器检测与URL解析,事件绑定、冒泡流程及代理应用,原生Ajax与JSONP跨域原理,并对比cookie、localStorage与sessionStorage的特性与限制,助力深入理解Web开发基础。

2.1 DOM操作
2.1.1 DOM的本质是什么?

/
<?xml version="1.0" encoding="UTF-8"> // 告诉浏览器以哪一种类型进行解析


/

<!DOCTYPE html>                                      // 以html类型进行解析文档内容
<html>
    <body>

    </body>
</html>

Copy
2.1.2 DOM操作的常用API有哪些?
节点查找API

  1. document.getElementById :根据ID查找元素,大小写敏感,如果有多个结果,只返回第一个;
  2. document.getElementsByClassName :根据类名查找元素,多个类名用空格分隔,返回一个 HTMLCollection 。注意兼容性为IE9+(含)。另外,不仅仅是document,其它元素也支持 getElementsByClassName 方法;
  3. document.getElementsByTagName :根据标签查找元素, * 表示查询所有标签,返回一个 HTMLCollection 。
  4. document.getElementsByName :根据元素的name属性查找,返回一个 NodeList 。
  5. document.querySelector :返回单个Node,IE8+(含),如果匹配到多个结果,只返回第一个。
  6. document.querySelectorAll :返回一个 NodeList ,IE8+(含)。
  7. document.forms :获取当前页面所有form,返回一个 HTMLCollection ;
    节点创建API
  8. createElement创建元素
  9. createTextNode创建文本节点
  10. cloneNode 克隆一个节点
  11. createDocumentFragment
    节点修改API
  12. appendChild
  13. insertBefore
  14. insertAdjacentHTML
  15. Element.insertAdjacentElement()
  16. removeChild
  17. replaceChild
    节点关系API
    ● 1、父关系API
    parentNode :每个节点都有一个parentNode属性,它表示元素的父节点。Element的父节点可能是Element,Document或DocumentFragment; parentElement :返回元素的父元素节点,与parentNode的区别在于,其父节点必须是一个Element元素,如果不是,则返回null;
    ● 2、子关系API
    children :返回一个实时的 HTMLCollection ,子节点都是Element,IE9以下浏览器不支持;
    childNodes :返回一个实时的 NodeList ,表示元素的子节点列表,注意子节点可能包含文本节点、注释节点等;
    firstChild :返回第一个子节点,不存在返回null,与之相对应的还有一个 firstElementChild ;
    lastChild :返回最后一个子节点,不存在返回null,与之相对应的还有一个 lastElementChild ;
    ● 3、兄弟关系型API
    previousSibling :节点的前一个节点,如果不存在则返回null。注意有可能拿到的节点是文本节点或注释节点,与预期的不符,要进行处理一下。
    nextSibling :节点的后一个节点,如果不存在则返回null。注意有可能拿到的节点是文本节点,与预期的不符,要进行处理一下。
    previousElementSibling :返回前一个元素节点,前一个节点必须是Element,注意IE9以下浏览器不支持。
    nextElementSibling :返回后一个元素节点,后一个节点必须是Element,注意IE9以下浏览器不支持。
    元素属性型API
    1、setAttribute 给元素设置属性: 2、getAttribute 3、hasAttribute
    样式操作API(面试考点)
    ● 1、直接修改元素的样式
    elem.style.color = 'red';
    elem.style.setProperty('font-size', '16px');
    elem.style.removeProperty('color');
    Copy
    ● 2、动态添加样式规则
    var style = document.createElement('style');
    style.innerHTML = 'body{color:red} #top:hover{background-color: red;color: white;}';
    document.head.appendChild(style);
    Copy
    ● 3、classList获取样式属性
    Note
    了解dom节点样式(classList)的remove, add, toggle, contains, replace等方法的使用。
    ● 4、window.getComputedStyle 通过 element.sytle.xxx 只能获取到内联样式,借助 window.getComputedStyle 可以获取应用到元素上的所有样式,IE8或更低版本不支持此方法。
    var style = window.getComputedStyle(element[, pseudoElt]);
    Copy
    2.1.3 DOM节点的attr和proerty的区别
    // proprty------>>>获取nodeName和nodeType(property实际上就是JS对象的一个属性)如:text : 3(#text), p : 1(p)
    console.log(pList[0].nodeName); // p
    console.log(pList[0].nodeType); // 1
    // attribute------>>> 获取一个HTML标签的属性信息(设置和修改)
    var a = document.getElementsByTagName('a')[0];
    console.log(a.getAttribute('data-origin'));
    console.log(a.getAttribute("href_name")); // 可以给一个HTML标签设置任意的属性名称,无论这个属性内部是否存在
    a. setAttribute('sex', 'male')
    // 区别:
    // (JS对象&HTML标签)property实际上是一个普通JS对象本身的基本属性,但是attribute实际上是一个HTML标签上面的属性信息
    Copy
    Note
    区别:(JS对象&HTML标签)property实际上是一个普通JS对象本身的基本属性,但是attribute实际上是一个HTML标签上面的属性信息
    2.2 BOM操作
    2.2.1 如何检测浏览器的类型?
    var ua = navigator.userAgent;
    var isChrome = ua.indexOf('Chrome');
    console.log('is Chrome?', isChrome > 0, navigator.userAgent);
    console.log('电脑屏幕大小:', screen.width, screen.height)
    Copy
    2.2.2 如何拆解URL的各个部分?
    // location
    console.log(location.href); // 完整的url地址,http://localhost:8080/JS-Professional/begin/02-%E5%9F%BA%E7%A1%80%E8%BF%9B%E9%98%B6/01-%E4%BB%8E%E5%9F%BA%E7%A1%80%E5%88%B0%E8%BF%9B%E9%98%B6.html?_ijt=i8ctmkc87dvh03tdaf51rn5v1i
    console.log(location.protocol) // http/https
    console.log(location.host, location.hostname) // www.52tech.tech
    console.log(location.pathname); // host之后的全部内容,JS-Professional/begin/02-%E5%9F%BA%E7%A1%80%E8%BF%9B%E9%98%B6/01-%E4%BB%8E%E5%9F%BA%E7%A1%80%E5%88%B0%E8%BF%9B%E9%98%B6.html
    console.log(location.search); // search就是?之后的全部内容,?_ijt=i8ctmkc87dvh03tdaf51rn5v1i(包括?)
    console.log(location.hash); // #后面的内容(包括#)
    Copy
    2.3 事件操作
    2.3.1 编写一个通用的事件监听函数
    function bindEvent(ele, type, selector, fn){

     if (fn == null) {
         fn = selector;
         selector = null;
     }
    
     // addEventListener 的最后一个参数默认是false, 表示的是事件冒泡,自下向上去捕获事件,true:表示事件捕获,表示事件自外向里的方式
     ele.addEventListener(type, function(e){
         if (selector) {
             // 使用的是代理的方式的话
             if (e.target.matches(selector)){            // 这里需要去判断当前点击的那个对象是不是我点击的那个对象
                 fn.call(e.target, e);
             }
         } else {
             // 不使用代理的话
             fn(e);
         }
     });
    

    }

// 这里表示对这个div1 内部的所有的a标签使用事件冒泡
bindEvent(document.getElementById('div4'), 'click', 'p', function (e) {
    console.log(this.innerHTML + 'hahaah')
})

版本2:
/**

 * 实现一个通用的事件绑定函数(代码简洁,减少了浏览器的占用)
 * @param element
 * @param type
 * @param selector
 * @param fn
 */
function bindEvent(element, type, selector, fn) {
    // if (fn === null || fn === undefined)        // 这里可以直接优化为fn == null
    if (fn == null){
        // 只有3个参数的话
        fn = selector;
        selector = null;
    }

    element.addEventListener(type, function (e) {
        var target;
        // 如果selector有的话,说明此事element就是一个代理
        if (selector) {
            target = e.target;
            // 如果当前的target满足这个选择器的话
            // 如果元素被指定的选择器字符串选择,Element.matches()  方法返回true; 否则返回false。
            // document.getElementById('div1').matches('div')    true
            // matches里面的参数实际上是一个HTML标签:a, p, div …………, 内容也是不区分大小写的,有点类似于nodeName 或者nodeType 这样的判断
            if (target.matches(selector)){
                // 修改this的指向
                fn.call(target, e);
            }
        }else {
            fn(e);
        }
    });
}

Copy
2.3.2 描述事件冒泡的流程
Note
"DOM2级事件”规定的事件流包含三个阶段:事件捕获阶段,处于目标阶段和事件冒泡阶段。首先发生的是事件捕获,然后是实际的目标接收到事件,最后阶段是冒泡阶段。
● 事件冒泡: 按照DOM树形结构向上冒泡(p --- >>> div --- >>> body --- >>> document),
● 事件捕获 (document—>—>—>的顺序进行传播的)
2.3.3 对于一个无线下拉图片的页面,如何给每一个图片绑定一个事件
// 代理(div3里面的所有a标签都需要进行事件的监听)
var div3 = document.getElementById('div3');
bindEvent(div3, 'click', function (e) {
e.preventDefault();
e.stopPropagation()
// 可以获取真实触发的那个元素
var target = e.target;
console.log(target.nodeName, target.nodeType); // 每一种页面标签实际上都有一个自己专属的nodeName
// 这里的目的主要是用于过滤,只处理a标签的事件处理
if ('A' === target.nodeName) {
// 如果当前点击的a标签就是自己
alert(target.innerHTML)
}
})
Copy
2.4 Ajax操作
2.4.1 手动编写一个ajax,不依赖第三方库
2.4.2 跨域的几种实现方式以及底层的实现原理

  1. 服务器端的使用:response.setHeader(Access-Control-Allow-Origin, "http://www.baidu.com, http://www.52tech.tech")允许跨域
  2. 使用JSONP
    2.4.3 JSONP的实现原理
    Note
    跨域:浏览器有同源策略,不允许ajax访问其他域接口
    ● 跨域条件:协议、域名、端口,有一个不同就是跨域
    可以跨域加载资源的3个标签:
    , ,
相关文章
|
2月前
|
Java 测试技术 Linux
生产环境发布管理
本文介绍大型团队如何通过自动化部署平台实现多环境(dev/test/pre/prod)高效发布与运维。涵盖各环境职责、基于Jenkins+K8S的CI/CD流程、分支管理、一键发布及Skywalking日志链路追踪,提升发布效率与问题排查速度。
|
2月前
|
敏捷开发 Dubbo Java
需求开发人日评估
本文介绍敏捷开发中工时评估的关键——人日估算方法,涵盖开发、自测、联调、测试及发布各阶段周期参考,并提供常见需求如增删改查、导入导出、跨服务调用等的典型人日标准,助力团队科学规划迭代。
|
2月前
|
存储 NoSQL 关系型数据库
4-MongoDB索引知识
MongoDB索引通过B树结构提升查询效率,避免全表扫描。支持单字段、复合、地理空间、文本及哈希索引,适用于等值、范围、排序及全文检索等场景,显著优化大数据量下的查询性能。
|
2月前
|
NoSQL Linux Shell
2-MongoDB单机部署
本文介绍MongoDB在Windows和Linux系统的安装启动方法,包括下载地址、版本选择、解压配置、命令行与配置文件启动方式,以及Shell连接和图形化工具Compass的使用。同时涵盖Linux下的部署、防火墙设置与服务启停操作,附带各环境安装包说明。
|
2月前
|
SQL 监控 机器人
钉钉通知
本文介绍如何通过Java调用钉钉机器人API实现系统告警消息发送,支持文本、Markdown等多种格式。需创建自定义机器人并设置关键词,每分钟限20条,超量将被限流。建议整合消息摘要发送。可通过封装工具类、结合Nacos配置管理实现灵活调用,用于异常日志、慢SQL等实时监控场景,提升问题响应效率。
|
2月前
|
存储 小程序 API
微信通知
基于企业微信与小程序集成,通过API获取access_token,查询通讯录并匹配医生信息,异步发送小程序通知消息。需配置企业微信appId、secret及小程序信息,实现订单提醒等场景的精准推送。
|
2月前
|
NoSQL Java 测试技术
5-MongoDB实战演练
基于SpringDataMongoDB实现头条文章评论功能,涵盖增删改查、按文章ID查询及评论点赞。通过MongoTemplate优化点赞操作,提升性能,并利用索引提高查询效率,构建高效稳定的微服务模块。
|
2月前
|
敏捷开发 Java 测试技术
为什么要单元测试
单元测试看似“踩刹车”,实则让开发跑得更快。本文从测试演进史切入,解析为何高质量单测能提升代码质量、加速迭代、增强重构信心,并揭示常见误区与反模式,倡导以单元测试筑牢软件根基,实现高效持续交付。
|
2月前
|
SQL 分布式计算 运维
XXLJOB:超长定时任务慢节点优化实践
本文针对ODPS大宽表任务耗时严重问题,通过定位卡点、资源调优与数据倾斜处理实现快速止血,并深入梳理代码结构,发现计算堆积、动态倾斜及回刷成本高等问题。最终提出视图落表、前置裁剪、分布式MapJoin优化及节点拆分等方案,将产出时间从13:00提前至8:30,提升效率4小时以上,显著降低资源消耗与维护成本。
|
2月前
|
存储 NoSQL 关系型数据库
1-MongoDB相关概念
MongoDB是一款高性能、无模式的文档型NoSQL数据库,适用于高并发、海量数据、高扩展性场景。适用于社交、游戏、物联网等写多读多、事务要求不高的应用,支持丰富查询、水平扩展与高可用,相比MySQL更灵活、成本更低。