Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Snow Leopards C18 -Geiselle H. and Bukunmi G. #77

Open
wants to merge 7 commits into
base: master
Choose a base branch
from

Conversation

geiselleholt
Copy link

No description provided.

Copy link

@anselrognlie anselrognlie left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice job!

All your tests are passing and your code is laid out well overall. There are a couple of functions that are working for the tests, but which would not work for more general cases. Check the comments for details. Nice use of using some built-ins as well your own approaches. But make sure you pay attention to any notes in the documentation about usage (sample is being used in a way that is causing some warnings). And of course, we always recommend rolling your own approach to practice breaking down problems on your own.

def draw_letters():
pass

LETTER_POOL = {

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since you used sample to draw your hand (which is non-destructive), this definition of LETTER_POOL could be moved outside the function to be global (to the module). This allows us to focus on the logic within the function (which is a single line in this case), rather than having the body of the function be dominated by the data definition.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

When using strings as the dictionary keys, I like to the use the dict() function-stlye of making a dictionary. It uses keyword argument calling to let us set string keys without needing to quote them. For example

    LETTER_POOL = dict(
        A=9,
        B=2,
        C=2,
        ...
    )

'Z': 1
}

letter_bank = sample(LETTER_POOL.keys(), k=10, counts=LETTER_POOL.values())

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Really nice, concise approach. When bringing in functions from imported modules (that we haven't covered in the curriculum), please consider including a comment about how you understand the function to work, and how you reasearched it. Now that we've discussed Big O, it's also useful to think about what the time/space complexity of the function might be. Thinking about how we can implement this functionality ourselves can help us gain insight into what that might be.

Remember that not every language has the same built-in functions, so we should be able to implement this same functionality ourselves. In the projects, we don't ask for any features that we don't think you should be able to write yourself. For drawing a random hand, the only external function that would be "required" is a way to pick a random number (such as randint). At this stage in our coding journeys, it's often more valuable to re-implement things by hand to practice thinking about how to breakdown and approach the problem than to use a single library function to solve the problem for us.

Be careful to note that the way this is being called is resulting in a warning in the tests:

tests/test_wave_01.py: 1002 warnings
  adagrams/game.py:36: DeprecationWarning: Sampling from a set deprecated
  since Python 3.9 and will be removed in a subsequent version.
    letter_bank = sample(LETTER_POOL.keys(), k=10, counts=LETTER_POOL.values())

Comment on lines +45 to +47
for letter in word:
if letter not in letter_bank:
return False

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This handles the case of checking for word letters that aren't in the bank at all. Note that the not in conceals a hidden loop, since we can only check whether something isn't in a list by checking every value in the list. Doing that in a loop can start us down the road towards quadratic time complexity. For adagrams, we do have some bounds on the data size that make this less of a concern, but think about the general case where the letter bank might have n tiles, and the word might be up to m letters long. If we assume that in the worst case, the word could have the same number of letters as the hand has tiles, this truly does approach O(n^2) time complexity.

Could this checking be combined with the general checking being done later on?

count_word = Counter(word)
count_letter_bank = Counter(letter_bank)

for k, v in count_word.items():

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Consider more descriptive variable names. Maybe word_letter and word_letter_count?

count_letter_bank = Counter(letter_bank)

for k, v in count_word.items():
if k in count_letter_bank:

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Here as well, bank_letter?

Comment on lines +91 to +95
for letter in word:
if letter in LETTERS_VALUES:
word_value += LETTERS_VALUES[letter]
if len(word) > 6:
word_value += 8

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👍 Nice job tallying up the score and adding the bonus.

You did a check to see whether a letter was in score table. This is fine and has the affect that any unexpected tiles in the word don't contribute to the score. But where would those unexpected tiles have come from? And should it be the scoring function's job to handle those? We might expect some other part of the game to validate that any words played by a player include only tiles from the tile bank, in which case this function could safely ignore the in check. In fact, we might want to let python raise a KeyError so that the program can find out that, somehow, bad data got into our word.

Comment on lines +100 to +105
for word in word_list:
score = score_word(word)
score_list.append(score)

word_score_tuples = list(zip(word_list, score_list))

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This would be a great opportunity for a list comprehension:

    word_score_tuples = [(word, score_word(word)) for word in word_list]

word_score_tuples = list(zip(word_list, score_list))

sorted_tuples = sorted(word_score_tuples, key=lambda t: -t[1])

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This will sort the word/score tuples in descending order, leaving the highest scores at the front of the list. Recall that sort/sorted must be considered O(n log n). So while this can be a step in how we find the winning word, keep in mind that there are O(n) (linear) time complexity approaches as well.

winning_word = sorted_tuples[0]
if sorted_tuples[0][1] == sorted_tuples[1][1]:
if len(sorted_tuples[0][0]) == 10:

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

One drawback to packing the words and scores into tuples from the beginning is that we have a lot of indexing in the calculations, which can be difficult to read. We could make some constants to make things a little more readable

    # define earlier in code
    WORD = 0
    SCORE = 1

    # then we can write
    sorted_tuples[0][WORD] # to get the word from the tuple
    sorted_tuples[0][SCORE] # to get the score from the tuple

sorted_tuples = sorted(word_score_tuples, key=lambda t: -t[1])
winning_word = sorted_tuples[0]
if sorted_tuples[0][1] == sorted_tuples[1][1]:

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This works for the test data, because there are never more than two words that are tied for the hieghest score, but this would not work if there were 3 or more words tied (e.g., ["BBBBBB", "BBBBBB", "ZX"]). This would also fail if there were only a single word in the word list (solitaire adagrams).

While passing all the tests is great, the tests are not (and can never truly be) exhaustive. Be sure you understand the general situation a test is looking for, and write your code to handle that general case, not just the specific case of the particular test. You can always add your own additional tests to check that your implementation is robust to a wider variety of data.

How could we modify this approach so that we process as many words as necessary (2, 3, 4, ...), and don't crash on a single word.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants