You are given a tree with n
nodes numbered from 0
to n - 1
in the form of a parent array parent
where parent[i]
is the parent of the ith
node. The root of the tree is node 0
, so parent[0] = -1
since it has no parent. You want to design a data structure that allows users to lock, unlock, and upgrade nodes in the tree.
The data structure should support the following functions:
Implement the LockingTree
class:
LockingTree(int[] parent)
initializes the data structure with the parent array.lock(int num, int user)
returns true
if it is possible for the user with id user
to lock the node num
, or false
otherwise. If it is possible, the node num
will become locked by the user with id user
.unlock(int num, int user)
returns true
if it is possible for the user with id user
to unlock the node num
, or false
otherwise. If it is possible, the node num
will become unlocked.upgrade(int num, int user)
returns true
if it is possible for the user with id user
to upgrade the node num
, or false
otherwise. If it is possible, the node num
will be upgraded.
Example 1:
Input ["LockingTree", "lock", "unlock", "unlock", "lock", "upgrade", "lock"] [[[-1, 0, 0, 1, 1, 2, 2]], [2, 2], [2, 3], [2, 2], [4, 5], [0, 1], [0, 1]] Output [null, true, false, true, true, true, false] Explanation LockingTree lockingTree = new LockingTree([-1, 0, 0, 1, 1, 2, 2]); lockingTree.lock(2, 2); // return true because node 2 is unlocked. // Node 2 will now be locked by user 2. lockingTree.unlock(2, 3); // return false because user 3 cannot unlock a node locked by user 2. lockingTree.unlock(2, 2); // return true because node 2 was previously locked by user 2. // Node 2 will now be unlocked. lockingTree.lock(4, 5); // return true because node 4 is unlocked. // Node 4 will now be locked by user 5. lockingTree.upgrade(0, 1); // return true because node 0 is unlocked and has at least one locked descendant (node 4). // Node 0 will now be locked by user 1 and node 4 will now be unlocked. lockingTree.lock(0, 1); // return false because node 0 is already locked.
Constraints:
n == parent.length
2 <= n <= 2000
0 <= parent[i] <= n - 1
for i != 0
parent[0] == -1
0 <= num <= n - 1
1 <= user <= 104
parent
represents a valid tree.2000
calls in total will be made to lock
, unlock
, and upgrade
.This problem involves designing a LockingTree
data structure that manages locking and unlocking of nodes in a tree. The tree is represented using a parent array, where parent[i]
indicates the parent of node i
. The root node is node 0, with parent[0] = -1
.
The data structure supports three operations:
lock(num, user)
: Locks node num
for user user
. Returns true
if successful (node was unlocked), false
otherwise.
unlock(num, user)
: Unlocks node num
for user user
. Returns true
if successful (node was locked by user
), false
otherwise.
upgrade(num, user)
: Upgrades node num
for user user
. This locks node num
for user user
and unlocks all its descendants, regardless of who locked them. The upgrade is only possible if:
num
is unlocked.num
has at least one locked descendant.num
has no locked ancestors.The solution uses a combination of arrays and potentially lists (depending on the language) to efficiently manage the tree structure and locking information.
Data Structures:
locked
: An array to store the user ID locking each node. -1
indicates the node is unlocked.parent
: The input parent array representing the tree structure.children
: A list/array of lists/arrays to represent the children of each node. This is built from the parent
array for efficient traversal.lock
and unlock
: These operations are straightforward. Check the locked
array to see if the node is already locked or if the unlocking user matches the locking user.
upgrade
: This is the most complex operation. It involves three checks:
parent
array. If any ancestor is locked, the upgrade fails.find
flag to true.true
.Time Complexity:
lock
and unlock
: O(1)upgrade
: O(N) in the worst case, where N is the number of nodes in the tree, due to the traversal of ancestors and descendants.Space Complexity:
locked
, parent
, and children
arrays/lists. The recursive DFS call stack in upgrade
also contributes to this space complexity.The Python code provided implements this approach efficiently. The __init__
method builds the children
array from the parent
array. The lock
, unlock
, and upgrade
methods implement the logic described above. The dfs
function in upgrade
performs the Depth-First Search to check for and unlock locked descendants.
The other language implementations follow the same underlying logic, with minor syntactic differences due to the language specifics. Each language version is structured similarly to optimize the operations and ensure efficient tree traversal.