Welcome to today's lesson! We're exploring how to introduce complex features while maintaining backward compatibility in software systems. Our journey will take us through a Potluck Dinner organization system where participants and their dishes are tracked for each round. Get ready to dive into Kotlin programming, unveiling strategies and crafting solutions. Let's embark on this exciting adventure!
Our Potluck Dinner organization system currently lets us add and remove participants and manage their dishes for each round. Here are the essential functions:
fun addParticipant(memberId: String): Boolean
: Adds a participant if they don't already exist. Returnstrue
if successful, orfalse
if the participant is already in the system.fun removeParticipant(memberId: String): Boolean
: Removes a participant and their dish if they exist. Returnstrue
if successful,false
otherwise.fun addDish(memberId: String, dishName: String): Boolean
: Allows a participant to add a dish. Returnstrue
if added successfully,false
if the participant is invalid or has already added a dish.
Let's implement these methods using Kotlin, following the initial state of our system:
Kotlin1class Potluck { 2 private val participants = mutableSetOf<String>() 3 private val dishes = mutableMapOf<String, String>() 4 5 fun addParticipant(memberId: String): Boolean { 6 return if (participants.contains(memberId)) { 7 false 8 } else { 9 participants.add(memberId) 10 true 11 } 12 } 13 14 fun removeParticipant(memberId: String): Boolean { 15 return if (!participants.contains(memberId)) { 16 false 17 } else { 18 participants.remove(memberId) 19 dishes.remove(memberId) 20 true 21 } 22 } 23 24 fun addDish(memberId: String, dishName: String): Boolean { 25 return if (!participants.contains(memberId) || dishes.containsKey(memberId)) { 26 false 27 } else { 28 dishes[memberId] = dishName 29 true 30 } 31 } 32}
With this setup, we use Kotlin's mutableSetOf
for unique participant IDs and mutableMapOf
to associate dishes with participant IDs. Now, let's extend our system with advanced features.
Currently, our Potluck Dinner organization system is simple yet effective. To enhance its capabilities, we're introducing a "Dish of the Day" feature. This allows participants to vote for a dish, with the most voted dish earning the title "Dish of the Day."
The new functionalities introduce these methods:
fun vote(memberId: String, voteId: String): Boolean
: Allows a participant to cast a vote. Each participant can vote only once per round. Returnsfalse
if the participant or vote action is invalid.fun dishOfTheDay(): String?
: Calculates and returns the "Dish of the Day." In cases of ties, the earliest joining participant's dish is prioritized. Returnsnull
if no votes exist.
To accommodate these features, we'll modify our class:
Kotlin1class Potluck { 2 private val participants = mutableMapOf<String, Long>() 3 private val dishes = mutableMapOf<String, String>() 4 private val votes = mutableMapOf<String, String>() 5}
Kotlin's mutableMapOf
efficiently suits our need to track participant IDs with their join time.
Kotlin1fun vote(memberId: String, voteId: String): Boolean { 2 return if (participants.containsKey(memberId) && !votes.containsKey(memberId)) { 3 votes[memberId] = voteId 4 true 5 } else { 6 false 7 } 8}
The vote
function confirms the existence of memberId
and checks whether a vote has been cast. If valid, it stores the vote; otherwise, it returns false
.
Kotlin1fun dishOfTheDay(): String? { 2 if (votes.isEmpty()) return null 3 4 val voteCount = votes.values.groupingBy { it }.eachCount() 5 val maxVotes = voteCount.values.maxOrNull() ?: 0 6 7 val maxVoteDishes = voteCount.filter { it.value == maxVotes }.keys 8 val earliestJoinTime = participants.filterKeys { it in maxVoteDishes } 9 .minByOrNull { it.value }?.key ?: return null 10 11 return "Participant: '$earliestJoinTime', Dish: '${dishes[earliestJoinTime]}'" 12}
Here, Kotlin's grouping and filtering methods elegantly compute vote counts and resolve ties while maintaining readability and leveraging Kotlin's safety features.
The addParticipant
function requires modification to integrate join times, maintaining our system's functionality:
Kotlin1fun addParticipant(memberId: String): Boolean { 2 return if (participants.containsKey(memberId)) { 3 false 4 } else { 5 participants[memberId] = System.currentTimeMillis() 6 true 7 } 8}
Updating it ensures join times are tracked for our new features.
Ensure the removeParticipant
function is comprehensive to maintain data integrity:
Kotlin1fun removeParticipant(memberId: String): Boolean { 2 return if (!participants.containsKey(memberId)) { 3 false 4 } else { 5 participants.remove(memberId) 6 dishes.remove(memberId) 7 votes.remove(memberId) 8 true 9 } 10}
Maintaining correctness across interactions remains pivotal, so addDish
functions are similarly concise:
Kotlin1fun addDish(memberId: String, dishName: String): Boolean { 2 return if (!participants.containsKey(memberId) || dishes.containsKey(memberId)) { 3 false 4 } else { 5 dishes[memberId] = dishName 6 true 7 } 8}
Here is your final Potluck implementation in Kotlin:
Kotlin1class Potluck { 2 private val participants = mutableMapOf<String, Long>() 3 private val dishes = mutableMapOf<String, String>() 4 private val votes = mutableMapOf<String, String>() 5 6 fun addParticipant(memberId: String): Boolean { 7 return if (participants.containsKey(memberId)) { 8 false 9 } else { 10 participants[memberId] = System.currentTimeMillis() 11 true 12 } 13 } 14 15 fun removeParticipant(memberId: String): Boolean { 16 return if (!participants.containsKey(memberId)) { 17 false 18 } else { 19 participants.remove(memberId) 20 dishes.remove(memberId) 21 votes.remove(memberId) 22 true 23 } 24 } 25 26 fun addDish(memberId: String, dishName: String): Boolean { 27 return if (!participants.containsKey(memberId) || dishes.containsKey(memberId)) { 28 false 29 } else { 30 dishes[memberId] = dishName 31 true 32 } 33 } 34 35 fun vote(memberId: String, voteId: String): Boolean { 36 return if (participants.containsKey(memberId) && !votes.containsKey(memberId)) { 37 votes[memberId] = voteId 38 true 39 } else { 40 false 41 } 42 } 43 44 fun dishOfTheDay(): String? { 45 if (votes.isEmpty()) return null 46 47 val voteCount = votes.values.groupingBy { it }.eachCount() 48 val maxVotes = voteCount.values.maxOrNull() ?: 0 49 50 val maxVoteDishes = voteCount.filter { it.value == maxVotes }.keys 51 val earliestJoinTime = participants.filterKeys { it in maxVoteDishes } 52 .minByOrNull { it.value }?.key ?: return null 53 54 return "Participant: '$earliestJoinTime', Dish: '${dishes[earliestJoinTime]}'" 55 } 56}
Congratulations on integrating complex functionalities within Kotlin while ensuring backward compatibility! This showcases how Kotlin's concise syntax and modern features can efficiently solve real-world engineering problems. Continue to hone these skills, and explore more Kotlin adventures ahead! Happy coding!