Ruby's Louvre

每天学习一点点算法

导航

< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

统计

节点排序

为了让自定义选择选择出的节点集合尽可能接近原生API选出的结果,我们往往要对结果集进行排序,此顺序当然是从上到下,从左到右的DOM树顺序。在IE中我们可以利用sourceIndex,标准浏览器我们可以用compareDocumentPosition,但对于旧一点的标准浏览器呢?XML呢?因此我们就需要根据一个节点的属性确定它与另一个节点的关系了。

我的思路很简单,如果它们相同,返回0(用于去重),如果它们的父节点相同,那么根据nextSibling确定两者的先后顺序,否则就找到其最近公共祖先与其他两个最接近这祖先的两个父节点(人性点说,是伯父与父亲),这时不就是与父节点相同的情况吗?!根据nextSibling确定它们的顺序,它们的顺序就是它们的孩子的顺序(因此有一个叫李刚的爹很重要,在这个世袭制的世界上!)不过,有时最近公共祖先就是比较双方的某一个呢,那当然是它最近了。

剩下的问题就是求最近公共祖先的问题了。我的思路也很简单,不一定高效,毕竟大学把数学都荒废了。不断向上取得它们的父节点,直到最顶的HTML元素,连同最初那个节点,组成两个数组。然后每次取数组最后的元素进行比较,如果相同就去掉它们,因为相同的都是公共祖先,不相同就往上取其中一方就行了。

下面是测试页面与源码:

<!doctype html>
<html>
  <head>
    <title>节点排序 </title>
 
    <script>
      window.onload = function(){
        function shuffle(a) {
          var array = a.concat();
  
          var i = array.length;
          while (i) {
            var j = Math.floor(Math.random()*i);
            var t = array[--i];
            array[i] = array[j];
            array[j] = t;
          }
 
          return array;
        }
        var log =function(s){
          window.console && window.console.log(s)
        }
        var sliceNodes = function(arr){
          var ret = [], i = arr.length;
          while (i) ret[--i] = arr[i];
          return ret;
        }
 
        var sortNodes = function(a,b){
         var p = "parentNode",ap = a[p],bp = b[p];
          if(a === b){
             return 0
          }else if(ap === bp){
            while(a = a.nextSibling){
              if(a === b){
                return -1
              }
            }
            return 1
          }else if(!ap){
            return -1
          }else if(!bp){
            return 1
         }
          var al = [], ap = a
          while(ap && ap.nodeType === 1){
            al[al.length] = ap
            ap = ap[p]
          }
          var bl = [],bp = b;
          while(bp && bp.nodeType === 1){
            bl[bl.length] = bp
            bp = bp[p]
          }
          ap = al.pop();
          bp = bl.pop();
          while(ap === bp){
            ap = al.pop();
            bp = bl.pop();
          }
          if(ap && bp){
            while(ap = ap.nextSibling){
              if(ap === bp){
                return -1
              }
            }
            return 1
          }
          return ap ? 1 : -1
        }
 
        var els = document.getElementsByTagName("div")
        els = sliceNodes(els);//转换成纯数组
        log(els);
        els = shuffle(els);//洗牌(模似用自定义的选择器取得的节点集合情况)
        log(els);
        els = els.sort(sortNodes)
        log(els)
      }
 
    </script>
  </head>
 
  <body>
    <form>
      <div id="aa1" >
        <div id="aa2">111</div>
        <div id="aa3">222<div>00</div></div>
        <div id="aa4">44<div id="aa5">111</div></div>
      </div>
      <div id="aa6">22</div>
      <div id="aa7"><p>Hello</p></div>
      <div id="aa8">
        <div id="aa9">888</div>
        <div id="aa10">999</div>
      </div>
    </form>
  </body>
</html>

由于使用了window.console,因此建议在firefox,IE8,chrome下查看结果。

如果您觉得此文有帮助,可以打赏点钱给我支付宝1669866773@qq.com ,或扫描二维码

posted on   司徒正美  阅读(2767)  评论(4编辑  收藏  举报

编辑推荐:
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
阅读排行:
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· .NET周刊【3月第1期 2025-03-02】
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· [AI/GPT/综述] AI Agent的设计模式综述
历史上的今天:
2010-01-28 Dmitry Baranovskiy的javascript谜题
2010-01-28 全世界最短的IE判定
点击右上角即可分享
微信分享提示