JS中DOM编程总结

Get Started

  • 网页结构
  • 节点的增删改查
  • 跨线程操作
  • 属性同步
  • Property 与 Attribute 的区别

网页其实是一颗树。

JS如何操作这个树,浏览器往window上加一个document即可。
JS用document操作网页,这就是Document Object Model文档对象模型。
事实:DOM很难用

获取元素,也叫标签(节点包括元素和文本等)

• 有很多API

    window.idxxx或者直接idxxx
    document.getElementByld('idxxx')
    document.getElementsByTagName('div')[0]
    document.getElementsByClassName('red')[0]
    document.querySelector('#idxxx')
    document.querySelectorAll('.red')[0]

• 有s的都得加下标
• 用哪个
工作中用querySelector和querySelectorAll
要兼容IE的才用getElement(s)ByXXX

获取特定元素

• 获取html元素
document.documentElement
• 注意:相当于属性,而不是函数,不用加()
• 获取head元素
document.head
• 获取body元素
document.body
• 获取窗口(窗口不是元素)虽然window不是标签,但是有时候会获取window加一些全局的事件监听
window
• 获取所有元素(将内部的所有标签一字排开)
document.all
这个是个奇葩,第6个falsy值,使用if之后为假。

我们获取到的元素是一个对象,抓取一只div对象看一下

console.dir(div1)看原型链
• ※Chrome显示错了(显示内容.pritotype)
• 自身属性:className、id、style等
• 第一层原型HTMLDivElement.prototype
• 这里面是所有div共有的属性,不用细看
• 第二层原型HTMLElement.prototype
• 这里面是所有HTML标签共有的属性,不用细看
• 第三层原型Element.prototype
• 这里面是多有XML、HTML标签的共有属性。
• 第四层原型Node.prototype
• 这里是所有节点共有的属性,节点包括XML标签文本注释、HTML标签文本注释等等
• 第五层原型EventTarget.prototype
• 里面最重要的函数属性是addEventListener
• 最后一层原型就是Object.prototype了

节点?元素?

节点Node包括一下几种
• MDN有完整描述,x.nodeType得到一个数字
• 1 表示元素Element,也叫标签Tag
• 3 表示文本Text
• 8 表示注释Comment
• 9 表示文档Document
• 11 表示文档片段DocumentFragment
• 记住1和3即可
注意:回车也算一个文本节点。

节点的增删改查

• 创建一个标签节点
let div1 = document.createElement('div')
document.createElement('style')
document.createElement('script')
document.createElement('li')
• 创建一个文本节点
text1= =document.createTextNode('你好')
• 标签里面插入文本
div1.appendChild(text1)(这个是Node提供的接口)
'div1.innerText = '你好'(IE) 或者 div.textContent = '你好''(标准)(这里是element提供的接口)
但是不能用div1.appendChild('你好')
• 插入页面中
创建的标签默认处于JS线程中
你必须把他查到head或者body里面,它才会生效
document.body.appendChild(div)
或者
已在页面中的元素.appendChild(div)
append
• 代码
页面中有div#test1和div#test2

    let div = document.createElement('div')
    test1.appendChild(div)
    test2.appendChild(div) 

• 请问最终div出现在哪里?
答:test2里面
一个元素不能出现在两个地方,除非复制一份。
用cloneNode(let div1 = div.cloneNode())

• 两种方法
旧方法:parentNode.removeChild(childNode)
div.parentNode.removeChild(div)
新方法:childNode.remove()(ie不支持)
div.remove()
• 思考
如果一个node被移出页面(DOM树)
那么它还可以再次回到页面中吗?
答:是可以的,只是移回了内存,还是存在的,将null赋值给div让其与内存断开连接,就会被内存回收掉

• 写标准属性
改class:div.className = 'blue'(全覆盖、class优先判定为关键字)(+=' red'一般不用)
改class:div.classList.add('red')
改style:div.style='width: 100px;color: blue';
改style的一部分:div.style.width='200px'
大小写:div.style.backgroundColor='frank'
改data-*属性:div.dataset.x = 'frank'
• 读标准属性
div.classList/a.href
div.getAttribute('class')/a.getAttribute('href')

image.png

