You can find our solution here. Some highlights:
In get_movies_by_director() we use csv.DictReader to parse the csv file:
with open(MOVIE_DATA) as f: for line in csv.DictReader(f): ...
We use a defaultdict(list) for our initial parsing of movies:
m = Movie(title=movie, year=year, score=score) directors[director].append(m)
get_average_scores() returns a directors dict via a dict comprehension (note the subtle refactoring, glad we had our tests), where keys are (director, mean score) and values their movies. We only take directors with at least MIN_MOVIES.
Although Python3 has statistics.mean we rolled our own (_calc_mean) because we first extract the score from the Movie namedtupe and added rounding (although that should maybe go in the print_results function).
Next time we would use statistics.mean, because the more you leverage the stdlib the better. These considerations happen when coding, reviewing your and others code, good learning.
Although movies should not be 0 we are defensive by never allowing the denominator to be 0:
mean = sum(ratings) / max(1, len(ratings))
You could also write:
mean = sum(ratings) / len(ratings) if ratings else 0
print_results() then prints the desired output. The enumerate is handy to get the sequence numbers for the top NUM_TOP_DIRECTORS directors. You can give it a start with a 2nd argument, 1 in this case.
We used zfill before to print 01 / 02 etc., but found out that you can tackle this in the format syntax. We might refactor all these formats to use F-string, it would make this a lot cleaner :)
Again it was nice to work on this code having tests:
$ python test_directors.py tests pass
We realized during the challenge that you could also solve this with SQL or Pandas. We will update the solutions branch when we get around this. It would be nice to give it a try. Of course if you took these (or other) approaches feel free to share your solution opening a PR against our community branch.
Here is another solution by atakume we merged in our community branch. What we like about this solution is the use of itertools.groupby, doing the sorting outside the print_results function, and the second namedtuple which adds readability:
Filmography = namedtuple('Filmography', 'director movies avg_score')
Next week we will let you play with decorators, a great feature for writing DRY, reusable code. It will be fun :)
Do you want to get 250+ concise and applicable Python tips in an ebook that will cost you less than 10 bucks (future updates included), check it out here.
"The discussions are succinct yet thorough enough to give you a solid grasp of the particular problem. I just wish I would have had this book when I started learning Python." - Daniel H
"Bob and Julian are the masters at aggregating these small snippets of code that can really make certain aspects of coding easier." - Jesse B
"This is now my favourite first Python go-to reference." - Anthony L
"Do you ever go on one of those cooking websites for a recipe and have to scroll for what feels like an eternity to get to the ingredients and the 4 steps the recipe actually takes? This is the opposite of that." - Sergio S