197 lines
3.2 KiB
Plaintext
197 lines
3.2 KiB
Plaintext
|
(*
|
||
|
BSD 2-Clause License
|
||
|
|
||
|
Copyright (c) 2018, Anton Krotov
|
||
|
All rights reserved.
|
||
|
*)
|
||
|
|
||
|
MODULE AVLTREES;
|
||
|
|
||
|
IMPORT C := COLLECTIONS;
|
||
|
|
||
|
|
||
|
TYPE
|
||
|
|
||
|
DATA* = POINTER TO RECORD (C.ITEM) END;
|
||
|
|
||
|
NODE* = POINTER TO RECORD (C.ITEM)
|
||
|
|
||
|
data*: DATA;
|
||
|
|
||
|
height: INTEGER;
|
||
|
|
||
|
left*, right*: NODE
|
||
|
|
||
|
END;
|
||
|
|
||
|
CMP* = PROCEDURE (a, b: DATA): INTEGER;
|
||
|
|
||
|
DESTRUCTOR* = PROCEDURE (VAR data: DATA);
|
||
|
|
||
|
|
||
|
VAR
|
||
|
|
||
|
nodes: C.COLLECTION;
|
||
|
|
||
|
|
||
|
PROCEDURE NewNode (data: DATA): NODE;
|
||
|
VAR
|
||
|
node: NODE;
|
||
|
citem: C.ITEM;
|
||
|
|
||
|
BEGIN
|
||
|
citem := C.pop(nodes);
|
||
|
IF citem = NIL THEN
|
||
|
NEW(node)
|
||
|
ELSE
|
||
|
node := citem(NODE)
|
||
|
END;
|
||
|
|
||
|
node.data := data;
|
||
|
node.left := NIL;
|
||
|
node.right := NIL;
|
||
|
node.height := 1
|
||
|
|
||
|
RETURN node
|
||
|
END NewNode;
|
||
|
|
||
|
|
||
|
PROCEDURE height (p: NODE): INTEGER;
|
||
|
VAR
|
||
|
res: INTEGER;
|
||
|
|
||
|
BEGIN
|
||
|
IF p = NIL THEN
|
||
|
res := 0
|
||
|
ELSE
|
||
|
res := p.height
|
||
|
END
|
||
|
|
||
|
RETURN res
|
||
|
END height;
|
||
|
|
||
|
|
||
|
PROCEDURE bfactor (p: NODE): INTEGER;
|
||
|
RETURN height(p.right) - height(p.left)
|
||
|
END bfactor;
|
||
|
|
||
|
|
||
|
PROCEDURE fixheight (p: NODE);
|
||
|
BEGIN
|
||
|
p.height := MAX(height(p.left), height(p.right)) + 1
|
||
|
END fixheight;
|
||
|
|
||
|
|
||
|
PROCEDURE rotateright (p: NODE): NODE;
|
||
|
VAR
|
||
|
q: NODE;
|
||
|
|
||
|
BEGIN
|
||
|
q := p.left;
|
||
|
p.left := q.right;
|
||
|
q.right := p;
|
||
|
fixheight(p);
|
||
|
fixheight(q)
|
||
|
|
||
|
RETURN q
|
||
|
END rotateright;
|
||
|
|
||
|
|
||
|
PROCEDURE rotateleft (q: NODE): NODE;
|
||
|
VAR
|
||
|
p: NODE;
|
||
|
|
||
|
BEGIN
|
||
|
p := q.right;
|
||
|
q.right := p.left;
|
||
|
p.left := q;
|
||
|
fixheight(q);
|
||
|
fixheight(p)
|
||
|
|
||
|
RETURN p
|
||
|
END rotateleft;
|
||
|
|
||
|
|
||
|
PROCEDURE balance (p: NODE): NODE;
|
||
|
VAR
|
||
|
res: NODE;
|
||
|
|
||
|
BEGIN
|
||
|
fixheight(p);
|
||
|
|
||
|
IF bfactor(p) = 2 THEN
|
||
|
IF bfactor(p.right) < 0 THEN
|
||
|
p.right := rotateright(p.right)
|
||
|
END;
|
||
|
res := rotateleft(p)
|
||
|
|
||
|
ELSIF bfactor(p) = -2 THEN
|
||
|
IF bfactor(p.left) > 0 THEN
|
||
|
p.left := rotateleft(p.left)
|
||
|
END;
|
||
|
res := rotateright(p)
|
||
|
|
||
|
ELSE
|
||
|
res := p
|
||
|
END
|
||
|
|
||
|
RETURN res
|
||
|
END balance;
|
||
|
|
||
|
|
||
|
PROCEDURE insert* (p: NODE; data: DATA; cmp: CMP; VAR newnode: BOOLEAN; VAR node: NODE): NODE;
|
||
|
VAR
|
||
|
res: NODE;
|
||
|
rescmp: INTEGER;
|
||
|
|
||
|
BEGIN
|
||
|
IF p = NIL THEN
|
||
|
res := NewNode(data);
|
||
|
node := res;
|
||
|
newnode := TRUE
|
||
|
ELSE
|
||
|
|
||
|
rescmp := cmp(data, p.data);
|
||
|
IF rescmp < 0 THEN
|
||
|
p.left := insert(p.left, data, cmp, newnode, node);
|
||
|
res := balance(p)
|
||
|
ELSIF rescmp > 0 THEN
|
||
|
p.right := insert(p.right, data, cmp, newnode, node);
|
||
|
res := balance(p)
|
||
|
ELSE
|
||
|
res := p;
|
||
|
node := res;
|
||
|
newnode := FALSE
|
||
|
END
|
||
|
|
||
|
END
|
||
|
|
||
|
RETURN res
|
||
|
END insert;
|
||
|
|
||
|
|
||
|
PROCEDURE destroy* (VAR node: NODE; destructor: DESTRUCTOR);
|
||
|
VAR
|
||
|
left, right: NODE;
|
||
|
|
||
|
BEGIN
|
||
|
IF node # NIL THEN
|
||
|
left := node.left;
|
||
|
right := node.right;
|
||
|
|
||
|
IF destructor # NIL THEN
|
||
|
destructor(node.data)
|
||
|
END;
|
||
|
|
||
|
C.push(nodes, node);
|
||
|
node := NIL;
|
||
|
|
||
|
destroy(left, destructor);
|
||
|
destroy(right, destructor)
|
||
|
END
|
||
|
END destroy;
|
||
|
|
||
|
|
||
|
BEGIN
|
||
|
nodes := C.create()
|
||
|
END AVLTREES.
|