Thursday, 4 December 2014

Does... Not... Compute!

When a function is not computable, it can bring a big headache to any programmer! That is why it is necessary for us to determine whether a function is computable or not. From what I was able to observe, a function is deemed non-computable when it has a definition that prompts it to return a value even when an inner function that was called inside does not halt.

To be able to determine properly if a function is non-computable, we need to reduce it to what we already know to be non-computable: our favourite function, halt().

Let us look at the following example:

def is_bigger_than_five(f, v)
    """
    Return True iff f(v) returns an int that is bigger than 5
    """
    #code here

    return True

At simple sight, we can tell that, if f(v) does not stop, then the function should return False. To be sure, we need to reduce it to halt in the following way:

def halt(f, i)

    def is_life(g, v)
        """
        Return True iff f(v) returns "life"
        """
        #code here

        return True

    def f_prime(x)
        """
        Ignore the argument x, call f with the fixed argument i (the one passed into halt)
        """

        f(i)

        return "life"

    return is_life(f_prime, v)

From the following, we can conclude:

  • If f(i) halts, then is_life(f_prime, v) is True
  • If f(i) does not halt, then is_life(f_prime, v) is False
Therefore, is_life() is not computable!

Sunday, 23 November 2014

Big O-Me-Ga' Let's Go From Left to Right

Big Omega is related to Big Oh in the way that, while the latter defines an upper bound, the former defines a lower bound for a given function. This is done, usually, by expressing the function in a simpler function, akin to the Big Oh's expression. In mathematical form, it takes the following shape:

f(x) ∈ Ω(g(x))

The full definition would be the following:

∃ c ∈ ℝ⁺, ∃ B ∈ ℕ, ∀ n ∈ ℕ, n ≥ B ⇒ f(x) ≥ c(g(x))


This is identical to the definition of Big Oh, except that instead of f(x) ≤ c(g(x)) we have f(x) ≥ c(g(x)). This shows how this second equation approaches the function from the other side, thus defining a lower bound. Once again, we need to use proofs to establish whether the left side of the inequality is equal to the right side of the inequality. To go from left to right, we need to be, once again, creative.

Let's work with the following example:

8n^4 + 5n^2 - n ∈ Ω(-n^5 + 2n^3 - 4)

Then this would be equal to:

∃ c ∈ ℝ⁺, ∃ B ∈ ℕ, ∀ n ∈ ℕ, n ≥ B ⇒ 8n^4 + 5n^2 - n ≥ c(-n^5 + 2n^3 - 4)

First, we should work on the inequality

8n^4 + 5n^2 - n ≥ 8n^4 + 5n^2 #we remove -n to start off the inequality
                          ≥ 8n^2 + 5n^2 #since n^4 ≥ n^2, this inequality would still be true
                          ≥ 13n^2 #addition
                          ≥ n∙n^2 #since we have 13 as a coefficient, we could make B = 13 so that n ≥ 13
                          ≥ n^3 #multiplication
                          ≥ 2n^3 - n^3 #still n^3
                          ≥ 2n^3 - n^5 #since - n^5 ≤ - n^3 this is still true
                          ≥ -n^5 + 2n^3 - 4 #subtracting 4, this is still true
                          ≥ 1(-n^5 + 2n^3 - 4) #we could make c = 1

We have established values for both c and B, so that c = 1 and B = 13. In order to work the proof, we would have to do the following arrangement:
8n^4 + 5n^2 - n ∈ Ω(-n^5 + 2n^3 - 4)
∃ c ∈ ℝ⁺, ∃ B ∈ ℕ, ∀ n ∈ ℕ, n ≥ B ⇒ 8n^4 + 5n^2 - n ≥ c(-n^5 + 2n^3 - 4)
Assume n ∈ ℕ and n ≥ 13 #domain assumption and antecedent
    Then 8n^4 + 5n^2 - n ≥ 8n^4 + 5n^2 #add n
                          ≥ 8n^2 + 5n^2 #since n^4 ≥ n^2
                          ≥ 13n^2 #add both together
                          ≥ n∙n^2 #since n ≥ 13
                          ≥ n^3 #multiply both together
                          ≥ 2n^3 - n^3
                          ≥ 2n^3 - n^5 #since - n^5 ≤ - n^3
                          ≥ -n^5 + 2n^3 - 4 #subtract 4
    Then ∀ n ∈ ℕ, n ≥ B ⇒ 8n^4 + 5n^2 - n ≥ c(-n^5 + 2n^3 - 4) #introduce ⇒
