From ff02d1069b7ff0a753a9dec66363a6813f44e51b Mon Sep 17 00:00:00 2001 From: Kishan Ved Date: Fri, 26 Jul 2024 18:49:11 +0530 Subject: [PATCH] C++ backend for all trees in `binary_trees.py` file complete (#569) --- AUTHORS | 1 + pydatastructs/trees/_backend/cpp/AVLTree.hpp | 16 +++++++++++++--- .../trees/_backend/cpp/BinarySearchTree.hpp | 12 +++++++++++- .../trees/_backend/cpp/RedBlackTree.hpp | 11 ++++++++++- .../_backend/cpp/SelfBalancingBinaryTree.hpp | 12 +++++++++++- pydatastructs/trees/_backend/cpp/SplayTree.hpp | 11 ++++++++++- pydatastructs/trees/_backend/cpp/Treap.hpp | 11 ++++++++++- pydatastructs/trees/tests/test_binary_trees.py | 7 ++++++- 8 files changed, 72 insertions(+), 9 deletions(-) diff --git a/AUTHORS b/AUTHORS index 478ec0785..943bf804d 100644 --- a/AUTHORS +++ b/AUTHORS @@ -10,3 +10,4 @@ Harsheet Pratik Goyal Jay Thorat Rajveer Singh Bharadwaj +Kishan Ved diff --git a/pydatastructs/trees/_backend/cpp/AVLTree.hpp b/pydatastructs/trees/_backend/cpp/AVLTree.hpp index 2183a63b2..f0de0b091 100644 --- a/pydatastructs/trees/_backend/cpp/AVLTree.hpp +++ b/pydatastructs/trees/_backend/cpp/AVLTree.hpp @@ -57,7 +57,7 @@ static long AVLTree_left_height(AVLTree* self, PyObject *args) { BinaryTree* bt = self->sbbt->bst->binary_tree; return reinterpret_cast(bt->tree->_one_dimensional_array->_data[PyLong_AsLong(node->left)])->height; } - else{ + else { return (-1); } } @@ -68,7 +68,7 @@ static long AVLTree_right_height(AVLTree* self, PyObject *args) { BinaryTree* bt = self->sbbt->bst->binary_tree; return reinterpret_cast(bt->tree->_one_dimensional_array->_data[PyLong_AsLong(node->right)])->height; } - else{ + else { return -1; } } @@ -306,6 +306,11 @@ static PyObject* AVLTree_delete(AVLTree* self, PyObject *args, PyObject *kwds) { Py_RETURN_TRUE; } +static PyObject* AVLTree_root_idx(AVLTree *self, void *closure) { + return self->sbbt->bst->binary_tree->root_idx; +} + + static struct PyMethodDef AVLTree_PyMethodDef[] = { {"search", (PyCFunction) AVLTree_search, METH_VARARGS | METH_KEYWORDS, NULL}, {"insert", (PyCFunction) AVLTree_insert, METH_VARARGS, NULL}, @@ -321,6 +326,11 @@ static struct PyMethodDef AVLTree_PyMethodDef[] = { {NULL} }; +static PyGetSetDef AVLTree_GetterSetters[] = { + {"root_idx", (getter) AVLTree_root_idx, NULL, "returns the index of the tree's root", NULL}, + {NULL} /* Sentinel */ +}; + static PyMemberDef AVLTree_PyMemberDef[] = { {"tree", T_OBJECT_EX, offsetof(AVLTree, tree), 0, "tree"}, {NULL} /* Sentinel */ @@ -356,7 +366,7 @@ static PyTypeObject AVLTreeType = { /* tp_iternext */ 0, /* tp_methods */ AVLTree_PyMethodDef, /* tp_members */ AVLTree_PyMemberDef, - /* tp_getset */ 0, + /* tp_getset */ AVLTree_GetterSetters, /* tp_base */ &SelfBalancingBinaryTreeType, /* tp_dict */ 0, /* tp_descr_get */ 0, diff --git a/pydatastructs/trees/_backend/cpp/BinarySearchTree.hpp b/pydatastructs/trees/_backend/cpp/BinarySearchTree.hpp index 352d91ce6..557384723 100644 --- a/pydatastructs/trees/_backend/cpp/BinarySearchTree.hpp +++ b/pydatastructs/trees/_backend/cpp/BinarySearchTree.hpp @@ -659,6 +659,11 @@ static PyObject* BinarySearchTree_select(BinarySearchTree* self, PyObject* args) Py_RETURN_NONE; // dummy return statement, never executed } +static PyObject* BinarySearchTree_root_idx(BinarySearchTree *self, void *closure) { + return self->binary_tree->root_idx; +} + + static struct PyMethodDef BinarySearchTree_PyMethodDef[] = { {"insert", (PyCFunction) BinarySearchTree_insert, METH_VARARGS | METH_KEYWORDS, NULL}, {"delete", (PyCFunction) BinarySearchTree_delete, METH_VARARGS | METH_KEYWORDS, NULL}, @@ -674,6 +679,11 @@ static struct PyMethodDef BinarySearchTree_PyMethodDef[] = { {NULL} }; +static PyGetSetDef BinarySearchTree_GetterSetters[] = { + {"root_idx", (getter) BinarySearchTree_root_idx, NULL, "returns the index of the tree's root", NULL}, + {NULL} /* Sentinel */ +}; + static PyMemberDef BinarySearchTree_PyMemberDef[] = { {"tree", T_OBJECT_EX, offsetof(BinarySearchTree, tree), 0, "tree"}, {NULL} /* Sentinel */ @@ -709,7 +719,7 @@ static PyTypeObject BinarySearchTreeType = { /* tp_iternext */ 0, /* tp_methods */ BinarySearchTree_PyMethodDef, /* tp_members */ BinarySearchTree_PyMemberDef, - /* tp_getset */ 0, + /* tp_getset */ BinarySearchTree_GetterSetters, /* tp_base */ &BinaryTreeType, /* tp_dict */ 0, /* tp_descr_get */ 0, diff --git a/pydatastructs/trees/_backend/cpp/RedBlackTree.hpp b/pydatastructs/trees/_backend/cpp/RedBlackTree.hpp index cfbebc438..360403141 100644 --- a/pydatastructs/trees/_backend/cpp/RedBlackTree.hpp +++ b/pydatastructs/trees/_backend/cpp/RedBlackTree.hpp @@ -578,6 +578,10 @@ static PyObject* RedBlackTree_search(RedBlackTree* self, PyObject *args, PyObjec return BinarySearchTree_search(self->sbbt->bst, args, kwds); } +static PyObject* RedBlackTree_root_idx(RedBlackTree *self, void *closure) { + return self->sbbt->bst->binary_tree->root_idx; +} + static struct PyMethodDef RedBlackTree_PyMethodDef[] = { {"insert", (PyCFunction) RedBlackTree_insert, METH_VARARGS, NULL}, @@ -592,6 +596,11 @@ static struct PyMethodDef RedBlackTree_PyMethodDef[] = { {NULL} }; +static PyGetSetDef RedBlackTree_GetterSetters[] = { + {"root_idx", (getter) RedBlackTree_root_idx, NULL, "returns the index of the tree's root", NULL}, + {NULL} /* Sentinel */ +}; + static PyMemberDef RedBlackTree_PyMemberDef[] = { {"tree", T_OBJECT_EX, offsetof(RedBlackTree, tree), 0, "tree"}, {NULL} /* Sentinel */ @@ -627,7 +636,7 @@ static PyTypeObject RedBlackTreeType = { /* tp_iternext */ 0, /* tp_methods */ RedBlackTree_PyMethodDef, /* tp_members */ RedBlackTree_PyMemberDef, - /* tp_getset */ 0, + /* tp_getset */ RedBlackTree_GetterSetters, /* tp_base */ &SelfBalancingBinaryTreeType, /* tp_dict */ 0, /* tp_descr_get */ 0, diff --git a/pydatastructs/trees/_backend/cpp/SelfBalancingBinaryTree.hpp b/pydatastructs/trees/_backend/cpp/SelfBalancingBinaryTree.hpp index 7e91d1a52..2f7fbf356 100644 --- a/pydatastructs/trees/_backend/cpp/SelfBalancingBinaryTree.hpp +++ b/pydatastructs/trees/_backend/cpp/SelfBalancingBinaryTree.hpp @@ -219,6 +219,11 @@ static PyObject* SelfBalancingBinaryTree__right_left_rotate(SelfBalancingBinaryT Py_RETURN_NONE; } +static PyObject* SelfBalancingBinaryTree_root_idx(SelfBalancingBinaryTree *self, void *closure) { + return self->bst->binary_tree->root_idx; +} + + static struct PyMethodDef SelfBalancingBinaryTree_PyMethodDef[] = { {"insert", (PyCFunction) SelfBalancingBinaryTree_insert, METH_VARARGS | METH_KEYWORDS, NULL}, {"delete", (PyCFunction) SelfBalancingBinaryTree_delete, METH_VARARGS | METH_KEYWORDS, NULL}, @@ -238,6 +243,11 @@ static struct PyMethodDef SelfBalancingBinaryTree_PyMethodDef[] = { {NULL} }; +static PyGetSetDef SelfBalancingBinaryTree_GetterSetters[] = { + {"root_idx", (getter) SelfBalancingBinaryTree_root_idx, NULL, "returns the index of the tree's root", NULL}, + {NULL} /* Sentinel */ +}; + static PyMemberDef SelfBalancingBinaryTree_PyMemberDef[] = { {"tree", T_OBJECT_EX, offsetof(SelfBalancingBinaryTree, tree), 0, "tree"}, {NULL} /* Sentinel */ @@ -273,7 +283,7 @@ static PyTypeObject SelfBalancingBinaryTreeType = { /* tp_iternext */ 0, /* tp_methods */ SelfBalancingBinaryTree_PyMethodDef, /* tp_members */ SelfBalancingBinaryTree_PyMemberDef, - /* tp_getset */ 0, + /* tp_getset */ SelfBalancingBinaryTree_GetterSetters, /* tp_base */ &BinarySearchTreeType, /* tp_dict */ 0, /* tp_descr_get */ 0, diff --git a/pydatastructs/trees/_backend/cpp/SplayTree.hpp b/pydatastructs/trees/_backend/cpp/SplayTree.hpp index 37cf0e9db..b41eb3dce 100644 --- a/pydatastructs/trees/_backend/cpp/SplayTree.hpp +++ b/pydatastructs/trees/_backend/cpp/SplayTree.hpp @@ -284,6 +284,11 @@ static PyObject* SplayTree_split(SplayTree *self, PyObject* args) { return reinterpret_cast(other); } +static PyObject* SplayTree_root_idx(SplayTree *self, void *closure) { + return self->sbbt->bst->binary_tree->root_idx; +} + + static struct PyMethodDef SplayTree_PyMethodDef[] = { {"insert", (PyCFunction) SplayTree_insert, METH_VARARGS, NULL}, {"delete", (PyCFunction) SplayTree_delete, METH_VARARGS, NULL}, @@ -292,6 +297,10 @@ static struct PyMethodDef SplayTree_PyMethodDef[] = { {NULL} }; +static PyGetSetDef SplayTree_GetterSetters[] = { + {"root_idx", (getter) SplayTree_root_idx, NULL, "returns the index of the tree's root", NULL}, + {NULL} /* Sentinel */ +}; static PyMemberDef SplayTree_PyMemberDef[] = { {"tree", T_OBJECT_EX, offsetof(SplayTree, tree), 0, "tree"}, @@ -328,7 +337,7 @@ static PyTypeObject SplayTreeType = { /* tp_iternext */ 0, /* tp_methods */ SplayTree_PyMethodDef, /* tp_members */ SplayTree_PyMemberDef, - /* tp_getset */ 0, + /* tp_getset */ SplayTree_GetterSetters, /* tp_base */ &SelfBalancingBinaryTreeType, /* tp_dict */ 0, /* tp_descr_get */ 0, diff --git a/pydatastructs/trees/_backend/cpp/Treap.hpp b/pydatastructs/trees/_backend/cpp/Treap.hpp index 5a9420ee7..5619d274f 100644 --- a/pydatastructs/trees/_backend/cpp/Treap.hpp +++ b/pydatastructs/trees/_backend/cpp/Treap.hpp @@ -62,6 +62,10 @@ static PyObject* Treap_insert(Treap *self, PyObject* args) { return CartesianTree_insert(self->ct, Py_BuildValue("(OOO)", key, priority, data)); } +static PyObject* Treap_root_idx(Treap *self, void *closure) { + return self->ct->sbbt->bst->binary_tree->root_idx; +} + static struct PyMethodDef Treap_PyMethodDef[] = { {"insert", (PyCFunction) Treap_insert, METH_VARARGS, NULL}, @@ -70,6 +74,11 @@ static struct PyMethodDef Treap_PyMethodDef[] = { {NULL} /* Sentinel */ }; +static PyGetSetDef Treap_GetterSetters[] = { + {"root_idx", (getter) Treap_root_idx, NULL, "returns the index of the tree's root", NULL}, + {NULL} /* Sentinel */ +}; + static PyMemberDef Treap_PyMemberDef[] = { {"tree", T_OBJECT_EX, offsetof(Treap, tree), 0, "tree"}, {NULL} /* Sentinel */ @@ -105,7 +114,7 @@ static PyTypeObject TreapType = { /* tp_iternext */ 0, /* tp_methods */ Treap_PyMethodDef, /* tp_members */ Treap_PyMemberDef, - /* tp_getset */ 0, + /* tp_getset */ Treap_GetterSetters, /* tp_base */ &CartesianTreeType, /* tp_dict */ 0, /* tp_descr_get */ 0, diff --git a/pydatastructs/trees/tests/test_binary_trees.py b/pydatastructs/trees/tests/test_binary_trees.py index e8f8748c2..826100b78 100644 --- a/pydatastructs/trees/tests/test_binary_trees.py +++ b/pydatastructs/trees/tests/test_binary_trees.py @@ -26,6 +26,7 @@ def _test_BinarySearchTree(backend): ("[(1, 8, 8, 2), (3, 3, 3, 4), (None, 10, 10, 7), (None, 1, 1, None), " "(5, 6, 6, 6), (None, 4, 4, None), (None, 7, 7, None), (8, 14, 14, None), " "(None, 13, 13, None)]") + assert b.root_idx == 0 assert b.tree[0].left == 1 assert b.tree[0].key == 8 @@ -171,6 +172,7 @@ def _test_AVLTree(backend): a.insert('H', 'H') a.insert('I', 'I') a.insert('A', 'A') + assert a.root_idx == 1 trav = BinaryTreeTraversal(a, backend=backend) in_order = trav.depth_first_search(order='in_order') @@ -465,7 +467,7 @@ def _test_Treap(backend): tree.insert(3, 3) tree.insert(4, 4) tree.insert(5, 5) - print(str(tree)) + assert isinstance(tree.tree[0].priority, float) tree.delete(1) assert tree.search(1) is None @@ -497,6 +499,7 @@ def _test_SelfBalancingBinaryTree(backend): assert tree.tree[3].parent == 5 assert tree.tree[2].right != 3 assert tree.tree[tree.tree[5].parent].right == 5 + assert tree.root_idx == 0 trav = BinaryTreeTraversal(tree, backend=backend) in_order = trav.depth_first_search(order='in_order') @@ -533,6 +536,7 @@ def _test_SplayTree(backend): t.insert(20, 20) t.insert(55, 55) assert str(t) == "[(None, 100, 100, None), (None, 50, 50, None), (0, 200, 200, None), (None, 40, 40, 1), (5, 30, 30, 3), (None, 20, 20, None), (4, 55, 55, 2)]" + assert t.root_idx == 6 trav = BinaryTreeTraversal(t, backend=backend) in_order = trav.depth_first_search(order='in_order') @@ -610,6 +614,7 @@ def _test_RedBlackTree(backend): tree.insert(17, 17) tree.insert(6, 6) assert str(tree) == "[(11, 10, 10, 3), (10, 18, 18, None), (None, 7, 7, None), (None, 15, 15, None), (0, 16, 16, 6), (None, 30, 30, None), (1, 25, 25, 7), (5, 40, 40, 8), (None, 60, 60, None), (None, 2, 2, None), (None, 17, 17, None), (9, 6, 6, 2)]" + assert tree.root_idx == 4 trav = BinaryTreeTraversal(tree, backend=backend) in_order = trav.depth_first_search(order='in_order')