All Downloads are FREE. Search and download functionalities are using the official Maven repository.

g1501_1600.s1600_throne_inheritance.ThroneInheritance Maven / Gradle / Ivy

There is a newer version: 1.38
Show newest version
package g1501_1600.s1600_throne_inheritance;

// #Medium #Hash_Table #Depth_First_Search #Tree #Design
// #2022_04_01_Time_267_ms_(90.38%)_Space_114.3_MB_(85.58%)

import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;

/**
 * 1600 - Throne Inheritance\.
 *
 * Medium
 *
 * A kingdom consists of a king, his children, his grandchildren, and so on. Every once in a while, someone in the family dies or a child is born.
 *
 * The kingdom has a well-defined order of inheritance that consists of the king as the first member. Let's define the recursive function `Successor(x, curOrder)`, which given a person `x` and the inheritance order so far, returns who should be the next person after `x` in the order of inheritance.
 *
 * Successor(x, curOrder): if x has no children or all of x's children are in curOrder: if x is the king return null else return Successor(x's parent, curOrder) else return x's oldest child who's not in curOrder 
 *
 * For example, assume we have a kingdom that consists of the king, his children Alice and Bob (Alice is older than Bob), and finally Alice's son Jack.
 *
 * 1.  In the beginning, `curOrder` will be `["king"]`.
 * 2.  Calling `Successor(king, curOrder)` will return Alice, so we append to `curOrder` to get `["king", "Alice"]`.
 * 3.  Calling `Successor(Alice, curOrder)` will return Jack, so we append to `curOrder` to get `["king", "Alice", "Jack"]`.
 * 4.  Calling `Successor(Jack, curOrder)` will return Bob, so we append to `curOrder` to get `["king", "Alice", "Jack", "Bob"]`.
 * 5.  Calling `Successor(Bob, curOrder)` will return `null`. Thus the order of inheritance will be `["king", "Alice", "Jack", "Bob"]`.
 *
 * Using the above function, we can always obtain a unique order of inheritance.
 *
 * Implement the `ThroneInheritance` class:
 *
 * *   `ThroneInheritance(string kingName)` Initializes an object of the `ThroneInheritance` class. The name of the king is given as part of the constructor.
 * *   `void birth(string parentName, string childName)` Indicates that `parentName` gave birth to `childName`.
 * *   `void death(string name)` Indicates the death of `name`. The death of the person doesn't affect the `Successor` function nor the current inheritance order. You can treat it as just marking the person as dead.
 * *   `string[] getInheritanceOrder()` Returns a list representing the current order of inheritance **excluding** dead people.
 *
 * **Example 1:**
 *
 * **Input**
 *
 * ["ThroneInheritance", "birth", "birth", "birth", "birth", "birth", "birth", "getInheritanceOrder", "death", "getInheritanceOrder"]
 *
 * [["king"], ["king", "andy"], ["king", "bob"], ["king", "catherine"], ["andy", "matthew"], ["bob", "alex"], ["bob", "asha"], [null], ["bob"], [null]]
 *
 * **Output:** [null, null, null, null, null, null, null, ["king", "andy", "matthew", "bob", "alex", "asha", "catherine"], null, ["king", "andy", "matthew", "alex", "asha", "catherine"]]
 *
 * **Explanation:**
 *
 * ThroneInheritance t= new ThroneInheritance("king"); // order: **king**
 *
 * t.birth("king", "andy"); // order: king > **andy**
 *
 * t.birth("king", "bob"); // order: king > andy > **bob**
 *
 * t.birth("king", "catherine"); // order: king > andy > bob > **catherine**
 *
 * t.birth("andy", "matthew"); // order: king > andy > **matthew** > bob > catherine
 *
 * t.birth("bob", "alex"); // order: king > andy > matthew > bob > **alex** > catherine
 *
 * t.birth("bob", "asha"); // order: king > andy > matthew > bob > alex > **asha** > catherine
 *
 * t.getInheritanceOrder(); // return ["king", "andy", "matthew", "bob", "alex", "asha", "catherine"]
 *
 * t.death("bob"); // order: king > andy > matthew > **bob** > alex > asha > catherine t.getInheritanceOrder(); // return ["king", "andy", "matthew", "alex", "asha", "catherine"] 
 *
 * **Constraints:**
 *
 * *   `1 <= kingName.length, parentName.length, childName.length, name.length <= 15`
 * *   `kingName`, `parentName`, `childName`, and `name` consist of lowercase English letters only.
 * *   All arguments `childName` and `kingName` are **distinct**.
 * *   All `name` arguments of `death` will be passed to either the constructor or as `childName` to `birth` first.
 * *   For each call to `birth(parentName, childName)`, it is guaranteed that `parentName` is alive.
 * *   At most 105 calls will be made to `birth` and `death`.
 * *   At most `10` calls will be made to `getInheritanceOrder`.
**/
public class ThroneInheritance {
    private String king;
    private HashMap> graph;
    private HashSet isDead;

    public ThroneInheritance(String kingName) {
        king = kingName;
        graph = new HashMap<>();
        isDead = new HashSet<>();
        graph.put(kingName, new LinkedHashSet<>());
    }

    public void birth(String parentName, String childName) {
        graph.putIfAbsent(parentName, new LinkedHashSet<>());
        graph.get(parentName).add(childName);
    }

    public void death(String name) {
        isDead.add(name);
    }

    public List getInheritanceOrder() {
        List inheritance = new ArrayList<>();
        HashSet visited = new HashSet<>();
        dfs(graph, king, inheritance, visited);
        return inheritance;
    }

    public void dfs(
            Map> graph,
            String src,
            List l,
            Set visited) {
        visited.add(src);
        if (!isDead.contains(src)) {
            l.add(src);
        }
        if (!graph.containsKey(src)) {
            return;
        }
        for (String s : graph.get(src)) {
            if (!visited.contains(s)) {
                dfs(graph, s, l, visited);
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy