Functions and programs

January 22, 2025

Functions

Programs

In this course, a program is a file or string that contains instructions written using a programming language (like C or Python) for how to compute a function.

Consider the following computer program, that takes an input string called input_string from {0, 1}*.

def contains_0101(input_string):
    if "0101" in input_string:
        return 1
    else:
        return 0

Decision Programs

  • Decision programs. A program that always returns a 0 or a 1 when it finishes is called a decision program.
  • Example Decision Programs

    These are all decision programs.

    def always_one(input_string):
        return 1
    def maybe_loop_forever(input_string):
        if "1" in input_string:
            n = 1
            while n > 0:
                n = n + 1
        else:
            return 0
    def more_than_100_bits(input_string):
        if len(input_string) > 100:
            return 1
        else:
            return 0

    Answer

    The Python function maybe_loop_forever does not correspond to a true function f : {0, 1}* → {0, 1}.

    def always_one(input_string):
        return 1
    def maybe_loop_forever(input_string):
        if "1" in input_string:
            n = 1
            while n > 0:
                n = n + 1
        else:
            return 0
    def more_than_100_bits(input_string):
        if len(input_string) > 100:
            return 1
        else:
            return 0

    Converting Programs to Strings

    Each program is written using ASCII characters. For example:

    def always_one(input_string):
        return 1

    has 43 ASCII characters (including an end of file EOF character). How many bits is that? Recall that standard ASCII codes go from 0 to 127.

    Inputing One Program Into Another

    What output would we get if we used the binary representation of the always_one program as the input_string for the more_than_100_bits program?

    more_than_100_bits(binary_encoding("always_one.py"))
    def maybe_loop_forever(input_string):
        if "1" in input_string:
            n = 1
            while n > 0:
                n = n + 1
        else:
            return 0

    Inputing a Program Into Itself

    What happens with each of these commands?

    always_one(binary_encoding("always_one.py"))
    maybe_loop_forever(binary_encoding("maybe_loop_forever.py"))
    more_than_100_bits(binary_encoding("more_than_100_bits.py"))

    A More Complicated Program

    Consider the following unfinished program:

    def check_if_program_returns_one(program_string, input_string):
        """return 1 if 
            1. program_string is the ASCII binary encoding of a valid Python program, and
            2. The program defined by program_string returns 1 when given input_string as its input.
        otherwise, return 0."""

    Here’s the idea:

    If the program check_if_program_returns_one is possible, then we could make the following Python function:

    def reverse_check(program_string):
        return 1 - check_if_program_returns_one(program_string, program_string)

    What will reverse_check do?

    1. If we feed reverse_check itself as input, why can’t it return 1?

    2. Why can’t it return 0?

    Conclusion

    Both check_if_program_returns_one and reverse_check are impossible programs.

    Un-programmable Functions

    There is an easier way to see that there are functions f : {0, 1}* → {0, 1} which are impossible to implement with computer programs based on cardinality.

    How Many Functions?

    Schroeder Bernstein Theorem

    Another really useful theorem is:

    Schroeder-Bernstein Theorem. If the are 1-to-1 functions f : A → B and g : B → A, then there is a bijection from A to B. In particular, |A| = |B|.