Then 8n^4 + 5n^2 - n ∈ Ω(-n^5 + 2n^3 - 4) #introduce Big Omega



Done! 

Sunday, 16 November 2014

Big Oh My God Let's Go From Left to Right

Big Oh is a mathematical expression meant to describe the behaviour of a function as its parameter tends to a certain value or infinity. This is done, usually, by expressing the function in a simpler function. In mathematical form, it takes the following shape:

f(x) ∈ O(g(x))

The full definition would be the following:

∃ c ∈ ℝ⁺, ∃ B ∈ ℕ, ∀ n ∈ ℕ, n ≥ B ⇒ f(x) ≤ c(g(x))


Relating this to everything we have learnt before, we need to use proofs to establish whether the left side of the inequality is equal to the right side of the inequality. To go from left to right, we need to be very, very creative.

Let's work with the following example:

n^4 - 5n^2 + 1 ∈ O(6n^5 - 5n^3 + 3n)

Then this would be equal to:

∃ c ∈ ℝ⁺, ∃ B ∈ ℕ, ∀ n ∈ ℕ, n ≥ B ⇒ n^4 - 5n^2 + 1 ≤ c(6n^5 - 5n^3 + 3n)

First, we should work on the inequality

n^4 - 5n^2 + 1 ≤ n^4 + 1 #we remove - 5n^2 to start off the inequality
                        ≤ n^4 + n^4 #since n^4 ≥ 1, this inequality would still be true
                        ≤ 2n^4 #addition
                        ≤ n∙n^4 #since we have 2 as a coefficient, we could make B = 2 so that n ≥ 2
                        ≤ n^5 #multiplication
                        ≤ 6n^5 - 5n^5 #still n^5
                        ≤ 6n^5 - 5n^3 #since - n^5 ≤ - n^3 this is still true
                        ≤ 6n^5 - 5n^3 + 3n #adding 3n, this is still true
                        ≤ 1(6n^5 - 5n^3 + 3n) #we could make c = 1

We have established values for both c and B, so that c = 1 and B = 2. In order to work the proof, we would have to do the following arrangement:
n^4 - 5n^2 + 1 ∈ O(6n^5 - 5n^3 + 3n)
∃ c ∈ ℝ⁺, ∃ B ∈ ℕ, ∀ n ∈ ℕ, n ≥ B ⇒ n^4 - 5n^2 + 1 ≤ c(6n^5 - 5n^3 + 3n)
Assume n ∈ ℕ and n ≥ 2 #domain assumption and antecedent
    Then n^4 - 5n^2 + 1 ≤ n^4 + 1 #add 5n^2
                                      ≤ n^4 + n^4 #since n^4 ≥ 1
                                      ≤ 2n^4 #add both together
                                      ≤ n∙n^4 #since n ≥ 2
                                      ≤ n^5 #multiply both together
                                      ≤ 6n^5 - 5n^5
                                      ≤ 6n^5 - 5n^3 #since - n^5 ≤ - n^3
                                      ≤ 6n^5 - 5n^3 + 3n #add 3n
    Then ∀ n ∈ ℕ, n ≥ 2 ⇒ n^4 - 5n^2 + 1 ≤ c(6n^5 - 5n^3 + 3n) #introduce ⇒
Then n^4 - 5n^2 + 1 ∈ O(6n^5 - 5n^3 + 3n) #introduce Big Oh

Done! 

Sunday, 9 November 2014

Algorithm Run Time - From a Specific Action to Steps

Computing power is a known limitation when programming. Though for the untrained eye, computers seem vastly powerful, when programming you are trying to reduce both the load on the processor and the running time. As we use less resources, we will have time to do more at the same time, and in less time. That is the rationale behind analyzing the run time of algorithms and code in general. When talking about code, there are three different ways in which run code is interpreted:

  • Linear: Each extra loop increases run time in the same way
  • Exponential: Each extra loop increases run time exponentially, so many extra loops will drastically increase run time
  • Logarithmic: Each extra loop increases run time logarithmically, so after one point extra loops have a minimal impact on run time
