मराठी

बायनरी सर्च ट्री (BST) च्या मूलभूत तत्त्वांचा शोध घ्या आणि जावास्क्रिप्टमध्ये त्यांची कार्यक्षमतेने अंमलबजावणी कशी करावी हे शिका. हे मार्गदर्शक जगभरातील डेव्हलपर्ससाठी BST रचना, ऑपरेशन्स आणि व्यावहारिक उदाहरणे समाविष्ट करते.

बायनरी सर्च ट्रीज: जावास्क्रिप्टमधील एक सर्वसमावेशक अंमलबजावणी मार्गदर्शक

बायनरी सर्च ट्री (BSTs) ही कॉम्प्युटर सायन्समध्ये एक मूलभूत डेटा स्ट्रक्चर आहे, जी डेटा कार्यक्षमतेने शोधण्यासाठी, क्रमवारी लावण्यासाठी आणि परत मिळवण्यासाठी मोठ्या प्रमाणावर वापरली जाते. त्यांची श्रेणीबद्ध रचना अनेक ऑपरेशन्समध्ये लॉगरिदमिक टाइम कॉम्प्लेक्सिटीला परवानगी देते, ज्यामुळे ते मोठ्या डेटासेट व्यवस्थापित करण्यासाठी एक शक्तिशाली साधन बनते. हे मार्गदर्शक BSTs चे सर्वसमावेशक विहंगावलोकन प्रदान करते आणि जगभरातील डेव्हलपर्ससाठी जावास्क्रिप्टमध्ये त्यांची अंमलबजावणी दर्शवते.

बायनरी सर्च ट्री समजून घेणे

बायनरी सर्च ट्री म्हणजे काय?

बायनरी सर्च ट्री ही एक ट्री-आधारित डेटा स्ट्रक्चर आहे जिथे प्रत्येक नोडला जास्तीत जास्त दोन चिल्ड्रेन (children) असतात, ज्यांना लेफ्ट चाइल्ड आणि राईट चाइल्ड म्हटले जाते. BST चा मुख्य गुणधर्म असा आहे की कोणत्याही दिलेल्या नोडसाठी:

हा गुणधर्म सुनिश्चित करतो की BST मधील घटक नेहमी क्रमवारीत असतात, ज्यामुळे कार्यक्षम शोध आणि पुनर्प्राप्ती शक्य होते.

मुख्य संकल्पना

जावास्क्रिप्टमध्ये बायनरी सर्च ट्रीची अंमलबजावणी

नोड क्लासची व्याख्या करणे

प्रथम, आपण BST मधील प्रत्येक नोड दर्शवण्यासाठी एक `Node` क्लास परिभाषित करतो. प्रत्येक नोडमध्ये डेटा संग्रहित करण्यासाठी एक `key` आणि त्याच्या चिल्ड्रेनसाठी `left` आणि `right` पॉइंटर्स असतील.


class Node {
  constructor(key) {
    this.key = key;
    this.left = null;
    this.right = null;
  }
}

बायनरी सर्च ट्री क्लासची व्याख्या करणे

पुढे, आपण `BinarySearchTree` क्लास परिभाषित करतो. या क्लासमध्ये रूट नोड आणि ट्री मध्ये इन्सर्ट, सर्च, डिलीट आणि ट्रॅव्हर्स करण्यासाठी मेथड्स असतील.


class BinarySearchTree {
  constructor() {
    this.root = null;
  }

  // मेथड्स येथे जोडल्या जातील
}

इन्सर्शन (Insertion)

`insert` मेथड दिलेल्या की सह एक नवीन नोड BST मध्ये जोडते. इन्सर्शन प्रक्रिया नवीन नोडला विद्यमान नोड्सच्या तुलनेत योग्य स्थितीत ठेवून BST गुणधर्म राखते.


insert(key) {
  const newNode = new Node(key);

  if (this.root === null) {
    this.root = newNode;
  } else {
    this.insertNode(this.root, newNode);
  }
}

