👋 In this repository, you will be following my progress through the course. I got the idea of doing it a few days ago so this is why there are some folders that were already created.
Degrees : 📖
I was provided with two folders one containing a set of a large data sets and another for small data sets. The goal is to find the shortest_path
from the source
's id to the target
's id. I go on to implement this by defining a starting_node
, visited
(empty list) , queue
list with the starting_node
as the first element and the shortest_path
being an empty list. Then in the while loop, we first check the neighbours of the first node in the lights of finding a relation to the target
. If nothing is found, the neighbours become nodes that aren't explored yet and they get added to the queue to be explored. Keeping track of the length of the shortest path to get to the target and the nodes that already been visited in visited
.
Tic-Tac-Toe : 📖
In this project, the Minimax algorithm used, goes over all the future actions possible calculating the min_value
or the max_value
based on the AI's position. The algorithm chooses the highest or lowest scores relatively.
The higher (or lower) the score the more win outcomes possible derive from the move.
Knights : 📖
Given a way to express logical sentences, I was asked to solve four puzzles given.
Minesweeper : 📖
In this project, we had to write an AI agent that can gather knowledge about the minesweeper board and automatically select cells that are known to be safe.
PageRank : 📖
For this one, I implemented a crawl()
function that parses a directory of HTML pages and checks for links to other pages using a regex that I created. This function will return a dict
where each key
is a page and their values are a list of all the other pages in the corpus that are linked to by the page.
Afterwards, I was told to define a transition_model()
function. This function returns the probability of distribution over which page to visit next.
The transition_model()
accepts as arguments, corpus
, page
the page that the random surfer is currently at and the damping_factor
being the damping factor which our case is a constant representing the damping factor used to calculate the probabilities.
Then, I was asked to implement two pagerank algorithms :
sample_pagerank
iterative_pagerank
In both algorithm, I start by initialising a starting a starting rank. In the sample_pagerank
I start with 0 and in the iterative_pagerank
I start with a starting rank being :
sample_pagerank
I just randomly jump from page to page (with the help of the random.choices()
function to calculate their probabilities based on the previous results.
For the iterative_pagerank
, I use a while loop that keeps running until the count variable is equal to the number of pages in the corpus. In the while loop the given formula will be used to calculate the pageranks.
d
Being the damping factor.
NumLinks(i)
Being the number of links on the current page.
N
Being the number of pages in the corpus.
Heredity : 📖
I found this assignment to be one of the easiest, it was pretty straightforward and I passed all the tests in the first go.
Going over each person individually in the people list, I set a few conditions to be able to calculate the probabilities checking if a person, is a parent
or child
.
Then the following dictionary helps me in getting the relative probabilities of passing the genes by the number of genes :
prob_parent_give_gene = {
0: 0.01,
1: 0.49,
2: 0.99
}
Next in the if else if
chain, I calculate the the person's probabilities of having the gene based on the number of genes in the parents. After it's calculated, I add them to the people probabilities to finally return the joint probability being the product of all the elements in people_probabilities
.
Crossword : 📖
This was a long one but a fun one to make. The feeling I got after seeing the table displaying the right words is indescribable.
I was a given a list of words in .txt
files and structures for the tables that were parsed by the crossword.py
script given, turning them into a Crossword
and a Variable
classes. Which are used to be able to solve the crossword.
It consisted of implementing the ac3
algorithm and basically going over the given word possibilities singling out the ones that were not consistent with the contraints and keeping the ones that were.
To speed things up I implement the least constraining value heuristic (which was suggested but not required). Using this heuristic, I ordered the values in the domain of var
(being the variable that we're currently on) starting with the one rules out the fewest values amongst the neighbours of var
.
Shopping : 📖
We are given a csv file that contains data about around 12000 users. With this data I should use the K neighbor classifier to be able to predict whether a user will make a purchase or not.
In the load_data
function, we read the csv file row by row, forming the evidence list and the label list.
train_model
: given the lists of evidence and labels that we formed in the previous function, we had to train the model with k = 1 and then return the trained model.
In the last function (evaluate
), the predictions that the model has made were given as well as their true labels. So going over the data given, I compared the labels to the predictions and added counters to keep track of :
- The true positive labels that were identified (
true_positive_count
) - The true negative labels that were identified (
true_negative_count
) - The total number of positives (
total_positives
) - The total number of negatives (
total_negatives
)
Then using these variables, I calculated the sensitivity
and specificity
.
Nim : 📖
Two files were given, one that has the classes for which I have to implement the methods (nim.py
). The other being the file that handles the gameplay (play.py
).
The first function to implement is the get_q_value
in the NimAI
class. All this function does, is check if there's a q
value for the given 'state' and 'action' if not return 0.
Using the formula :
NVM
being the New Value EstimateOVE
being the Old Value Estimate
The q value gets updated in the update_q_value
function.
In the last function to implement (choose_action
) :
-
In the case where epsilon is
True
:- I set two choices either a random move or the best move in the goal of either exploring or exploiting.
- Based on the epsilon value as a weight for the
random.choices()
the AI takes the next action (either choses randomly from the list of actions or chooses based on the highestq
value).
-
In the case where epsilon is
False
:- I go over the possible actions in the current state. Checking if they have a
q
value or not (if they don't theirq
value will be considered 0). - Comparing their values until it finds the one with the highest
q
value. - Otherwise, it will just choose the last one on the list of actions.
Playing against the AI, it does take smart moves but with a little bit of cleverness it can be defeated.
- I go over the possible actions in the current state. Checking if they have a
Traffic : 📖
For this project, a README is required in which I should go over my experimentation process. (📄)
Parser : 📖
We are asked to determine the structure of a sentence using context-free grammar. We are supplied with a TERMINAL
set of symbols and our main task will be to create NONTERMINALS
that are general enough to generate the sentences provided in the /sentences
folder. Keeping in my that we should avoid over-generation as well as under-generation.
The first task is to implement a function that will accept a sentence as a string and we should return the words in that sentence to lowecase in a list. Which was pretty simple, all I had to do was in the help of the nltk.tokenize.word_tokenize(sentence.lower())
split the sentence into tokens and then filter the tokens that weren't words.
We're asked in the second task to complete the NONTERMINALS
so that when they're combined with the TERMINALS
we can parse all the sentences in the set. To be able to complete the complex sets, it helped me to draw the trees on a piece of paper. By doing so, I was able to visualize the structures of the phrases and see patterns to add new phrases.
Finally, we have to work on the np_chunks
function that takes the tree as a parameter and should return a list of all noun phrases that don't contain other noun phrases within them. Nothing very complicated in this one, the only challenge was the structure of the tree.
Questions : 📖
In this last and final project of this great CS50 AI course, we are asked to design an answering system like IBM's Watson. We are given a bunch of documents from which our system will have to identify which ones are the most relevant to our query (question asked) and select the sentence(s) that answers the question the most.
As a first task, we are asked to implement the load_files
function. Nothing too fancy, we just have to make sure we don't hardcode the paths so our functions works on other os. In addition, we have to exclude files that don't have a .txt
extension.
Then, we have to implement the tokenize
function. Using the nltk.tokenize.word_tokenize()
and passing it the lowercased document I will have a list of the tokenized document. However, we should filter out punctuations and stopwords. I do so by going over the tokenized document and removing the tokens that satisfy the condition. Next, I remove the duplicates from the list to avoid counting the word occurrences in the same list. Finally, after getting the occurrences of each word in all the documents I loop over the words to calculate their idf values.
top_files
function should return the n
top files that have the highest tf-idf score which will be calculated in the function. For that, I firstly loop over the words in the query storing the ones that are available in the idfs
dictionary which will save us immense time later for computing the scores. Next I go over the files counting the frequencies of the words in each file and storing them in another dictionary. After getting the frequencies, I calculate the score of the file and store it a dictionary for comparison later. Finally, in the return statement I sort the files based on their score then cast the dict to a list of keys and extract in another list the first n files with the highest score.
Same for the top_sentences
it should return the n
top sentences that have the highest matching word measure. However, instead of calculating the term frequencies we compute the matching word measure of each sentence. The matching word measure is computed by summing the idf value of each word present in the sentence. To do so I take a very similar approach to what I did in the top_files
function. But, one step was added before return the list of top_sentences
which is verifying if two sentences have the same value and if they do checking their terms density and then permuting their positions on the list if necessary.
My implementation works on most cases, except for the "how do neurons connect in a neural network?" where I get the answer : "They can be pooling, where a group of neurons in one layer connect to a single neuron in the next layer, thereby reducing the number of neurons in that layer.". Which makes sense because of how we calculate the matching word measure.