两种方法都可以,但值可能稍微有些不同
改使事件处理函数
• div.onclick默认为null
默认点击div不会有任何事情发生
但是如果把div.onclick改为一个函数fn
那么点击div的时候,浏览器就会调用这个函数
并且使这样调用的fn.call(div, event)
div会被当做this
event则包含了点击事件的所有信息,如坐标
• div.addEventListener
是div.onclick的升级版,之后的课程单独讲
改内容
• 改文本内容
div.innerText = 'xxx'
div.textContent = 'xxx'
两者几乎没有区别
• HTML内容
div.innerHTML = '<strong>重要内容</strong>'
• 改标签
div.innerHTML = ''先清空
div.appendChild(div2)再加内容
改爸爸
• 想找一个新爸爸
newParent.appendChild(div)
直接这样就可以了,直接从原来的地方消失

• 查爸爸
node.parentNode或者node.parentElement
• 查爷爷
node.parentNode.parentNode
• 查子代
node.childNodes或者node.children(childNodes如果有空格的话会被当成文本占用长度)(都会实时更新)
注意:若用document.querySelectorAll('li')来获取元素,那么获取的元素长度不变
• 查兄弟姐妹
node.parentNode.childNodes还要排除自己和所有的文本节点
node.parentNode.children还要排除自己
• 查看老大
node.firstChild
• 查看老幺
node.lastChild
• 查看上一个哥哥/姐姐
node.previousSibling(这里面会有文本节点,node.previousElementSibling没有)
• 查看下一个弟弟/妹妹
node.nextSibling
• 遍历一个div里面的所有元素

    travel = (node, fn) => {
      fn(node)
      if(node.children){
        for(let i=0; i<node.children.length; i++){
          travel(node.children[i], fn)
        }
      }
    }
    travel(div1, (node) => console.log(node))

跨线程操作

  • 各线程各司其职
    JS引擎不能操作页面,只能操作JS。渲染引擎不能操作JS,只能操作页面。
  • 跨线程通信
    当浏览器发现JS在body里面加了个div1对象,浏览器就会通知渲染引擎在页面里也新增一个div元素,新增的div元素所有属性都照抄div1对象。
    在浏览器发现需要渲染引擎去渲染新的元素时,浏览器会通知渲染引擎,这个过程是需要消耗时间的。

插入一个新的标签的完整过程

  1. 在div1放入页面之前,对div1所有的操作都属于JS线程内的操作
  2. 把div1放入页面之时,就会通知渲染线程在页面中渲染div1对应的元素
  3. 把div1放入页面之后,对div1的操作都有可能会触发重新渲染,div1.id = 'newld'可能会重新渲染,也可能不会,div1.title = 'new'可能会重新渲染,也可能不会,如果你连续对div1多次操作,浏览器可能会合并成一次操作,也可能不会(之前在动画里提到过)

属性同步

  • 标准属性
    对div1的标准属性的修改,会被浏览器同步到页面中,比如id、class Name、title等
  • data-*属性
    同上
  • 非标准属性
    对非标准属性的修改,则只会停留在JS线程中,不会同步到页面里。
    启发:如果你有自定义属性,又想被同步到页面中,请使用data- 作为前缀作为属性名。


    非标准属性.png

    示意图.png

Property 与 Attribute 的区别

  • property属性
    JS线程中div1 对象的所有属性,叫做div1的 property
  • attribute也是属性
    渲染引擎中div1对应标签的属性,叫做attribute,或者说页面标签中的属性
  • 区别
    大部分时候,同名的property和attribute值相等,但如果不是标准属性,那么它俩只会在一开始时相等。但注意attribute只支持字符串,而property支持字符串、布尔等类型。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容

  • 今天感恩节哎,感谢一直在我身边的亲朋好友。感恩相遇!感恩不离不弃。 中午开了第一次的党会,身份的转变要...
    迷月闪星情阅读 10,548评论 0 11
  • 彩排完,天已黑
    刘凯书法阅读 4,182评论 1 3
  • 表情是什么,我认为表情就是表现出来的情绪。表情可以传达很多信息。高兴了当然就笑了,难过就哭了。两者是相互影响密不可...
    Persistenc_6aea阅读 124,053评论 2 7