insertNode(node, newNode) {
  if (newNode.key < node.key) {
    if (node.left === null) {
      node.left = newNode;
    } else {
      this.insertNode(node.left, newNode);
    }
  } else {
    if (node.right === null) {
      node.right = newNode;
    } else {
      this.insertNode(node.right, newNode);
    }
  }
}

उदाहरण: BST मध्ये व्हॅल्यूज इन्सर्ट करणे


const bst = new BinarySearchTree();
bst.insert(11);
bst.insert(7);
bst.insert(15);
bst.insert(5);
bst.insert(3);
bst.insert(9);
bst.insert(8);
bst.insert(10);
bst.insert(13);
bst.insert(12);
bst.insert(14);
bst.insert(20);
bst.insert(18);
bst.insert(25);

सर्चिंग (Searching)

`search` मेथड तपासते की दिलेल्या की सह एक नोड BST मध्ये अस्तित्वात आहे की नाही. हे कीची तुलना वर्तमान नोडच्या कीशी करून आणि त्यानुसार डाव्या किंवा उजव्या सबट्रीकडे जाऊन ट्री ट्रॅव्हर्स करते.


search(key) {
  return this.searchNode(this.root, key);
}

searchNode(node, key) {
  if (node === null) {
    return false;
  }

  if (key < node.key) {
    return this.searchNode(node.left, key);
  } else if (key > node.key) {
    return this.searchNode(node.right, key);
  } else {
    return true;
  }
}

उदाहरण: BST मध्ये व्हॅल्यू शोधणे


console.log(bst.search(9));  // आउटपुट: true
console.log(bst.search(2));  // आउटपुट: false

डिलीशन (Deletion)

`remove` मेथड दिलेल्या की सह एक नोड BST मधून डिलीट करते. हे सर्वात गुंतागुंतीचे ऑपरेशन आहे कारण नोड काढून टाकताना BST गुणधर्म राखणे आवश्यक आहे. विचारात घेण्यासाठी तीन प्रकरणे आहेत:


remove(key) {
  this.root = this.removeNode(this.root, key);
}

removeNode(node, key) {
  if (node === null) {
    return null;
  }

  if (key < node.key) {
    node.left = this.removeNode(node.left, key);
    return node;
  } else if (key > node.key) {
    node.right = this.removeNode(node.right, key);
    return node;
  } else {
    // की ही node.key च्या बरोबर आहे

    // प्रकरण १ - एक लीफ नोड
    if (node.left === null && node.right === null) {
      node = null;
      return node;
    }

    // प्रकरण २ - नोडला फक्त १ चाइल्ड आहे
    if (node.left === null) {
      node = node.right;
      return node;
    } else if (node.right === null) {
      node = node.left;
      return node;
    }

    // प्रकरण ३ - नोडला २ चिल्ड्रेन आहेत
    const aux = this.findMinNode(node.right);
    node.key = aux.key;
    node.right = this.removeNode(node.right, aux.key);
    return node;
  }
}

findMinNode(node) {
  let current = node;
  while (current != null && current.left != null) {
    current = current.left;
  }
  return current;
}

उदाहरण: BST मधून व्हॅल्यू काढून टाकणे


bst.remove(7);
console.log(bst.search(7)); // आउटपुट: false

ट्री ट्रॅव्हर्सल (Tree Traversal)

ट्री ट्रॅव्हर्सलमध्ये ट्री मधील प्रत्येक नोडला एका विशिष्ट क्रमाने भेट देणे समाविष्ट आहे. अनेक सामान्य ट्रॅव्हर्सल पद्धती आहेत:


inOrderTraverse(callback) {
  this.inOrderTraverseNode(this.root, callback);
}

inOrderTraverseNode(node, callback) {
  if (node !== null) {
    this.inOrderTraverseNode(node.left, callback);
    callback(node.key);
    this.inOrderTraverseNode(node.right, callback);
  }
}

