javascript - 在具有任意数量子 Node 的树中,查找 Node 是否在给定 Node 的左半子树或右半子树中

标签 javascript node.js algorithm data-structures tree

问题:判断一个 Node 是在 Node root的左半子树还是右半子树中。


左半边和右半边的含义:假设一个 Node root有n个子 Node ,每个子 Node 都有子树。

如果 Node 编号为[1..(Math.floor(n/2))],则称该 Node 位于root 的左半子树中。从左到右或者它是它们的一个子树中的一个 Node ,否则它在 root 的右半子树中 [(Math.floor(n/2) + 1)..n] .

左半部分的所有 Node 都得到key小于根的值,右半部分的所有 Node 都得到 key值大于根。


想法:预处理

对树进行中序遍历,并为adjacencyLists.node中的每个 Node 分配(整数) .然后查询,比较key判断一个 Node 是在左半子树还是右半子树的值:

伪代码:

QUERY(root, queryNode)
  if (queryNode.key < root.key) {
    print "queryNode is in left half of root"
  } else {
    print "queryNode is in the right half of root"
  }

实现:

var adjacencyLists = {
  root: 'A',
  nodes: {
    A: {
      id: 'A',
      connectedNodes: ['B', 'C']
    },
    B: {
      id: 'B',
      connectedNodes: ['D', 'E']
    },
    C: {
      id: 'C',
      connectedNodes: ['F', 'G', 'H', 'Q', 'R']
    },
    D: {
      id: 'D',
      connectedNodes: []
    },
    E: {
      id: 'E',
      connectedNodes: ['K']
    },
    F: {
      id: 'F',
      connectedNodes: ['I']
    },
    G: {
      id: 'G',
      connectedNodes: ['J', 'L', 'N', 'P']
    },
    H: {
      id: 'H',
      connectedNodes: ['M', 'O']
    },
    K: {
      id: 'K',
      connectedNodes: []
    },
    I: {
      id: 'I',
      connectedNodes: []
    },
    J: {
      id: 'J',
      connectedNodes: []
    },
    L: {
      id: 'L',
      connectedNodes: []
    },
    M: {
      id: 'M',
      connectedNodes: []
    },
    N: {
      id: 'N',
      connectedNodes: []
    },
    O: {
      id: 'O',
      connectedNodes: []
    },
    P: {
      id: 'P',
      connectedNodes: []
    },
    Q: {
      id: 'Q',
      connectedNodes: []
    },
    R: {
      id: 'R',
      connectedNodes: []
    },
  }
}

var keyLookup = {};
var count = 0;

function inorderTraversalNumberingOfNodes(cur) {
  if (adjacencyLists.nodes[cur].connectedNodes.length) {
    // recurse left half subtrees
    for (let i = 0; i < Math.ceil(adjacencyLists.nodes[cur].connectedNodes.length / 2); i++) {
      inorderTraversalNumberingOfNodes(adjacencyLists.nodes[cur].connectedNodes[i]);
    }
    // recurse right half subtrees
    for (let i = Math.ceil(adjacencyLists.nodes[cur].connectedNodes.length / 2); i < adjacencyLists.nodes[cur].connectedNodes.length; i++) {
      inorderTraversalNumberingOfNodes(adjacencyLists.nodes[cur].connectedNodes[i]);
    }
    count++;
    keyLookup[cur] = { key: count };
  } else {
    count++;
    keyLookup[cur] = {key : count };
  }
}

inorderTraversalNumberingOfNodes(adjacencyLists.root);
console.log(keyLookup)

// query to determine whether a node is in the left half or right half of root

function query(rootNodeId, queryNodeId) {
  if (keyLookup[queryNodeId].key < keyLookup[rootNodeId].key) {
    console.log(`query node ${queryNodeId} is in the left half of root node ${rootNodeId}`);
  } else {
    console.log(`query node ${queryNodeId} is in the right half of root node ${rootNodeId}`);
  }
}

query('A', 'D');
query('M', 'C');


Node 的预期键值:邻接表的中序遍历应该为 Node 分配以下键:

{
  "D": {
    "key": 1
  },
  "K": {
    "key": 3
  },
  "E": {
    "key": 4
  },
  "B": {
    "key": 2
  },
  "I": {
    "key": 6
  },
  "F": {
    "key": 7
  },
  "J": {
    "key": 8
  },
  "L": {
    "key": 9
  },
  "N": {
    "key": 11
  },
  "P": {
    "key": 12
  },
  "G": {
    "key": 10
  },
  "M": {
    "key": 14
  },
  "O": {
    "key": 16
  },
  "H": {
    "key": 15
  },
  "Q": {
    "key": 17
  },
  "R": {
    "key": 18
  },
  "C": {
    "key": 13
  },
  "A": {
    "key": 5
  }
}

现在,在 QUERY(A, D) ,输出应该是queryNode is in left half of root , 自 1 < 5 .

我没有得到预期的答案,因为我无法分配正确的 key到 Node 。

最佳答案

可以先获取顺序,再取索引值获取边。

var adjacencyLists = { root: 'A', nodes: { A: { id: 'A', connectedNodes: ['B', 'C'] }, B: { id: 'B', connectedNodes: ['D', 'E'] }, C: { id: 'C', connectedNodes: ['F', 'G', 'H', 'Q', 'R'] }, D: { id: 'D', connectedNodes: [] }, E: { id: 'E', connectedNodes: ['K'] }, F: { id: 'F', connectedNodes: ['I'] }, G: { id: 'G', connectedNodes: ['J', 'L', 'N', 'P'] }, H: { id: 'H', connectedNodes: ['M', 'O'] }, K: { id: 'K', connectedNodes: [] }, I: { id: 'I', connectedNodes: [] }, J: { id: 'J', connectedNodes: [] }, L: { id: 'L', connectedNodes: [] }, M: { id: 'M', connectedNodes: [] }, N: { id: 'N', connectedNodes: [] }, O: { id: 'O', connectedNodes: [] }, P: { id: 'P', connectedNodes: [] }, Q: { id: 'Q', connectedNodes: [] }, R: { id: 'R', connectedNodes: [] } } },
    index = 0,
    getOrder = parent => value => {
        if (!adjacencyLists.nodes[value].connectedNodes.length) {
            adjacencyLists.nodes[value].index = adjacencyLists.nodes[value].index || ++index;
        }
        adjacencyLists.nodes[value].connectedNodes.forEach(getOrder(value));
        adjacencyLists.nodes[parent].index = adjacencyLists.nodes[parent].index || ++index;
    },
    query = (a, b) => a === b
        ? 'middle'
        : adjacencyLists.nodes[a].index < adjacencyLists.nodes[b].index
            ? 'left'
            : 'right';

getOrder('A')('A');

console.log(query('A', 'D')); // or vice versa ...?
console.log(adjacencyLists);
.as-console-wrapper { max-height: 100% !important; top: 0; }

关于javascript - 在具有任意数量子 Node 的树中,查找 Node 是否在给定 Node 的左半子树或右半子树中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59577515/

相关文章:

javascript - Twitter Bootstrap 附加奇怪的行为

javascript - 组件名称需要以大写字母开头?

node.js - express + typescript : Property 'body' does not exist on type 'Request'

javascript - 如何在 NodeJS 上使用 nanoid 模块?

javascript - CkEditor 无法设置未定义的属性 'dir'

javascript - 多个脚本不起作用

javascript - 如何在 javascript 中为对象实现 onclick?

c++ - 以最少的移动交换盒子

c - 什么时候使用 CORDIC 或多项式近似更有效?

algorithm - 理解归并排序的递归