# // Design a data structure that follows the constraints of a Least Recently Used (LRU) cache.
# // Implement the LRUCache class:
# // LRUCache(int capacity) Initialize the LRU cache with positive size capacity.
# // int get(int key) Return the value of the key if the key exists, otherwise return -1.
# // void put(int key, int value) Update the value of the key if the key exists. Otherwise, add the key-value pair to the cache. If the number of keys exceeds the capacity from this operation, evict the least recently used key.
# // The functions get and put must each run in O(1) average time complexity.
# // Example 1:
# // Input
# // ["LRUCache", "put", "put", "get", "put", "get", "put", "get", "get", "get"]
# // [[2], [1, 1], [2, 2], [1], [3, 3], [2], [4, 4], [1], [3], [4]]
# // Output
# // [null, null, null, 1, null, -1, null, -1, 3, 4]
# // Explanation
# // LRUCache lRUCache = new LRUCache(2);
# // lRUCache.put(1, 1); // cache is {1=1}
# // lRUCache.put(2, 2); // cache is {1=1, 2=2}
# // lRUCache.get(1); // return 1
# // lRUCache.put(3, 3); // LRU key was 2, evicts key 2, cache is {1=1, 3=3}
# // lRUCache.get(2); // returns -1 (not found)
# // lRUCache.put(4, 4); // LRU key was 1, evicts key 1, cache is {4=4, 3=3}
# // lRUCache.get(1); // return -1 (not found)
# // lRUCache.get(3); // return 3
# // lRUCache.get(4); // return 4
# // Constraints:
# // 1 <= capacity <= 3000
# // 0 <= key <= 104
# // 0 <= value <= 105
# // At most 2 * 105 calls will be made to get and put.
# [1, 2] <- 3
# [2,3]
# [3,2]
# [1, 3, 4 , 2]
class LinkedListNode:
def __init__ ( self , key, val) :
self .key = key
self .val = val
self .prev = None
self .next = None
def updateValue( self , val) :
self .val = val
class LinkedList:
def __init__ ( self ) :
self .llHead = LinkedListNode( -1 , -1 )
self .llTail = LinkedListNode( -1 , -1 )
self .llHead .next = self .llTail
self .llTail .prev = self .llHead
def getHead( self ) :
return self .llHead .next if self .llHead .next != self .llTail else None
def addValue( self , key, val) :
node = LinkedListNode( key, val)
self .addNodeToEnd ( node)
def addNodeToEnd( self , node) :
currentTail = self .llTail .prev
# handle next
self .llTail .prev = node
node.next = self .llTail
# handle prev
currentTail.next = node
node.prev = currentTail
def moveNodeToEnd( self , node) :
self .deleteNodeFromList ( node)
self .addNodeToEnd ( node)
def deleteNodeFromList( self , node) :
nodePrev = node.prev
nodeNext = node.next
nodePrev.next = nodeNext
nodeNext.prev = nodePrev
node.prev = None
node.next = None
class LRU:
def __init__ ( self , capacity) :
self .capacity = capacity
self .linkedList = LinkedList( )
self .nodesMap = dict ( )
def put( self , key, value) :
if key in self .nodesMap :
node = self .nodesMap .get ( key)
node.updateValue ( value)
self .linkedList .moveNodeToEnd ( node)
else :
if len ( self .nodesMap ) == self .capacity :
lrNode = self .linkedList .getHead ( )
self .linkedList .deleteNodeFromList ( lrNode)
self .nodesMap .pop ( lrNode.key )
newNode = LinkedListNode( key, value)
self .nodesMap [ key] = newNode
self .linkedList .addNodeToEnd ( newNode)
def get( self , key) :
if key not in self .nodesMap :
return -1
node = self .nodesMap [ key]
self .linkedList .moveNodeToEnd ( node)
return node.val
# # ["LRUCache", "put", "put", "get", "put", "get", "put", "get", "get", "get"]
# # // [[2], [1, 1], [2, 2], [1], [3, 3], [2], [4, 4], [1], [3], [4]]
# if __name__ == '__main__':
# cache = LRU(2)
# cache.put(1, 1)
# cache.put(2, 2)
# print(cache.get(1))
# cache.put(3,3)
# print(cache.get(2))
# cache.put(4, 4)
# print(cache.get(1))
# print(cache.get(3))
# print(cache.get(4))
# There are a total of numCourses courses you have to take, labeled from 0 to numCourses - 1. You are given an array prerequisites where prerequisites[i] = [ai, bi] indicates that you must take course bi first if you want to take course ai.
# For example, the pair [0, 1], indicates that to take course 0 you have to first take course 1.
# Return the ordering of courses you should take to finish all courses. If there are many valid answers, return any of them. If it is impossible to finish all courses, return an empty array.
# Example 1:
# Input: numCourses = 2, prerequisites = [[1,0]]
# Output: [0,1]
# Explanation: There are a total of 2 courses to take. To take course 1 you should have finished course 0. So the correct course order is [0,1].
# Example 2:
# Input: numCourses = 4, prerequisites = [[1,0],[2,0],[3,1],[3,2]]
# Output: [0,2,1,3]
# Explanation: There are a total of 4 courses to take. To take course 3 you should have finished both courses 1 and 2. Both courses 1 and 2 should be taken after you finished course 0.
# So one correct course order is [0,1,2,3]. Another correct ordering is [0,2,1,3].
# Example 3:
# Input: numCourses = 1, prerequisites = []
# Output: [0]
# Constraints:
# 1 <= numCourses <= 2000
# 0 <= prerequisites.length <= numCourses * (numCourses - 1)
# prerequisites[i].length == 2
# 0 <= ai, bi < numCourses
# ai != bi
# All the pairs [ai, bi] are distinct.
# [0, 1]
# 0<-1
# class GraphNode:
# def __init__(self, val):
# self.val = val
# space - O(max(courses, relationships))
# time - O(courses + relationships)
def getCoursesOrder( courses, courseRelationship) :
independentCourse = [ ] # O(courses)
dependencyCounter = dict ( ) # O(courses)
depMap = dict ( ) # O(relationships)
for dep in courseRelationship: # O(relationships)
c1, c2 = dep
if c1 not in dependencyCounter:
dependencyCounter[ c1] = 0
dependencyCounter[ c1] += 1
if c2 not in depMap:
depMap[ c2] = [ ]
depMap[ c2] .append ( c1)
for i in range ( courses) : # O(courses)
if dependencyCounter.get ( i, 0 ) == 0 :
independentCourse.append ( i)
cp = 0
result = [ ] # O(courses)
while cp < len ( independentCourse) :
currentCourse = independentCourse[ cp]
result.append ( currentCourse)
dependentCourses = depMap.get ( currentCourse, [ ] )
for course in dependentCourses: # O(relationships)
dependencyCounter[ course] -= 1
if dependencyCounter[ course] == 0 :
independentCourse.append ( course)
cp += 1
return result
if __name__ == '__main__' :
result = getCoursesOrder( 4 , [ [ 0 , 1 ] , [ 2 , 0 ] , [ 3 , 1 ] , [ 3 , 2 ] ] )
print ( result)
# [[1,0],[2,0],[3,1],[3,2]]
# [0]
# {1:0, 2:0, 3:0}
# {0:[1, 2], 1:[3], 2:[3]}
# [0, 1, 2, 3]
# result [0, 1, 2, 3]
# // Design a data structure that follows the constraints of a Least Recently Used (LRU) cache.

