Refactor Your Code
00:00 I’ve got a chance to make my point about mutable lists and we have a working program that even fixes a little bug that was in there that you may or may not have encountered because maybe you’ll just always create one poem and you would’ve never run into it.
00:14 But in this lesson, you are going to look at one of the most important and somewhat final steps of when you’re writing a program, which is think about what you did and refactor to make it more coherent and simpler, more understandable and better readable.
00:33 So I was playing around with different ways of picking out these random words from a collection, but if you’re actually going to write this program, it would be probably best to just stick with one.
00:44 Exploring these different ways of doing it gave me a chance to try out a couple of ways and find one that works maybe the best and that I want to commit to.
00:52
And in this case, I would say that using random.sample()
seems like the best approach. It’s straightforward, it’s very flexible. I can pick different amounts.
01:04
They’re always going to be unique and I just need to change the argument to k
. So this seems really nice. I don’t need to shuffle, I don’t need to pop, I don’t need to remove anything.
01:13
And this actually also takes care of the issue with mutability of lists because random.sample()
just picks out of the list. It doesn’t actually mutate it.
01:23
So in this step I want to go ahead and refactor all of my random word picking to use random.sample()
. Let’s go ahead and do that right away. I want to do random.sample()
.
01:35
Also for adjectives, I need to pick out of the adjectives list, which sits at index two. And I want to get three random adjectives. I’m going to change k
to three.
01:48
I’ll copy that again so I can reuse it. Then I can get rid of this shuffling. For example, up here, I can get rid of the whole for
loop.
01:58 I’m picking out from words at index one and I want to get three verbs. You can see like I can refactor all of this code that I have here to one line each.
02:10 All the steps are similar.
02:14
I don’t even need all these code comments anymore at this point. I feel like I can completely get rid of them because the code is quite self-explanatory. I’m using the random
module, I’m using the sample()
function, and it tells me how many I’m picking out by just looking at the k
parameter over there.
02:34
All right. I will stick with doing adverb random.choice()
instead of random.sample()
because I know I just need one. So that’s also like more descriptive here that I’m only going to pick one and I don’t need to deal with the nested list afterwards.
02:48 One more thing this allows me to do, because I’m not mutating the lists anymore. I can also get rid of needing to deep copy the words to pull. I can just delete that, don’t have to worry about it and get rid of an additional import that I’m not using anymore.
03:06 As you can see, like this made the code quite a bit shorter. So now the logic is down to one block that is picking the random words and then another block deciding which article to use.
03:18 And then finally, I’m just assembling the poem. So I have like these three steps. Let’s give the first one also a comment just for consistency here. Randomly pick the words to use
03:32
and I want to decide whether to use "A"
or "An"
as the article. And then I want to create the poem.
03:39
And by refactoring the part about picking the words to use, we cut down a lot of lines of code. We made everything work similarly except like one time we use choice()
instead of sample()
, but it makes it more straightforward to keep an overview of what you’re doing in here.
03:54 Less code to read is also easier to maintain, less imports. It’s going to make your script faster. No need to do that deep copy. It’s also going to make the script faster.
04:05 Let’s make sure that it still works by running the code
04:09 and creating all the ten poems. And you can see, you’re not running into an error here. One last thing I want to do is to make these poems a little easier to read.
04:22 I’m going to add a new line at the beginning and one at the end. That’s just going to be part of the poetic structure of this construct here. Let me run that again and you can see it’s a bit easier to read.
04:36 Now we have some spacing in between the different poems.
04:40 All right, this looks better. And this is, in a way, you could think of it as an unnecessary step because the poem generator was working before. We didn’t change anything about the output, but you change some things about the internals, which makes it easier to read faster and less complex.
04:59 All of these things are super important for writing code. I like to have this last step of looking at the code again, figuring out what did you learn in the process of writing it, and then apply the best thing that you’ve found.
05:13 There may be other ways of doing it that are even better, but in your process, you found out one way that works the best, and then you just want to apply that one consistently and clean up your code to have a nice script that you can also imagine maintaining into the future.
05:29 Thanks for joining me also in this refactoring trip at the end. In the next lesson, let’s talk about a couple of takeaways and then wrap up this course.
roshny on Oct. 17, 2024
😊
from random import choice
from random import sample
nouns = ["badger", "horse", "aardvark", "mouse", "gorilla", "elephant", "eagle", "sparrow"]
verbs = ["hugs", "bounces", "meows", "hauls", "whispers", "flutters", "gallops", "shimmers"]
adjectives = ["furry", "incredulous", "fragrant", "exuberant", "glistening", "melancholic", "serene"]
prepositions = ["against", "after", "into", "beneath", "upon", "for", "in", "like", "over", "within"]
adverbs = ["curiously", "extravagantly", "graciously", "reluctantly", "meticulously", "vigorously"]
def get_article(adj):
return "An" if adj[0] in "aeiou" else "A"
def make_poem():
noun1, noun2, noun3 = sample(nouns, k=3)
verb1, verb2, verb3 = sample(verbs, k=3)
adj1, adj2, adj3 = sample(adjectives, k=3)
prep1, prep2 = sample(prepositions, k=2)
adverb1 = choice(adverbs)
poem = f"""{get_article(adj1)} {adj1} {noun1}
{get_article(adj1)} {adj1} {noun1} {verb1} {prep1} the {adj2} {noun2}
{adverb1}, the {noun1} {verb2}
the {noun2} {verb3} {prep2} the {adj3} {noun3}"""
return poem
print(make_poem())
Pankaj Poddar on Nov. 20, 2024
import random
nouns = ["badger", "horse", "aardvark", "mouse", "gorilla", "elephant", "eagle", "sparrow"]
verbs = ["hugs", "bounces", "meows", "hauls", "whispers", "flutters", "gallops", "shimmers"]
adjectives = ["furry", "incredulous", "fragrant", "exuberant", "glistening", "melancholic", "serene"]
prepositions = ["against", "after", "into", "beneath", "upon", "for", "in", "like", "over", "within"]
adverbs = ["curiously", "extravagantly", "graciously", "reluctantly", "meticulously", "vigorously"]
def noun_selection(list_of_nouns):
return random.sample(list_of_nouns, k=3)
def verb_selection(list_of_verbs):
return random.sample(list_of_verbs, k=3)
def adj_selection(list_of_adj):
return random.sample(list_of_adj, k=3)
def prep_selection(list_of_prep):
return random.sample(list_of_prep, k=2)
def adverb_selection(list_of_adv):
return random.choice(list_of_adv)
def article_selection(adj):
if adj[0] in "aeiou":
return "An"
return "A"
def poem():
poetry = f"""
{article} {adj1} {noun1} {verb1} {prep1} the {adj2} {noun2} {adverb1},
the {noun1} {verb2} the {noun2} {verb3} {prep2} the {adj3} {noun3}.
"""
return poetry
print(f"❤️ Your Poem ❤️")
for _ in range(3):
noun1, noun2, noun3 = noun_selection(nouns)
verb1, verb2, verb3 = verb_selection(verbs)
adj1, adj2, adj3 = adj_selection(adjectives)
prep1, prep2 = prep_selection(prepositions)
adverb1 = adverb_selection(adverbs)
article = article_selection(adj1)
heading = f"{article} {adj1} {noun1}"
print(heading)
phrase = poem()
print(phrase)
print("👌")
Colin on Jan. 2, 2025
I was quite keen to automate how many values to extract per word set. Not sure if it is bad practice to nest functions like this within each function.
words = ((nouns, 3), (verbs, 3), (adjectives, 3), (prepositions, 2), (adverbs, 1))
def determine_article(adj):
if adj[0] in "aoeiuy":
article = "An"
else:
article = "A"
return article
def randomise_words(word, n):
rand_word = random.sample(word, k=n)
return rand_word
def poetry_generator():
rnoun, rverb, radj, rprep, radv = [randomise_words(word, n) for word, n in words]
rart = determine_article(radj)
poem = (
f"{rart} {radj[0]} {rnoun[0]}\n\n"
f"{rart} {radj[0]} {rnoun[0]} {rprep[0]} the {radj[1]} {rnoun[1]}\n"
f"{radv[0]}, the {rnoun[1]} {rverb[2]}\n"
f"the {rnoun[1]} {rverb[2]} {rprep[1]} the {radj[2]} {rnoun[2]}\n"
)
return poem
print(poetry_generator())
I liked the tip about using multiple f strings to keep the f string readable, I did not know how to get around that when working on my solution.
Martin Breuss RP Team on Jan. 7, 2025
@Colin by “nesting functions in functions”, do you mean how you’re calling randomise_words()
and determine_article()
inside of poetry_generator()
?
If so, that’s not bad practice at all! This is a great modular way of writing your code where you define a function with a single responsibility, then call it wherever you need that functionality :)
Become a Member to join the conversation.
doink on June 2, 2024
Thank you Martin for guidance.
This is my solution: