Splay Tree-
Splay tree is a binary search tree. In a splay tree, M consecutive operations can be performed in O (M log N) time.
A single operation may require O(N) time but average time to perform M operations will need O (M Log N) time.
When a node is accessed, it is moved to the top through a set of operations known as splaying. Splaying technique is similar to rotation in an AVL tree. This will make the future access of the node cheaper.
Unlike AVL tree, splay trees do not have the requirement of storing Balance Factor of every node. This saves the space and simplifies algorithm to a great extent.
There are two standard techniques of splaying.
1. Bottom up Splaying
2. Top Down Splaying
1. Bottom up Splaying:-
Idea behind bottom up splaying is explained below: Rotation is performed bottom up along the access path.
Let X be a (non root) node on the access path at which we are rotating.
a) If the parent of X is the root of the tree, rotate X and the parent of X. This will be the last rotation required.
b) If X has both a parent (P) and Grand parent (G) then like an AVL tree there could be four cases.
These four cases are:
1. X is a left child and P is a left child.
2. X is a left child and P is right child
3. X is a right child and P is a left child
4. X is a right child and P is a right child.
2.Top Down Splaying:-
When an item X is inserted as a leaf, a series of tree rotations brings X at the root. These rotations are known as splaying. A splay is also performed during searches, and if an item is not found, a splay is performed on the last node on the access path.
A top down traversal is performed to locate the leaf node.
Splaying is done using bottom up traversal.
This can be done by storing the access path, during top down traversal on a stack.
Top down splaying is based on splaying on the initial traversal path. A stack is not required to save the traversal path.
At any point in the access:
1. A current node X that is the root of its sub tree and represented as the “middle” tree.
2. Tree L stores nodes in the tree T that are less than X, but not in the X’s sub tree.
3. Tree R stores nodes in the tree T that are larger than X, but not in X’s sub tree.
4. Initially, X is the root of T, and L and R are empty.
The worst case time complexity of Binary Search Tree (BST) operations like search, delete, insert is O(n). The worst case occurs when the tree is skewed. We can get the worst case time complexity as O(Logn) with AVL and Red-Black Trees.
Can we do better than AVL or Red-Black trees in practical situations?
Like AVL and Red-Black Trees, Splay tree is also self-balancing BST. The main idea of splay tree is to bring the recently accessed item to root of the tree, this makes the recently searched item to be accessible in O(1) time if accessed again. The idea is to use locality of reference (In a typical application, 80% of the access are to 20% of the items). Imagine a situation where we have millions or billions of keys and only few of them are accessed frequently, which is very likely in many practical applications.
All splay tree operations run in O(log n) time on average, where n is the number of entries in the tree. Any single operation can take Theta(n) time in the worst case.
Search Operation
The search operation in Splay tree does the standard BST search, in addition to search, it also splays (move a node to the root). If the search is successful, then the node that is found is splayed and becomes the new root. Else the last node accessed prior to reaching the NULL is splayed and becomes the new root.
There are following cases for the node being accessed.
1) Node is root We simply return the root, don’t do anything else as the accessed node is already root.
2) Zig: Node is child of root (the node has no grandparent). Node is either a left child of root (we do a right rotation) or node is a right child of its parent (we do a left rotation).
T1, T2 and T3 are subtrees of the tree rooted with y (on left side) or x (on right side)
y x
/ \ Zig (Right Rotation) / \
x T3 – - – - – - – - - -> T1 y
/ \ < - - - - - - - - - / \
T1 T2 Zag (Left Rotation) T2 T3
3) Node has both parent and grandparent. There can be following subcases.
……..3.a) Zig-Zig and Zag-Zag Node is left child of parent and parent is also left child of grand parent (Two right rotations) OR node is right child of its parent and parent is also right child of grand parent (Two Left Rotations).
Zig-Zig (Left Left Case):
G P X
/ \ / \ / \
P T4 rightRotate(G) X G rightRotate(P) T1 P
/ \ ============> / \ / \ ============> / \
X T3 T1 T2 T3 T4 T2 G
/ \ / \
T1 T2 T3 T4
Zag-Zag (Right Right Case):
G P X
/ \ / \ / \
T1 P leftRotate(G) G X leftRotate(P) P T4
/ \ ============> / \ / \ ============> / \
T2 X T1 T2 T3 T4 G T3
/ \ / \
T3 T4 T1 T2
……..3.b) Zig-Zag and Zag-Zig Node is right child of parent and parent is left child of grand parent (Left Rotation followed by right rotation) OR node is left child of its parent and parent is right child of grand parent (Right Rotation followed by left rotation).
Zag-Zig (Left Right Case):
G G X
/ \ / \ / \
P T4 leftRotate(P) X T4 rightRotate(G) P G
/ \ ============> / \ ============> / \ / \
T1 X P T3 T1 T2 T3 T4
/ \ / \
T2 T3 T1 T2
Zig-Zag (Right Left Case):
G G X
/ \ / \ / \
T1 P rightRotate(P) T1 X leftRotate(G) G P
/ \ =============> / \ ============> / \ / \
X T4 T2 P T1 T2 T3 T4
/ \ / \
T2 T3 T3 T4
Example:
100 100 [20]
/ \ / \ \
50 200 50 200 50
/ search(20) / search(20) / \
40 ======> [20] ========> 30 100
/ 1. Zig-Zig \ 2. Zig-Zig \ \
30 at 40 30 at 100 40 200
/ \
[20] 40
The important thing to note is, the search or splay operation not only brings the searched key to root, but also balances the BST. For example in above case, height of BST is reduced by 1.
Implementation:
C++
#include <bits/stdc++.h>
using namespace std;
class node
{
public :
int key;
node *left, *right;
};
node* newNode( int key)
{
node* Node = new node();
Node->key = key;
Node->left = Node->right = NULL;
return (Node);
}
node *rightRotate(node *x)
{
node *y = x->left;
x->left = y->right;
y->right = x;
return y;
}
node *leftRotate(node *x)
{
node *y = x->right;
x->right = y->left;
y->left = x;
return y;
}
node *splay(node *root, int key)
{
if (root == NULL || root->key == key)
return root;
if (root->key > key)
{
if (root->left == NULL) return root;
if (root->left->key > key)
{
root->left->left = splay(root->left->left, key);
root = rightRotate(root);
}
else if (root->left->key < key)
{
root->left->right = splay(root->left->right, key);
if (root->left->right != NULL)
root->left = leftRotate(root->left);
}
return (root->left == NULL)? root: rightRotate(root);
}
else
{
if (root->right == NULL) return root;
if (root->right->key > key)
{
root->right->left = splay(root->right->left, key);
if (root->right->left != NULL)
root->right = rightRotate(root->right);
}
else if (root->right->key < key)
{
root->right->right = splay(root->right->right, key);
root = leftRotate(root);
}
return (root->right == NULL)? root: leftRotate(root);
}
}
node *search(node *root, int key)
{
return splay(root, key);
}
void preOrder(node *root)
{
if (root != NULL)
{
cout<<root->key<< " " ;
preOrder(root->left);
preOrder(root->right);
}
}
int main()
{
node *root = newNode(100);
root->left = newNode(50);
root->right = newNode(200);
root->left->left = newNode(40);
root->left->left->left = newNode(30);
root->left->left->left->left = newNode(20);
root = search(root, 20);
cout << "Preorder traversal of the modified Splay tree is \n" ;
preOrder(root);
return 0;
}
|
C
#include<stdio.h>
#include<stdlib.h>
struct node
{
int key;
struct node *left, *right;
};
struct node* newNode( int key)
{
struct node* node = ( struct node*) malloc ( sizeof ( struct node));
node->key = key;
node->left = node->right = NULL;
return (node);
}
struct node *rightRotate( struct node *x)
{
struct node *y = x->left;
x->left = y->right;
y->right = x;
return y;
}
struct node *leftRotate( struct node *x)
{
struct node *y = x->right;
x->right = y->left;
y->left = x;
return y;
}
struct node *splay( struct node *root, int key)
{
if (root == NULL || root->key == key)
return root;
if (root->key > key)
{
if (root->left == NULL) return root;
if (root->left->key > key)
{
root->left->left = splay(root->left->left, key);
root = rightRotate(root);
}
else if (root->left->key < key)
{
root->left->right = splay(root->left->right, key);
if (root->left->right != NULL)
root->left = leftRotate(root->left);
}
return (root->left == NULL)? root: rightRotate(root);
}
else
{
if (root->right == NULL) return root;
if (root->right->key > key)
{
root->right->left = splay(root->right->left, key);
if (root->right->left != NULL)
root->right = rightRotate(root->right);
}
else if (root->right->key < key)
{
root->right->right = splay(root->right->right, key);
root = leftRotate(root);
}
return (root->right == NULL)? root: leftRotate(root);
}
}
struct node *search( struct node *root, int key)
{
return splay(root, key);
}
void preOrder( struct node *root)
{
if (root != NULL)
{
printf ( "%d " , root->key);
preOrder(root->left);
preOrder(root->right);
}
}
int main()
{
struct node *root = newNode(100);
root->left = newNode(50);
root->right = newNode(200);
root->left->left = newNode(40);
root->left->left->left = newNode(30);
root->left->left->left->left = newNode(20);
root = search(root, 20);
printf ( "Preorder traversal of the modified Splay tree is \n" );
preOrder(root);
return 0;
}
|
Java
class GFG
{
static class node
{
int key;
node left, right;
};
static node newNode( int key)
{
node Node = new node();
Node.key = key;
Node.left = Node.right = null ;
return (Node);
}
static node rightRotate(node x)
{
node y = x.left;
x.left = y.right;
y.right = x;
return y;
}
static node leftRotate(node x)
{
node y = x.right;
x.right = y.left;
y.left = x;
return y;
}
static node splay(node root, int key)
{
if (root == null || root.key == key)
return root;
if (root.key > key)
{
if (root.left == null ) return root;
if (root.left.key > key)
{
root.left.left = splay(root.left.left, key);
root = rightRotate(root);
}
else if (root.left.key < key)
{
root.left.right = splay(root.left.right, key);
if (root.left.right != null )
root.left = leftRotate(root.left);
}
return (root.left == null ) ?
root : rightRotate(root);
}
else
{
if (root.right == null ) return root;
if (root.right.key > key)
{
root.right.left = splay(root.right.left, key);
if (root.right.left != null )
root.right = rightRotate(root.right);
}
else if (root.right.key < key)
{
root.right.right = splay(root.right.right, key);
root = leftRotate(root);
}
return (root.right == null ) ?
root : leftRotate(root);
}
}
static node search(node root, int key)
{
return splay(root, key);
}
static void preOrder(node root)
{
if (root != null )
{
System.out.print(root.key + " " );
preOrder(root.left);
preOrder(root.right);
}
}
public static void main(String[] args)
{
node root = newNode( 100 );
root.left = newNode( 50 );
root.right = newNode( 200 );
root.left.left = newNode( 40 );
root.left.left.left = newNode( 30 );
root.left.left.left.left = newNode( 20 );
root = search(root, 20 );
System.out.print( "Preorder traversal of the" +
" modified Splay tree is \n" );
preOrder(root);
}
}
|
Python3
class Node:
def __init__( self , key):
self .key = key
self .left = None
self .right = None
def newNode(key):
node = Node(key)
return node
def rightRotate(x):
y = x.left
x.left = y.right
y.right = x
return y
def leftRotate(x):
y = x.right
x.right = y.left
y.left = x
return y
def splay(root, key):
if root is None or root.key = = key:
return root
if root.key > key:
if root.left is None :
return root
if root.left.key > key:
root.left.left = splay(root.left.left, key)
root = rightRotate(root)
elif root.left.key < key:
root.left.right = splay(root.left.right, key)
if root.left.right:
root.left = leftRotate(root.left)
return (root.left is None ) and root or rightRotate(root)
else :
if root.right is None :
return root
if root.right.key > key:
root.right.left = splay(root.right.left, key)
if root.right.left:
root.right = rightRotate(root.right)
elif root.right.key < key:
root.right.right = splay(root.right.right, key)
root = leftRotate(root)
return (root.right is None ) and root or leftRotate(root)
def search(root, key):
return splay(root, key)
def insert(root, key):
if root is None :
return newNode(key)
root = splay(root, key)
if root.key = = key:
return root
if root.key > key:
new_node = newNode(key)
new_node.right = root
new_node.left = root.left
root.left = None
return new_node
else :
new_node = newNode(key)
new_node.left = root
new_node.right = root.right
root.right = None
return new_node
def delete(root, key):
if root is None :
return root
root = splay(root, key)
if root.key ! = key:
return root
if root.left is None :
new_root = root.right
else :
new_root = splay(root.left, key)
new_root.right = root.right
return new_root
def preOrder(root):
if root:
print (root.key, end = ' ' )
preOrder(root.left)
preOrder(root.right)
if __name__ = = '__main__' :
root = newNode( 100 )
root.left = newNode( 50 )
root.right = newNode( 200 )
root.left.left = newNode( 40 )
root.left.left.left = newNode( 30 )
root.left.left.left.left = newNode( 20 )
root = splay(root, 20 )
print ( "Preorder traversal of the modified Splay tree is" )
preOrder(root)
|
C#
using System;
class GFG
{
public class node
{
public int key;
public node left, right;
};
static node newNode( int key)
{
node Node = new node();
Node.key = key;
Node.left = Node.right = null ;
return (Node);
}
static node rightRotate(node x)
{
node y = x.left;
x.left = y.right;
y.right = x;
return y;
}
static node leftRotate(node x)
{
node y = x.right;
x.right = y.left;
y.left = x;
return y;
}
static node splay(node root, int key)
{
if (root == null || root.key == key)
return root;
if (root.key > key)
{
if (root.left == null ) return root;
if (root.left.key > key)
{
root.left.left = splay(root.left.left, key);
root = rightRotate(root);
}
else if (root.left.key < key)
{
root.left.right = splay(root.left.right, key);
if (root.left.right != null )
root.left = leftRotate(root.left);
}
return (root.left == null ) ?
root : rightRotate(root);
}
else
{
if (root.right == null ) return root;
if (root.right.key > key)
{
root.right.left = splay(root.right.left, key);
if (root.right.left != null )
root.right = rightRotate(root.right);
}
else if (root.right.key < key)
{
root.right.right = splay(root.right.right, key);
root = leftRotate(root);
}
return (root.right == null ) ?
root : leftRotate(root);
}
}
static node search(node root, int key)
{
return splay(root, key);
}
static void preOrder(node root)
{
if (root != null )
{
Console.Write(root.key + " " );
preOrder(root.left);
preOrder(root.right);
}
}
public static void Main(String[] args)
{
node root = newNode(100);
root.left = newNode(50);
root.right = newNode(200);
root.left.left = newNode(40);
root.left.left.left = newNode(30);
root.left.left.left.left = newNode(20);
root = search(root, 20);
Console.Write( "Preorder traversal of the" +
" modified Splay tree is \n" );
preOrder(root);
}
}
|
Javascript
<script>
class Node
{
constructor(key)
{
this .key = key;
this .left = this .right = null ;
}
}
function rightRotate(x)
{
let y = x.left;
x.left = y.right;
y.right = x;
return y;
}
function leftRotate(x)
{
let y = x.right;
x.right = y.left;
y.left = x;
return y;
}
function splay(root,key)
{
if (root == null || root.key == key)
return root;
if (root.key > key)
{
if (root.left == null ) return root;
if (root.left.key > key)
{
root.left.left = splay(root.left.left, key);
root = rightRotate(root);
}
else if (root.left.key < key)
{
root.left.right = splay(root.left.right, key);
if (root.left.right != null )
root.left = leftRotate(root.left);
}
return (root.left == null ) ?
root : rightRotate(root);
}
else
{
if (root.right == null ) return root;
if (root.right.key > key)
{
root.right.left = splay(root.right.left, key);
if (root.right.left != null )
root.right = rightRotate(root.right);
}
else if (root.right.key < key)
{
root.right.right = splay(root.right.right, key);
root = leftRotate(root);
}
return (root.right == null ) ?
root : leftRotate(root);
}
}
function search(root,key)
{
return splay(root, key);
}
function preOrder(root)
{
if (root != null )
{
document.write(root.key + " " );
preOrder(root.left);
preOrder(root.right);
}
}
let root = new Node(100);
root.left = new Node(50);
root.right = new Node(200);
root.left.left = new Node(40);
root.left.left.left = new Node(30);
root.left.left.left.left = new Node(20);
root = search(root, 20);
document.write( "Preorder traversal of the" +
" modified Splay tree is <br>" );
preOrder(root);
</script>
|
Output:
Preorder traversal of the modified Splay tree is
20 50 30 40 100 200
Time complexity – The time complexity of splay tree operations depends on the height of the tree, which is usually O(log n) in the average case, where n is the number of nodes in the tree. However, in the worst case, the tree can degenerate into a linear chain, resulting in O(n) time complexity for certain operations.
Space complexity –The space complexity of the splay tree depends on the number of nodes in the tree. In the worst case, when the tree degenerates into a linear chain, the space complexity is O(n). However, in the average case, it is O(n log n).
Summary
1) Splay trees have excellent locality properties. Frequently accessed items are easy to find. Infrequent items are out of way.
2) All splay tree operations take O(Logn) time on average. Splay trees can be rigorously shown to run in O(log n) average time per operation, over any sequence of operations (assuming we start from an empty tree)
3) Splay trees are simpler compared to AVL and Red-Black Trees as no extra field is required in every tree node.
4) Unlike AVL tree, a splay tree can change even with read-only operations like search.
Applications of Splay Trees
Splay trees have become the most widely used basic data structure invented in the last 30 years, because they’re the fastest type of balanced search tree for many applications.
Splay trees are used in Windows NT (in the virtual memory, networking, and file system code), the gcc compiler and GNU C++ library, the sed string editor, For Systems network routers, the most popular implementation of Unix malloc, Linux loadable kernel modules, and in much other software.
See Splay Tree | Set 2 (Insert) for splay tree insertion.
Advantages of Splay Trees:
- Useful for implementing caches and garbage collection algorithms.
- Require less space as there is no balance information is required.
- Splay trees provide good performance with nodes containing identical keys.
Disadvantages of Splay Trees:
- The height of a splay tree can be linear when accessing elements in non decreasing order.
- The performance of a splay tree will be worse than a balanced simple binary search tree in case of uniform access.
Unlock your potential with our DSA Self-Paced course, designed to help you master Data Structures and Algorithms at your own pace. In 90 days, you’ll learn the core concepts of DSA, tackle real-world problems, and boost your problem-solving skills, all at a speed that fits your schedule. With comprehensive lessons and practical exercises, this course will set you up for success in technical interviews and beyond.
And here's the challenge – join the Three 90 Challenge! Complete 90% of the course in 90 days, and you’ll earn a 90% refund. It's a great way to stay motivated, track your progress, and reward yourself for your dedication. Start the challenge today, and push your limits to achieve mastery in DSA!
Similar Reads
Advanced Data Structures
Advanced Data Structures refer to complex and specialized arrangements of data that enable efficient storage, retrieval, and manipulation of information in computer science and programming. These structures go beyond basic data types like arrays and lists, offering sophisticated ways to organize and
3 min read
Generic Linked List in C
A generic linked list is a type of linked list that allows the storage of different types of data in a single linked list structure, providing more versatility and reusability in various applications. Unlike C++ and Java, C doesn't directly support generics. However, we can write generic code using
5 min read
Memory efficient doubly linked list
We need to implement a doubly linked list with the use of a single pointer in each node. For that we are given a stream of data of size n for the linked list, your task is to make the function insert() and getList(). The insert() function pushes (or inserts at the beginning) the given data in the li
9 min read
XOR Linked List - A Memory Efficient Doubly Linked List | Set 1
In this post, we're going to talk about how XOR linked lists are used to reduce the memory requirements of doubly-linked lists. We know that each node in a doubly-linked list has two pointer fields which contain the addresses of the previous and next node. On the other hand, each node of the XOR lin
15+ min read
XOR Linked List – A Memory Efficient Doubly Linked List | Set 2
In the previous post, we discussed how a Doubly Linked can be created using only one space for the address field with every node. In this post, we will discuss the implementation of a memory-efficient doubly linked list. We will mainly discuss the following two simple functions. A function to insert
10 min read
Skip List - Efficient Search, Insert and Delete in Linked List
A skip list is a data structure that allows for efficient search, insertion and deletion of elements in a sorted list. It is a probabilistic data structure, meaning that its average time complexity is determined through a probabilistic analysis. In a skip list, elements are organized in layers, with
6 min read
Self Organizing List | Set 1 (Introduction)
The worst case search time for a sorted linked list is O(n). With a Balanced Binary Search Tree, we can skip almost half of the nodes after one comparison with root. For a sorted array, we have random access and we can apply Binary Search on arrays. One idea to make search faster for Linked Lists is
3 min read
Unrolled Linked List | Set 1 (Introduction)
Like array and linked list, the unrolled Linked List is also a linear data structure and is a variant of a linked list. Why do we need unrolled linked list? One of the biggest advantages of linked lists over arrays is that inserting an element at any location takes only O(1). However, the catch here
10 min read
Scape Goat Tree and Treap