preOrderTraverse(callback) {
  this.preOrderTraverseNode(this.root, callback);
}

preOrderTraverseNode(node, callback) {
  if (node !== null) {
    callback(node.key);
    this.preOrderTraverseNode(node.left, callback);
    this.preOrderTraverseNode(node.right, callback);
  }
}

postOrderTraverse(callback) {
  this.postOrderTraverseNode(this.root, callback);
}

postOrderTraverseNode(node, callback) {
  if (node !== null) {
    this.postOrderTraverseNode(node.left, callback);
    this.postOrderTraverseNode(node.right, callback);
    callback(node.key);
  }
}

उदाहरण: BST ट्रॅव्हर्स करणे


const printNode = (value) => console.log(value);

bst.inOrderTraverse(printNode);   // आउटपुट: 3 5 8 9 10 11 12 13 14 15 18 20 25
bst.preOrderTraverse(printNode);  // आउटपुट: 11 5 3 9 8 10 15 13 12 14 20 18 25
bst.postOrderTraverse(printNode); // आउटपुट: 3 8 10 9 12 14 13 18 25 20 15 11

किमान आणि कमाल व्हॅल्यूज

BST मधील किमान आणि कमाल व्हॅल्यूज शोधणे त्याच्या क्रमबद्ध स्वरूपामुळे सरळ आहे.


min() {
  return this.minNode(this.root);
}

minNode(node) {
  let current = node;
  while (current !== null && current.left !== null) {
    current = current.left;
  }
  return current;
}

max() {
  return this.maxNode(this.root);
}

maxNode(node) {
  let current = node;
  while (current !== null && current.right !== null) {
    current = current.right;
  }
  return current;
}

उदाहरण: किमान आणि कमाल व्हॅल्यूज शोधणे


console.log(bst.min().key); // आउटपुट: 3
console.log(bst.max().key); // आउटपुट: 25

बायनरी सर्च ट्रीचे व्यावहारिक उपयोग

बायनरी सर्च ट्री विविध अनुप्रयोगांमध्ये वापरले जातात, ज्यात समाविष्ट आहे:

कार्यप्रदर्शन विचार

BST चे कार्यप्रदर्शन त्याच्या रचनेवर अवलंबून असते. सर्वोत्तम परिस्थितीत, एक संतुलित BST इन्सर्शन, सर्च आणि डिलीशन ऑपरेशन्ससाठी लॉगरिदमिक टाइम कॉम्प्लेक्सिटीला परवानगी देते. तथापि, सर्वात वाईट परिस्थितीत (उदा. एक स्क्यूड ट्री), टाइम कॉम्प्लेक्सिटी लीनियर टाइमपर्यंत खालावू शकते.

संतुलित विरुद्ध असंतुलित ट्रीज

संतुलित BST म्हणजे जिथे प्रत्येक नोडच्या डाव्या आणि उजव्या सबट्रीजच्या उंचीमध्ये जास्तीत जास्त एकाचा फरक असतो. सेल्फ-बॅलेंसिंग अल्गोरिदम, जसे की AVL ट्री आणि रेड-ब्लॅक ट्री, हे सुनिश्चित करतात की ट्री संतुलित राहते, ज्यामुळे सातत्यपूर्ण कार्यप्रदर्शन मिळते. सर्व्हरवरील लोडवर आधारित वेगवेगळ्या प्रदेशांना वेगवेगळ्या ऑप्टिमायझेशन स्तरांची आवश्यकता असू शकते; संतुलन उच्च जागतिक वापराखाली कार्यप्रदर्शन टिकवून ठेवण्यास मदत करते.

टाइम कॉम्प्लेक्सिटी (Time Complexity)

प्रगत BST संकल्पना

सेल्फ-बॅलेंसिंग ट्रीज