# // Implement the LRUCache class:

# // LRUCache(int capacity) Initialize the LRU cache with positive size capacity.
# // int get(int key) Return the value of the key if the key exists, otherwise return -1.
# // void put(int key, int value) Update the value of the key if the key exists. Otherwise, add the key-value pair to the cache. If the number of keys exceeds the capacity from this operation, evict the least recently used key.
# // The functions get and put must each run in O(1) average time complexity.

 

# // Example 1:

# // Input
# // ["LRUCache", "put", "put", "get", "put", "get", "put", "get", "get", "get"]
# // [[2], [1, 1], [2, 2], [1], [3, 3], [2], [4, 4], [1], [3], [4]]
# // Output
# // [null, null, null, 1, null, -1, null, -1, 3, 4]

# // Explanation
# // LRUCache lRUCache = new LRUCache(2);
# // lRUCache.put(1, 1); // cache is {1=1}
# // lRUCache.put(2, 2); // cache is {1=1, 2=2}
# // lRUCache.get(1);    // return 1
# // lRUCache.put(3, 3); // LRU key was 2, evicts key 2, cache is {1=1, 3=3}
# // lRUCache.get(2);    // returns -1 (not found)
# // lRUCache.put(4, 4); // LRU key was 1, evicts key 1, cache is {4=4, 3=3}
# // lRUCache.get(1);    // return -1 (not found)
# // lRUCache.get(3);    // return 3
# // lRUCache.get(4);    // return 4
 

# // Constraints:

# // 1 <= capacity <= 3000
# // 0 <= key <= 104
# // 0 <= value <= 105
# // At most 2 * 105 calls will be made to get and put.


# [1, 2] <- 3
# [2,3]
# [3,2]
# [1, 3, 4 , 2] 

class LinkedListNode:
    def __init__(self, key, val):
        self.key = key
        self.val = val
        self.prev = None
        self.next = None

    def updateValue(self, val):
        self.val = val

