Your challenge is to design and implement a Least Recently Used (LRU) cache data structure that supports the following operations with optimal time complexity (preferably O(1) for each operation):
get(key): Retrieve the value associated with the key if it exists in the cache; otherwise, return -1.
put(key, value): Insert or update the value associated with the key. If the cache exceeds its predetermined capacity after the insertion, it should invalidate and remove the least recently used item.
Requirements:
Implement your solution in the programming language of your choice and include tests or examples that demonstrate the correctness of your implementation.