Python from C

A gentle introduction to Python for C programmers



~ $ python
Python 3.3.0 (default, Dec 15 2012, 17:19:46)
[GCC 4.5.2] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> import ian
>>> help(ian)













Enough about me...

Help on module ian:

DESCRIPTION
Ian Cordasco
~~~~~~~~~~~~

- Developer at Bendyworks
- Open Source Lover
* Maintainer of Flake8 (python code-quality tool)
* Maintainer of Requests (python library for HTTP/1.1)
* Author of github3.py (wrapper around the GitHub API)
* Author of betamax (recording your HTTP interactions so the NSA doesn't have to)
* Author of uritemplate.py (implementation of RFC 6570)
* Contributor to lots of other stuff
- Python developer for the last 2 years
- C developer before that (still loves it)

lines 1-23/23 (END)

Core Topics

  • Type disciplines (which apply to C and which apply to Python)
    • Duck typed
    • Dynamically typed
    • Statically typed
  • Caveat into Exceptions
  • Creating "full programs"
  • Everything is an object
  • Creating your own objects

Type Disciplines

  • Duck typed
    • If it looks, acts, and sounds like a duck, it is most likely a duck.
    • This applies to Python (and we'll see how soon)
  • Strongly typed
    • Interactions between types are well defined
    • This applies to Python (not C)
  • Statically typed
    • Developers must specify the type they wish to use
    • This applies to C
  • Dynamically typed
    • Developers do not have to specify the type they wish to use
    • This applies to Python


And now for some examples

Hello World

In C

#include <stdio.h>

int main(char **argv, int argc){
puts("Hello World!\n");
return 0; /* Technically not even necessary anymore in C11 */
}

In Python

print("Hello World!")

Now that we have that out of the way

...

Something still trivial

But not exactly...

C

int add(int a, int b){
return a + b;
}

int main(void){ /* We're all adults here */
(void)add(1, 2);
(void)add(1, 'c');
(void)add(1, "characters");
return 0;
}

Python

def add(a, b):
return a + b

add(1, 2)
add(1, "foo") # Wait, what?!

What is actually happening here?!

Duck Typing


Duck Typing

In that code example:

def add(a, b):
return a + b

add(1, 2)
add(1, "foo")

What is actually happening is something along these lines:


  • 1 and 2 look and act alike. Let's try 1 + 2. Okay, we're good
  • 1 and "foo" act alike (they can both be added). Let's try adding them.


The difference is that the second case will raise a TypeError

Some questions

  • Will that C example ever run? (Hint: For the purpose of the example, I'm not using strict compilation flags)
  • If it does, how?
  • Does the Python example ever run?
  • How does the Python example run?


Some answers

(I don't like suspense)

  • The C example compiles and will run. (The third call to add will cause a warning on compilation.)
  • It runs because of "weak" typing. (No don't start pressing your keys harder.)
  • The Python example does run up until it encounters the TypeError

Exceptions

Let's look at that code example again

def add(a, b):
return a + b

add(1, "string")

We see a TypeError in the console

~ $ python add.py
Traceback (most recent call last):
File "add.py", line 4, in <module>
add(1, "string")
File "add.py", line 2, in add
return a + b
TypeError: unsupported operand type(s) for +: 'int' and 'str'

We get the line numbers and an explanation of the exception

Exceptions cont'd

Our C example, when compiled provides

~ $ gcc add.c -o add
add.c: In function 'main':
add.c:10: warning: passing argument 2 of 'add' makes integer from pointer without a cast

Who said anything about pointers?

This is a common complaint among new programmers


But who said anything about new programmers?

(And who is dog?)

I did. Just now. Did you miss it?

Why I Like Exceptions

They make everything so accessible

  • The exception
    • triggers a full stack trace
    • stops execution entirely
    • provides a more intuitive message
  • Standard compilation
    • merely warns you
    • compiles anyway
    • assumes you know what you're doing


Program execution

In C
#include <stdio.h>

int main(void){
printf("%d\n", 1 + 2);
return 0;
}
In Python
def main():
print("%d" % (1 + 2))

Which one prints out 3?

How can I make Python work?

If you run the following instead

def main():
print("%d" % (1 + 2))

main()

or even just

print("%d" % (1 + 2))

you'll see 3 printed to the screen

Why?

Program Execution

  • When the python interpreter opens a file it executes its contents
  • If all you have are function definitions they are simply added to
    the namespace
  • For something to happen you have to tell Python you want it
    to happen
  • Canonical python script:
def function():
# This function does something

if __name__ == '__main__':
function()
Woah what is this "__name__" thing?!

__name__

  • __name__ is a special variable set in each file
  • Let's make the following files and run one of them:
    • my_main.py:
      import fake_module
      if __name__ == '__main__':
      print("From within my_main.py")
      print("fake_module.__name__ = %s" % fake_module.__name__)
      print("my_main.__name__ = %s" % __name__)
    • fake_module.py
      print("From within fake_module.py")
      print("fake_module.__name__ = %s" % __name__)

 ~ $ python my_main.py
From within fake_module.py
fake_module.__name__ = fake_module
From within my_main.py
fake_module.__name__ = fake_module
my_main.__name__ = __main__

__name__

  • So everything in a python file is executed even on import
  • That is, assuming it is not a function or class definition
  • As is demonstrated flow-control statements are executed
    (if, else, elif)
  • Loops will also be executed (while, for)
  • Variables will be evaluated and assigned to
  • Calls to functions, methods called on class instances,
    and calls to class methods are all executed
  • Why? Because that's how Guido wanted it and he
    is our Benevolent Dictator for Life (BDFL)

Everything is an Object

Seriously, even integers have methods

~ $ python
Python 3.3.0 (default, Dec 15 2012, 17:19:46)
[GCC 4.5.2] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> help(1)
Help on int object:

class int(object)
| int(x[, base]) -> integer
|
| Convert a string or number to an integer, if possible. A floating
| point argument will be truncated towards zero (this does not include a
| string representation of a floating point number!) When converting a
| string, use the optional base. It is an error to supply a base when
| converting a non-string.
|
| Methods defined here:
|
| __abs__(...)
| x.__abs__() <==> abs(x)
|
| __add__(...)
| x.__add__(y) <==> x+y
...

Object-Oriented Programming

Everything has methods

So simple integer arithmetic can be written two ways:

1 + 1
1.__add__(1)

In Python, the obvious way is the right way

# |-1| == 1
assert abs(-1) == 1
assert (-1).__abs__() == 1

In many cases there are built-in functions that exist

simply to call methods on built-in objects.

Built-in Objects

  • List
    • Implementation: linked list
    • Literal syntax "[1, 2, 3]"
  • Dictionary (hash)
    • Implementation: unordered hash
    • Literal syntax "{'key': 'value'}"
  • Tuples
    • "Implementation": fixed-size array
    • Literal syntax "(1, 2, 3)"
  • Sets
    • Literal syntax "{1, 2, 3,}"
  • Strings, Bytearrays
  • Integers, Complex numbers, floats

Example Usage

a = [1, 2, 3]
print(a[0]) # prints 1
print(a[-1]) # prints 3
print(a[-3]) # prints 1
a.append(4)
print(a[-1]) # prints 4
a[2] = 10

b = {1: 2, 'key': 'value'}
print(b[1]) # prints 2
print(b['key']) # prints value
b['new key'] = 'new value'

c = {1, 2, 3, 4}
d = {5, 6, 7, 8}
print(c.union(d)) # prints {1, 2, 3, 4, 5, 6, 7, 8}
e = (1, 2, 3)
print(e[0]) # prints 1
print(e[-1]) # prints 3

f = 'string'
print(f[0]) # prints s
print(f[0:3]) # prints str

g = 1 + 1j # Yay for Electrical Engineers -_-
print(1 + g) # prints (2+1j)
print(1j + g) # prints (1+2j)

Iteration

List iterations in C and Python

Rough C:

for (t_node node = list->head; node; node = node->next) {
dump_node(node);
}

Python:

l = [1, 2, 3, 4, 5]
for i in l:
print(i)

Uhm, what?

  • Lists, sets, tuples, dictionaries, & strings can all be iterated over
    • Lists, sets, and tuples iterate on items
    • Dictionaries iterate on keys
    • Strings iterate on "characters"

for item in [1, 2, 3]:    # prints "1 2 3"
print(item, end=' ')
d = {'key0': 'value0', 'key0': 'value0'}
for item in d:    # prints "key1 => value1 key0 => value0"
print("%s => %s" % (item, d[item]), end=" ")
for c in 'this is a string':   # prints "t h i s  i s  a  s t r i n g" 
print(c, end=" ")

This last one, most people despise

Iteration vs Canonical C

Arrays

Let's assume arrays are like tuples

int[10] a = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
int i = 0;

for (; i < 10; i++) {
printf("%d ", a[i]);
}
a = (1, 2, 3, 4, 5, 6, 7, 8, 9, 10)
i = 0

while i < 10:
print(a[i], end=" ")
i += 1

vs

a = (1, 2, 3, 4, 5, 6, 7, 8, 9, 10)
for i in a:
print(i, end=" ")

Bye, bye loop counter. I shall miss you.

Quick Note on Slicing

a = "string foo bar"
print(a[:6]) # prints string
print(a[0:6])
print(a[7:10]) # prints foo
print(a[11:]) # prints bar

b = [1, 2, 3, 4]
assert b[:3] == [1, 2]
assert b[2:] == [2, 3, 4]

Slices seem like magic but they're awesome

Unfortunately they're a bit beyond the scope of this talk

But after the next topic I can discuss them a bit if I have time

Creating Your Own Objects

In C:

/* user.h */
typedef struct { /* We're all good citizens who use typedefs, right? */
int id;
char* login;
char* real_name;
/* ... */
} t_gh_user;

t_gh_user* gh_new_user(int, char*, char*, /* ... */);
int gh_update_user(t_gh_user, char*, char*, /* ... */);
/* And many more "methods" */

In Python:

class User(object):
def __init__(self, id, login, real_name,
# ...
):
self.id = id
self.login = login
# ...

def update(self, # ...
):
# ...

Objects, cont'd

Making and manipulating objects

In C:

t_gh_user* u = gh_new_user(1, "mojombo", "Tom Preston Warner", /* ... */);
if (!gh_update_user(u, "Tom Preston (the Awesome) Warner", /* ... */) {
error("We can not update mojombo's profile! Abort! Abort!");
}

In Python:

u = User(login="mojombo", id=1, real_name="Tom Preston Warner", #...
)
try:
u.update(real_name="Tom Preston (the Awesome) Warner", #...
)
except Exception:
error("We can not update mojombo's profile! Abort! Abort!)

But what is __init__?!

  • __init__
    • is a method on an object
    • is not a constructor
    • is an initializer
    • is called to initialze an instantiated object
  • You might notice that __init__'s first parameter is self
    • This is the object that was constructed
    • We then set attributes on self
    • We never return self.




Where are the clowns?

I meant to say constructors

You should never need to see them

See possible up-coming talk on metaprogramming in Python

This may or may not be a vision test

Thanks!



@sigmavirus24

(Twitter, GitHub, StackOverflow)


ian@bendyworks.com


Python from C

By Manoj Pandey

Python from C

  • 1,931