class LinkedList:
    def __init__(self):
        self.llHead = LinkedListNode(-1, -1)
        self.llTail = LinkedListNode(-1, -1)
        self.llHead.next = self.llTail
        self.llTail.prev = self.llHead


    def getHead(self):
        return self.llHead.next if self.llHead.next != self.llTail else None
    
    def addValue(self, key, val):
        node = LinkedListNode(key, val)
        self.addNodeToEnd(node)

    def addNodeToEnd(self, node):
        currentTail = self.llTail.prev
        # handle next
        self.llTail.prev = node
        node.next = self.llTail
        # handle prev
        currentTail.next = node
        node.prev = currentTail

    
    def moveNodeToEnd(self, node):
        self.deleteNodeFromList(node)
        self.addNodeToEnd(node)


    def deleteNodeFromList(self, node):
        nodePrev = node.prev
        nodeNext = node.next
        nodePrev.next = nodeNext
        nodeNext.prev = nodePrev
        node.prev = None
        node.next = None
        
    

class LRU:
    def __init__(self, capacity):
        self.capacity = capacity
        self.linkedList = LinkedList()
        self.nodesMap = dict()

    def put(self, key, value):
        if key in self.nodesMap:
            node = self.nodesMap.get(key)
            node.updateValue(value)
            self.linkedList.moveNodeToEnd(node)
        else:
            if len(self.nodesMap) == self.capacity:
                lrNode = self.linkedList.getHead()
                self.linkedList.deleteNodeFromList(lrNode)
                self.nodesMap.pop(lrNode.key)
            newNode = LinkedListNode(key, value)
            self.nodesMap[key] = newNode
            self.linkedList.addNodeToEnd(newNode)

    
    def get(self, key):
        if key not in self.nodesMap:
            return -1
        node = self.nodesMap[key]
        self.linkedList.moveNodeToEnd(node)
        return node.val

# # ["LRUCache", "put", "put", "get", "put", "get", "put", "get", "get", "get"]
# # // [[2], [1, 1], [2, 2], [1], [3, 3], [2], [4, 4], [1], [3], [4]]
# if __name__ == '__main__':
#     cache = LRU(2)
#     cache.put(1, 1)
#     cache.put(2, 2)
#     print(cache.get(1))
#     cache.put(3,3)
#     print(cache.get(2))
#     cache.put(4, 4)
#     print(cache.get(1))
#     print(cache.get(3))
#     print(cache.get(4))

    
# There are a total of numCourses courses you have to take, labeled from 0 to numCourses - 1. You are given an array prerequisites where prerequisites[i] = [ai, bi] indicates that you must take course bi first if you want to take course ai.

# For example, the pair [0, 1], indicates that to take course 0 you have to first take course 1.
# Return the ordering of courses you should take to finish all courses. If there are many valid answers, return any of them. If it is impossible to finish all courses, return an empty array.

 

# Example 1:

# Input: numCourses = 2, prerequisites = [[1,0]]
# Output: [0,1]
# Explanation: There are a total of 2 courses to take. To take course 1 you should have finished course 0. So the correct course order is [0,1].
# Example 2:

# Input: numCourses = 4, prerequisites = [[1,0],[2,0],[3,1],[3,2]]
# Output: [0,2,1,3]
# Explanation: There are a total of 4 courses to take. To take course 3 you should have finished both courses 1 and 2. Both courses 1 and 2 should be taken after you finished course 0.
# So one correct course order is [0,1,2,3]. Another correct ordering is [0,2,1,3].
# Example 3:

# Input: numCourses = 1, prerequisites = []
# Output: [0]
 

# Constraints:

# 1 <= numCourses <= 2000
# 0 <= prerequisites.length <= numCourses * (numCourses - 1)
# prerequisites[i].length == 2
# 0 <= ai, bi < numCourses
# ai != bi
# All the pairs [ai, bi] are distinct.

# [0, 1]
# 0<-1

# class GraphNode:
#     def __init__(self, val):
#         self.val = val

# space  - O(max(courses, relationships))
# time - O(courses + relationships)
def getCoursesOrder(courses, courseRelationship):
    independentCourse = [] # O(courses)
    dependencyCounter = dict() # O(courses)
    depMap = dict() # O(relationships)

    for dep in courseRelationship: # O(relationships)
        c1, c2 = dep
        if c1 not in dependencyCounter:
            dependencyCounter[c1] = 0
        dependencyCounter[c1] += 1
        if c2 not in depMap:
            depMap[c2] = []
        depMap[c2].append(c1)
    
    
    for i in range(courses): # O(courses)
        if dependencyCounter.get(i, 0) == 0:
            independentCourse.append(i)
    cp = 0
    result = [] # O(courses)

    while cp < len(independentCourse):
        currentCourse = independentCourse[cp]
        result.append(currentCourse)
        dependentCourses = depMap.get(currentCourse, [])
        for course in dependentCourses: # O(relationships)
            dependencyCounter[course] -= 1
            if dependencyCounter[course] == 0:
                independentCourse.append(course)
        cp += 1
    return result

if __name__ == '__main__':
    result = getCoursesOrder(4, [[0,1],[2,0],[3,1],[3,2]])
    print(result)

# [[1,0],[2,0],[3,1],[3,2]]
# [0] 
# {1:0, 2:0, 3:0}
# {0:[1, 2], 1:[3], 2:[3]}

# [0, 1, 2, 3]
# result [0, 1, 2, 3]

            