सेल्फ-बॅलेंसिंग ट्रीज हे BSTs आहेत जे संतुलन राखण्यासाठी आपोआप आपली रचना समायोजित करतात. हे सुनिश्चित करते की ट्रीची उंची लॉगरिदमिक राहते, ज्यामुळे सर्व ऑपरेशन्ससाठी सातत्यपूर्ण कार्यप्रदर्शन मिळते. सामान्य सेल्फ-बॅलेंसिंग ट्रीमध्ये AVL ट्री आणि रेड-ब्लॅक ट्री यांचा समावेश आहे.

AVL ट्रीज

AVL ट्री कोणत्याही नोडच्या डाव्या आणि उजव्या सबट्रीजमधील उंचीचा फरक जास्तीत जास्त एक असल्याची खात्री करून संतुलन राखतात. जेव्हा हे संतुलन बिघडते, तेव्हा संतुलन पुनर्संचयित करण्यासाठी रोटेशन्स केले जातात.

रेड-ब्लॅक ट्रीज

रेड-ब्लॅक ट्री संतुलन राखण्यासाठी रंगाचे गुणधर्म (लाल किंवा काळा) वापरतात. ते AVL ट्रीपेक्षा अधिक गुंतागुंतीचे आहेत परंतु काही विशिष्ट परिस्थितीत चांगले कार्यप्रदर्शन देतात.

जावास्क्रिप्ट कोड उदाहरण: संपूर्ण बायनरी सर्च ट्री अंमलबजावणी


class Node {
  constructor(key) {
    this.key = key;
    this.left = null;
    this.right = null;
  }
}

class BinarySearchTree {
  constructor() {
    this.root = null;
  }

  insert(key) {
    const newNode = new Node(key);

    if (this.root === null) {
      this.root = newNode;
    } else {
      this.insertNode(this.root, newNode);
    }
  }

  insertNode(node, newNode) {
    if (newNode.key < node.key) {
      if (node.left === null) {
        node.left = newNode;
      } else {
        this.insertNode(node.left, newNode);
      }
    } else {
      if (node.right === null) {
        node.right = newNode;
      } else {
        this.insertNode(node.right, newNode);
      }
    }
  }

  search(key) {
    return this.searchNode(this.root, key);
  }

  searchNode(node, key) {
    if (node === null) {
      return false;
    }

    if (key < node.key) {
      return this.searchNode(node.left, key);
    } else if (key > node.key) {
      return this.searchNode(node.right, key);
    } else {
      return true;
    }
  }

  remove(key) {
    this.root = this.removeNode(this.root, key);
  }

  removeNode(node, key) {
    if (node === null) {
      return null;
    }

    if (key < node.key) {
      node.left = this.removeNode(node.left, key);
      return node;
    } else if (key > node.key) {
      node.right = this.removeNode(node.right, key);
      return node;
    } else {
      // की ही node.key च्या बरोबर आहे

      // प्रकरण १ - एक लीफ नोड
      if (node.left === null && node.right === null) {
        node = null;
        return node;
      }

      // प्रकरण २ - नोडला फक्त १ चाइल्ड आहे
      if (node.left === null) {
        node = node.right;
        return node;
      } else if (node.right === null) {
        node = node.left;
        return node;
      }

      // प्रकरण ३ - नोडला २ चिल्ड्रेन आहेत
      const aux = this.findMinNode(node.right);
      node.key = aux.key;
      node.right = this.removeNode(node.right, aux.key);
      return node;
    }
  }

  findMinNode(node) {
    let current = node;
    while (current != null && current.left != null) {
      current = current.left;
    }
    return current;
  }

  min() {
    return this.minNode(this.root);
  }

  minNode(node) {
    let current = node;
    while (current !== null && current.left !== null) {
      current = current.left;
    }
    return current;
  }

  max() {
    return this.maxNode(this.root);
  }