To calculate run time, we need to pay close look at our code in order to establish how many steps and loops are given. Let's look at this piece of code and its outcome:

    for i in range(l):
        print(i)

If we use l = 10, how many times will print() be used?
Print will be used 10 times. The output would be the following:

0
1
2
3
4
5
6
7
8
9


What about the number of steps? Would it be 10 steps? 
We need to consider that there are two steps involved: for and print(). In this case, we would be talking about 20 steps in total, one for for and one for print() per loop.

Now let's change the code a little bit:


    for i in range(l):
        print(i)
        for j in range(l):
            print(j)

If we use l = 10, how many times will print() be used?
Print will be used 10 times in the outer loop, and 10 times in the inner loop for each loop in the outer loop. This means that print will be used 10 * 10 = 100. The output would be the following:


0
0
1
2
3
4
5
6
7
8
9
1
0
1
2
3
4
5
6
7
8
9
2
0
1
2
3
4
5
6
7
8
9
3
0
1
2
3
4
5
6
7
8
9
4
0
1
2
3
4
5
6
7
8
9
5
0
1
2
3
4
5
6
7
8
9
6
0
1
2
3
4
5
6
7
8
9
7
0
1
2
3
4
5
6
7
8
9
8
0
1
2
3
4
5
6
7
8
9
9
0
1
2
3
4
5
6
7
8
9

What about the number of steps? Would it be 100 steps?
Similarly to the previous question, we need to consider both inner and outer loop, and their interaction. The inner loop has 2 steps, and the outer loop has 2 steps, and each time the outer loop runs, the inner loop runs l times. Hence, in this case, we are talking about 20 steps per each run of the inner loop, so it would be 200 steps of the inner loop with 20 steps of the outer loop. In total, 220 steps.

As we can see, proper analysis of the code needs to take place in order to decipher how the loops interact and, finally, how many steps will be taken. 

Sunday, 2 November 2014

Solving a Mystery, Case by Case

Some of the proofs we have seen before were pretty direct. When you have P(x) ⇒ Q(x), there is little that can go wrong~! But things do become a little bit more complicated when we encounter ∧. This conjunction symbol indicates that, if we are to prove something, we need to show that both sides are true. When you find a conjunction on the antecedent, such as having P(x) ⇒ Q(x) were P(x) ⇔ R(x) ∧ S(x), you can just leave it as the assumption.

For example:

∀ x ∈ ℕ, R(x) ∧ S(x) ⇒ Q(x)

Would lead us to:
Assume x ∈ ℕ
    Assume R(x) ∧ S(x)
         ....
    Then R(x) ∧ S(x) ⇒ Q(x)
Then ∀ x ∈ ℕ, R(x) ∧ S(x) ⇒ Q(x)

This is fairly easy, and makes our lives' easier! Nonetheless, there will be plenty of times when we will want to disprove a conjunction. To do this, we will have to work on different cases. Let's look at the converse of the previous statement, and let's assume it is true:

∀ x ∈ ℕ, Q(x) ⇒ R(x) ∧ S(x)

This would lead us to:
Assume x ∈ ℕ
    Assume Q(x)
         Then...
             Assume... #case 1 that leads to proving R(x)
                 Then...
             Assume... #case 2 that leads to proving S(x)
                 Then...
         Then...
    Then Q(x) ⇒ R(x) ∧ S(x)
Then ∀ x ∈ ℕ, Q(x) ⇒ R(x) ∧ S(x)

As you can see, separate cases spawn while we are working on this proof. The important thing is to remember that these cases need to lead to proving R(x) and S(x) and need to be within context of Q(x). That sounds like a lot of work, but many times it is easy to see the light!

Sunday, 26 October 2014

Finding the Perfect Substitution

While we were looking at the form structure when finding proof, I noticed that I had no recollection whatsoever on how to do it. I remember, while studying number theory, that there were some neat tricks on how to prove a certain expression for n and then n + 1, and finally for n + k. In essence, we are finding proof here of a single expression, but when we have to related expressions, the dynamics are a bit different, so here we will now focus on a postulate that involves two expressions and the way to tackle this problem, as well as finding the perfect substitution to get our desired result.

If we were to use the example:

∀ x ∈ Z, ∃ y ∈ Z, x = y + 1 ⇒ ∃ z ∈ Z, x^2 = z + 1

We would have the following steps for establishing the proof:

        Assume ∀ x ∈ Z # domain assumption
                Assume ∃ y ∈ Z, x = y + 1 # antecedent 
                        Let y' be a value that depends on x... # some value that depends on x 
                        Then, y' ∈ Z 
                        Let z' = y^2 + 2y
                        Then, z' ∈ Z
                        If x^2 = (y' + 1)^2 = y'^2 + 2y' + 1 and z' = y^2 + 2y
                        Then, x^2 = z' + 1
                Then ∃ z ∈ Z, x^2 = z + 1 # introducing ∃
        Then ∀ x ∈ Z, ∃ y ∈ Z, x = y + 1 ⇒ ∃ z ∈ Z, x^2 = z + 1

Let us focus now on the bold part. How did we find it? Easy: we knew where we wanted to go. If we are going from x to x^2, then we know that the expression on the left will be squared. Rearrange things a little bit, and voila!

Sunday, 19 October 2014

Floors and Ceilings - Almost getting there!

From this week's lecture I found quite intriguing the importance that floor and ceiling functions play in computer theory. At the beginning I was rather skeptical of their usefulness, but that just came to show how vast the applications of computational logic really are. I'll just go ahead and offer two examples that have to do with my current programming level with Python:
  • Mod Operand (% in Python)
    • x % y = x - y floor (x/y)
    • In mathematical expression: ∀ x in Z, ∀ y in Z, ∃ z in Z, z = x - y (floor (x/y))
  • Rounding (round() in Python)
    • round(x) = floor (x + 1/2)
    • In mathematical expression: ∀ x in R, ∃ y in Z, y = floor (x + 1/2)
The important lesson here is not think harshly about some functions in mathematical reasoning. Perhaps we should apply the logic that, if it has its own symbol, it must be rather important. While doing some research about the floor and ceiling functions, I found that there are plenty more applications to it! Hopefully in a near future we will be able to see some more.

Sunday, 12 October 2014

Proof me wrong! - The order when establishing proof

An important part of complying with scientific standards is the use of the scientific method to determine whether an equation holds more than empirical truth in itself. What we need here is a method for understanding, properly, if a mathematical expression holds truth. We will focus on the form this time around.

If we were to use the example:

∀ x ∈ Z, | x | > 0 ⇒ ∃ z ∈ Z, x < z

We would have the following steps for establishing the proof:

        Assume ∀ x ∈ Z # domain assumption 
                Assume | x | > 0 # antecedent 
                        Let z' = ... # some value that depends on x 
                        [proof of z' ∈ Z] 
                        Then, z' ∈ Z 
                        [proof of x < z']
                        Then, x < z' 
                Then ∃ z ∈ Z, x < z # introducing ∃
        Then ∀ x ∈ Z, | x | > 0 ⇒ ∃ z ∈ Z, x < z

Let's dissect these steps, separating it by their indentation. We would have, in this case, 3 indents:
  • Overall proof
    • If 'antecedent', then 'right-hand side statement'
      • Proof of 'right-hand side statement' for limited or universal variable
If we were to work it the other way around, we would have:
  • Proof that the 'right-hand side statement' holds itself true for a variable that belongs to the desired domain and scope (∃ or ∀)
    • Since it was proven that the 'right-hand side statementis true for a variable that belongs to the domain and scope, given the antecedent, we can conclude that it holds true for the desired domain
      • Since it was proven with the antecedent, and the antecedent depends on the domain limitations, then the whole statement must be true.
While this was explained using several words, it is important not only to learn the structure, but also to understand why it is used in such a way.



Friday, 10 October 2014

Does the order of establishing domains change the way you read something?

A few weeks ago, just as we had started the semester, I read a question that asked whether these two sentences were the same:

∃ x ∈ A, ∀ y ∈ A, P(x, y) ⇒ Q(x)
∀ y ∈ A, ∃ x ∈ A, P(x, y) ⇒ Q(x)

I remember thinking that these two sentences might be the same in one of those fashions were the order of the factors does not affect the result. Oh boy was I wrong! It is very easy to notice the difference when we put it in words:

For some x that belong to A, all y that belong to A satisfy this...
For all y that belong to A, there is an x that belongs to A such that...

I tried not to dwell too much into the latter part of the equation since it is not relevant to our point. The final point being that it does not only make a difference the way in which the variables and domains are established, but also that it becomes easy to understand once you try it out in English!

Saturday, 27 September 2014

For All This is False = For Not Some, This is True

What I found rather charming from last week was that there is more than one Mathematical way to say the same thing. While I was commenting on how clean and refined mathematical expressions could be, in such a way that they reduce ambiguity to zero, I found this to be some sort of conundrum (specially when you have an approach to explaining things, and the recipient has another).

Let's focus on one expression that has two different approaches! Let's say for example x and y are courses that belong to C, which is a set that is made by courses taught at UofT. We also have a function P(x, y), where x is a prerequisite of y. The question is, how should we right the following statement?

MAT135 has no prerequisites


We would then have to put it in terms we could understand, correctly using the proper quantifiers. So we could say that:

  • For any course x, there is no course that is a prerequisite of MAT135.

x ∈ C, ¬ P(x, MAT135)

  •  For not some x, there is a course that is a prerequisite of MAT135. 

¬ ∃ x ∈ C, P(x, MAT135)

This is nothing but a small example of how things could be approached in a different way. While this seems a little obvious since it is a game that could be simplified as Positive - Negative = Negative - Positive, there are some other complex statements that could be described in different ways.

Bottomline, not everything has one single answer. Or perhaps we should say...

∃ ∈ Ay ∈ A, ∃ z ∈ Q, S(x, z) ∧ S(y, z) ∧ y

Note: Where x and y are answers that belong to Az is a question that belongs to Q, and S(x, z) is a function such that x is a solution to z.

Thursday, 18 September 2014

The 1, 2, 3... and 4 of Problem Solving

Mathematical expressions are never misunderstood. When a person states 6 + 6, the meaning of that statement will be the same wherever you go. That is, unlike modern languages, math has a precise meaning for every construction. Hence, when constructing a computer algorithm, we need to make sure that our thoughts are conveyed properly by machines. That is where Mathematical Expressions come in.

In other words, we use math as the medium through which we are going to solve problems. Once the problem is properly conveyed in mathematical form, we can attempt to solve it using computer applications.

Polya and the 4 Steps to Solving Problems
Decades ago, G. Polya came with several steps on how to solve problems. While some people will find this obvious or redundant, it is important to remember that true scientists apply a structured, consisten methodology in everything they do. This includes problem solving.

Quickly stated, Polya suggests that we:
  1. Understand the problem
  2. Devise a plan
  3. Carry out the plan
  4. Look back
Taken from G. Polya, "How to Solve It", 2nd ed., Princeton University Press, 1957, ISBN 0-691-08097-6.

Example:
A study conducted by Toronto's City Council has shown that there is a positive correlation between the quality of street performers in the city and the amount of tourists coming to the city. The Mayor believes that most street performers are talentless, thus explaining the decline in tourism for the last decade. He wants you to come up with a way to improve the quality of street performers in an economical way. As a scientifically driven person, you apply Polya's steps on "How to Solve It".

Step 1: Understand the problem
What do we know about street performers? Are all street performers talentless? How do we determine if a street performer is good or not? By asking ourselves as many questions about and surrounding the problem, we are able to fully get a grasp of what needs to be done. Visualizing the problem also helps understanding it.

S is the universe of street performers in Toronto.
G is the set that contains good street performers.
A is the set that contains active street performers.

So we want the intersection between G and A to increase in size; we probably want to aim for A to become a subset of G.

Step 2: Devise a plan
Has there ever been a similar problem that has been solved? If that were the case, we could apply that solution to this problem. Since this is a unique problem, we need to come up with a unique solution. To make A a subset of G, we would need to either: a) encourage more good street performers to become active while deterring the rest from performing, or b) make the rest become good.

The most economical way would be a) since it trusts on good performers not needing additional training. Hence, the plan will consist of forcing street performers to apply for permits. When applying, they need to show their skills. A panel of judges will decide whether they are worthy of a permit or not.

Step 3: Carry out the plan
Did the plan solve the problem? Apparently, the plan was a limited success. Some people started renting their permits, creating a situation where some permit owners would have other performers working for them. While this was not the case of the majority, it still needs to be addressed.

Step 4: Look back
We need to review every step in the plan. Can any improvement be done to the steps in order to avoid people renting their permits? If modifying previous steps doesn't solve the problem, perhaps we could introduce extra steps to ensure the success of the plan.