  maxNode(node) {
    let current = node;
    while (current !== null && current.right !== null) {
      current = current.right;
    }
    return current;
  }

  inOrderTraverse(callback) {
    this.inOrderTraverseNode(this.root, callback);
  }

  inOrderTraverseNode(node, callback) {
    if (node !== null) {
      this.inOrderTraverseNode(node.left, callback);
      callback(node.key);
      this.inOrderTraverseNode(node.right, callback);
    }
  }

  preOrderTraverse(callback) {
    this.preOrderTraverseNode(this.root, callback);
  }

  preOrderTraverseNode(node, callback) {
    if (node !== null) {
      callback(node.key);
      this.preOrderTraverseNode(node.left, callback);
      this.preOrderTraverseNode(node.right, callback);
    }
  }

  postOrderTraverse(callback) {
    this.postOrderTraverseNode(this.root, callback);
  }

  postOrderTraverseNode(node, callback) {
    if (node !== null) {
      this.postOrderTraverseNode(node.left, callback);
      this.postOrderTraverseNode(node.right, callback);
      callback(node.key);
    }
  }
}

// उदाहरण वापर
const bst = new BinarySearchTree();
bst.insert(11);
bst.insert(7);
bst.insert(15);
bst.insert(5);
bst.insert(3);
bst.insert(9);
bst.insert(8);
bst.insert(10);
bst.insert(13);
bst.insert(12);
bst.insert(14);
bst.insert(20);
bst.insert(18);
bst.insert(25);

const printNode = (value) => console.log(value);

console.log("इन-ऑर्डर ट्रॅव्हर्सल:");
bst.inOrderTraverse(printNode);

console.log("प्री-ऑर्डर ट्रॅव्हर्सल:");
bst.preOrderTraverse(printNode);

console.log("पोस्ट-ऑर्डर ट्रॅव्हर्सल:");
bst.postOrderTraverse(printNode);

console.log("किमान व्हॅल्यू:", bst.min().key);
console.log("कमाल व्हॅल्यू:", bst.max().key);

console.log("9 साठी शोधा:", bst.search(9));
console.log("2 साठी शोधा:", bst.search(2));

bst.remove(7);
console.log("काढून टाकल्यानंतर 7 साठी शोधा:", bst.search(7));

निष्कर्ष

बायनरी सर्च ट्री ही एक शक्तिशाली आणि बहुमुखी डेटा स्ट्रक्चर आहे ज्याचे असंख्य उपयोग आहेत. या मार्गदर्शकाने BSTs चे सर्वसमावेशक विहंगावलोकन प्रदान केले आहे, ज्यात त्यांची रचना, ऑपरेशन्स आणि जावास्क्रिप्टमधील अंमलबजावणी समाविष्ट आहे. या मार्गदर्शकात चर्चा केलेल्या तत्त्वे आणि तंत्रे समजून घेऊन, जगभरातील डेव्हलपर्स सॉफ्टवेअर डेव्हलपमेंटमधील विविध समस्या सोडवण्यासाठी BSTs चा प्रभावीपणे वापर करू शकतात. जागतिक डेटाबेस व्यवस्थापित करण्यापासून ते शोध अल्गोरिदम ऑप्टिमाइझ करण्यापर्यंत, BSTs चे ज्ञान कोणत्याही प्रोग्रामरसाठी एक अनमोल संपत्ती आहे.

तुम्ही कॉम्प्युटर सायन्समध्ये आपला प्रवास सुरू ठेवत असताना, सेल्फ-बॅलेंसिंग ट्री आणि त्यांच्या विविध अंमलबजावणीसारख्या प्रगत संकल्पनांचा शोध घेतल्यास तुमची समज आणि क्षमता आणखी वाढेल. बायनरी सर्च ट्री प्रभावीपणे वापरण्याची कला आत्मसात करण्यासाठी वेगवेगळ्या परिस्थितींसह सराव आणि प्रयोग करत रहा.