Comprehensive Python Cheatsheet =============================== ![Monty Python](web/image_888.jpeg) Main ---- ```python if __name__ == '__main__': main() ``` List ---- ```python [from_inclusive : to_exclusive : step_size] .append() .extend() .sort() .reverse() = sorted() = reversed() ``` ```python sum_of_elements = sum() elementwise_sum = [sum(pair) for pair in zip(list_a, list_b)] sorted_by_second = sorted(, key=lambda el: el[1]) sorted_by_both = sorted(, key=lambda el: (el[1], el[0])) flattened_list = [item for sublist in for item in sublist] list_of_chars = list() ``` ```python index = .index() # Returns first index of item. .insert(index, ) # Inserts item at index and moves the rest to the right. = .pop([index]) # Removes and returns item at index or from the end. .remove() # Removes first occurrence of item. .clear() # Removes all items. ``` Dictionary ---------- ```python .items() .get(key, default) .setdefault(key, default) .update() ``` ```python collections.defaultdict() # Creates a dictionary with default values. dict() # Initiates a dict from list of key/value pairs. dict(zip(keys, values)) # Initiates a dict from two lists. {k: v for k, v in .items() if k in } # Filters a dict by keys. ``` ### Counter ```python >>> from collections import Counter >>> z = ['blue', 'red', 'blue', 'yellow', 'blue', 'red'] >>> Counter(z) Counter({'blue': 3, 'red': 2, 'yellow': 1}) ``` Set --- ```python = set() .add() .update() .union() .intersection() .difference() .issubset() .issuperset() ``` ### Frozenset #### Is hashable and can be used as a key in dictionary: ```python = frozenset() ``` Range ----- ```python range(to_exclusive) range(from_inclusive, to_exclusive) range(from_inclusive, to_exclusive, step_size) range(from_inclusive, to_exclusive, -step_size) ``` Enumerate --------- ```python for i, in enumerate( [, i_start]) ``` Named Tuple ----------- ```python >>> Point = collections.namedtuple('Point', ['x', 'y']) >>> a = Point(1, y=2) Point(x=1, y=2) >>> a.x 1 >>> getattr(a, 'y') 2 ``` Iterator -------- #### Skips first element: ```python next() for element in : ... ``` #### Reads input until it reaches an empty line: ```python for line in iter(input, ''): ... ``` #### Same, but prints a message every time: ```python from functools import partial for line in iter(partial(input, 'Please enter value'), ''): ... ``` Generator --------- ```python def step(start, step): while True: yield start start += step ``` ```python stepper = step(10, 2) next(stepper) # 10 (, 12, 14, ...) ``` Type ---- ```python type() # / / ... ``` ```python import numbers isinstance(, numbers.Number) # Integral, Real, Rational, Complex ``` String ------ ```python str.replace(text, old, new) .split(sep=None, maxsplit=-1) .strip() .join() .startswith() # Pass tuple of strings for multiple options. .isnumeric() # True if str contains only numeric characters. ``` ### Print ```python print( [, , end='', sep='', file=]) # Use 'file=sys.stderr' for err. ``` ### Regex ```python import re re.sub(, new, text, count=0) re.search(, text) # Searches for first occurrence of pattern. re.match(, text) # Searches only at the beginning of the string. re.findall(, text) re.split(, text, maxsplit=0) # Use brackets in regex to keep the matches. ``` **'Search' and 'match' functions return a 'Match' object. Use '.group()' method on it to get the whole match, or '.group(1)' to get the part in first bracket.** **Parameter 'flags=re.IGNORECASE' can be used with all functions. Parameter 'flags=re.DOTALL' makes dot also accept newline.** **Use '\\\\1' or r'\1' for backreference.** #### Special Sequences: ```python # Use capital letter for negation. '\d' == '[0-9]' # Digit '\s' == '[ \t\n\r\f\v]' # Whitespace '\w' == '[a-zA-Z0-9_]' # Alphanumeric ``` ### Format ```python '{}'.format( [, , ...]) ``` ```python {:min_width} # ' ' {:>min_width} # ' ' {:^min_width} # ' ' {:_____' {:.max_width} # '' {:max_width.min_width} # ' ' {:max_width.no_of_decimalsf} # ' 3.14' ``` ```python >>> person = {'name': 'Jean-Luc', 'height': 187.1} >>> '{p[height]:.0f}'.format(p=person) '187' >>> f"{person['height']:.0f}" '187' ``` #### Binary, at least 10 spaces wide, filled with zeros: ```python >>> f'{123:010b}' '0001111011' ``` #### Integer presentation types: * 'b' - Binary * 'c' - Character * 'o' - Octal * 'x' - Hex * 'X' - HEX ### Text Wrap ```python import textwrap textwrap.wrap(text, width) ``` Random ------ ```python import random random.random() random.randint(from_inclusive, to_inclusive) random.shuffle() random.choice() ``` Infinity -------- ```python float('inf') ``` Datetime -------- ```python import datetime now = datetime.datetime.now() now.month # 3 now.strftime('%Y%m%d') # 20180315 now.strftime('%Y%m%d%H%M%S') # 20180315002834 ``` Arguments --------- **"*" is the splat operator, that takes a list as input, and expands it into actual positional arguments in the function call:** ```python args = (1, 2) kwargs = {'x': 3, 'y': 4, 'z': 5} func(*args, **kwargs) ``` #### Is the same as: ```python func(1, 2, x=3, y=4, z=5) ``` #### Splat operator can also be used in function declarations: ```python >>> def add(*a): ... return sum(a) >>> add(1, 2, 3) 6 ``` #### And in some other places: ```python >>> a = (1, 2, 3) >>> [*a] [1, 2, 3] ``` ```python >>> head, *body, tail = [1, 2, 3, 4] >>> body [2, 3] ``` Inline ------ ### Lambda ```python lambda: lambda , : ``` ### Comprehension ```python [i+1 for i in range(10)] # [1, 2, ..., 10] [i for i in range(10) if i>5] # [6, 7, ..., 9] {i: i*2 for i in range(10)} # {0: 0, 1: 2, ..., 9: 18} (x+5 for x in range(0, 10)) # (5, 6, ..., 14) -> Generator ``` ```python [i+j for i in range(10) for j in range(10)] ``` #### Is the same as: ```python out = [] for i in range(10): for j in range(10): out.append(i+j) ``` ### Map, Filter, Reduce ```python map(lambda x: x+1, range(10)) # [1, 2, ..., 10] filter(lambda x: x>5, range(10)) # [6, 7, ..., 9] functools.reduce(lambda sum, x: sum+x, range(10)) # 45 ``` ### Any, All ```python any(el[1] for el in ) ``` ### If - Else ```python if else ``` ```python >>> [a if a else 2 for a in [0, 1, 0, 3]] [2, 1, 2, 3] ``` ### Namedtuple, Enum, Class ```python from collections import namedtuple from enum import Enum Point = namedtuple('Point', list('xy')) Direction = Enum('Direction', list('nesw')) Creature = type('Creature', (), {'position': Point(0, 0), 'direction': Direction.n}) ``` Closure ------- ```python def multiply_closure(x): def wrapped(y): return x * y return wrapped multiply_by_3 = multiply_closure(3) ``` #### Or: ```python from functools import partial partial(, [, , ...]) ``` Decorator --------- ```python @closure_name def function_that_gets_passed_to_closure(): pass ``` #### Debugger example: ```python from functools import wraps def debug(func): @wraps(func) # Needed for metadata copying (func name, ...). def wrapper(*args, **kwargs): print(func.__name__) return func(*args, **kwargs) return wrapper @debug def add(x, y): return x + y ``` Class ----- ```python class : def __init__(self, a): self.a = a def __repr__(self): return str({'a': self.a}) # Use f'{s.__dict__}' for all members. def __str__(self): return str(self.a) ``` ### Enum ```python import enum class (enum.Enum): = = = enum.auto() # Can be used for automatic indexing. ... ``` ```python . # == [''] # == (value) # == .name # == .value # == ``` ```python Cutlery = Enum('Cutlery', ['knife', 'fork', 'spoon']) list() # == [, , ...] random.choice(list()) # == random ``` ### Copy ```python import copy copy.copy() copy.deepcopy() ``` System ------ ### Arguments ```python import sys sys.argv ``` ### Read File ```python def read_file(filename): with open(filename, encoding='utf-8') as file: return file.readlines() ``` ### Write to File ```python def write_to_file(filename, text): with open(filename, 'w', encoding='utf-8') as file: file.write(text) ``` ### Path ```python import os os.path.exists() os.path.isfile() os.path.isdir() os.listdir() ``` ### Execute Command ```python import os os.popen().read() ``` #### Or: ```python >>> import subprocess >>> a = subprocess.run(['ls', '-a'], stdout=subprocess.PIPE) >>> a.stdout b'.\n..\nfile1.txt\nfile2.txt\n' >>> a.returncode 0 ``` ### Input ```python filename = input('Enter a file name: ') ``` #### Prints lines until EOF: ```python while True: try: print(input()) except EOFError: break ``` JSON ---- ```python import json ``` ### Serialization ```python = json.dumps(, ensure_ascii=True, indent=None) = json.loads() ``` ### Read File ```python def read_json_file(filename): with open(filename, encoding='utf-8') as file: return json.load(file) ``` ### Write to File ```python def write_to_json_file(filename, an_object): with open(filename, 'w', encoding='utf-8') as file: json.dump(an_object, file, ensure_ascii=False, indent=2) ``` SQLite ------ ```python import sqlite3 db = sqlite3.connect(filename) ``` ### Read ```python cursor = db.execute() if cursor: cursor.fetchall() # Or cursor.fetchone() db.close() ``` ### Write ```python db.execute() db.commit() ``` Exceptions ---------- ```python while True: try: x = int(input("Please enter a number: ")) except ValueError: print("Oops! That was no valid number. Try again...") else: print("Thank you.") break ``` Threading --------- ```python import threading ``` ### Thread ```python thread = threading.Thread(target=, args=(, )) thread.start() thread.join() ``` ### Lock ```python lock = threading.Rlock() lock.acquire() lock.release() ``` Itertools --------- **Every function returns a generator and can accept any collection. If you want to print an output of generator, as in examples, you need to pass it to the list() function.** ```python from itertools import * ``` ### Chain ```python >>> chain([1, 2], range(3, 5)) [1, 2, 3, 4] ``` ### Combinations ```python >>> combinations("abc", 2) [('a', 'b'), ('a', 'c'), ('b', 'c')] ``` ### Permutations ```python >>> permutations("abc", 2) [('a', 'b'), ('a', 'c'), ('b', 'a'), ('b', 'c'), ('c', 'a'), ('c', 'b')] ``` ### Product ```python >>> list(product('ab', [1, 2])) [('a', 1), ('a', 2), ('b', 1), ('b', 2)] ``` ### Compress ```python >>> compress("abc", [True, 0, 23]) ['a', 'c'] ``` ### Count ```python >>> a = count(5, 2) >>> next(a), next(a) (5, 7) ``` ### Cycle ```python >>> a = cycle("abc") >>> [next(a) for _ in range(10)] ['a', 'b', 'c', 'a', 'b', 'c', 'a', 'b', 'c', 'a'] ``` ### Groupby ```python >>> a = [{"id": 1, "name": "bob"}, {"id": 2, "name": "bob"}, {"id": 3, "name": "peter"}] >>> {k: list(v) for k, v in groupby(a, key=lambda x: x["name"])} {'bob': [{'id': 1, 'name': 'bob'}, {'id': 2, 'name': 'bob'}], 'peter': [{'id': 3, 'name': 'peter'}]} ``` ### Islice ```python islice([1, 2, 3], 1, None) [2, 3] ``` ### Ifilter, imap and izip #### Filter, map and zip functions that return generators instead of iterators. Introspection and Metaprograming -------------------------------- **Inspecting code at runtime and code that generates code. You can:** * **Look at the attributes** * **Set new attributes** * **Create functions dynamically** * **Traverse the parent classes** * **Change values in the class** ```python >>> class Z: ... def __init__(self): ... self.a = 'abcde' ... self.b = 12345 >>> z = Z() ``` ### Getattr, Hasattr, Setattr ```python >>> getattr(z, 'a') # Same as Z.__getattribute__(z, 'a') 'abcde' >>> hasattr(z, 'c') False >>> setattr(z, 'c', 10) ``` ### Type **Type is the root class. If only passed the object it returns it's type. Otherwise it creates a new class (and not the instance!):** ```python type(class_name, parents, attributes) ``` ```python >>> Z = type('Z', (), {'a': 'abcde', 'b': 12345}) >>> z = Z() ``` ### MetaClass #### Class that creates class: ```python def my_meta_class(name, parents, attrs): ... return type(name, parents, attrs) ``` #### Or: ```python class MyMetaClass(type): def __new__(klass, name, parents, attrs): ... return type.__new__(klass, name, parents, attrs) ``` ### Metaclass Attr **When class is created it checks if it has metaclass defined. If not, it recursively checks if any of his parents has it defined, and eventually comes to type:** ```python class BlaBla: __metaclass__ = Bla ``` Eval ---- ```python import ast import operator as op # Supported operators operators = {ast.Add: op.add, ast.Sub: op.sub, ast.Mult: op.mul, ast.Div: op.truediv, ast.Pow: op.pow, ast.BitXor: op.xor, ast.USub: op.neg} def eval_expr(expr): return eval_(ast.parse(expr, mode='eval').body) def eval_(node): if isinstance(node, ast.Num): # return node.n elif isinstance(node, ast.BinOp): # return operators[type(node.op)](eval_(node.left), eval_(node.right)) elif isinstance(node, ast.UnaryOp): # e.g., -1 return operators[type(node.op)](eval_(node.operand)) else: raise TypeError(node) ``` ```python >>> eval_expr('2^6') 4 >>> eval_expr('2**6') 64 >>> eval_expr('1 + 2*3**(4^5) / (6 + -7)') -5.0 ```


Libraries ========= Plot ---- ```python # pip3 install matplotlib from matplotlib import pyplot pyplot.plot( [, ]) pyplot.show() pyplot.savefig(filename) ``` Web --- ```python # pip3 install bottle import bottle import urllib ``` ### Run ```python bottle.run(host='localhost', port=8080) bottle.run(host='0.0.0.0', port=80, server='cherrypy') ``` ### Static request ```python @route('/img/') def send_image(image): return static_file(image, 'images/', mimetype='image/png') ``` ### Dynamic request ```python @route('/') def send_page(sport): sport = urllib.parse.unquote(sport).lower() page = read_file(sport) return template(page) ``` ### REST request ```python @post('/p/') def p_handler(sport): team = request.forms.get('team') team = urllib.parse.unquote(team).lower() db = sqlite3.connect(conf.DB_PATH) p_h, p_a = get_p(db, sport, team) db.close() response.headers['Content-Type'] = 'application/json' response.headers['Cache-Control'] = 'no-cache' return json.dumps([p_h, p_a]) ``` Curses ------ ```python # pip3 install curses import curses def main(): curses.wrapper(draw) def draw(screen): screen.clear() screen.addstr(0, 0, "Press ESC to quit.") while screen.getch() != 27: pass def get_border(screen): Coords = collections.namedtuple('Coords', ['x', 'y']) height, width = screen.getmaxyx() return Coords(width - 1, height - 1) ``` #### Gets char from int: ```python = chr() = ord() ``` Profile ------- #### Basic: ```python from time import time start_time = time() duration = time() - start_time ``` #### Times execution of the passed code: ```python from timeit import timeit timeit('"-".join(str(n) for n in range(100))', number=1000000) ``` #### Generates a PNG image of call graph and highlights the bottlenecks: ```python # pip3 install pycallgraph import pycallgraph graph = pycallgraph.output.GraphvizOutput() graph.output_file = get_filename() with pycallgraph.PyCallGraph(output=graph): ``` #### Utility code for unique PNG filenames: ```python def get_filename(): return "{}-{}.png".format("profile", get_current_datetime_string()) def get_current_datetime_string(): now = datetime.datetime.now() return get_datetime_string(now) def get_datetime_string(a_datetime): return a_datetime.strftime('%Y%m%d%H%M%S') ``` Audio ----- #### Saves list of floats of size 0 to 1 to a WAV file: ```python import wave, struct frames = [struct.pack("%dh"%(1), int((a-0.5)*60000)) for a in ] wf = wave.open(filename, 'wb') wf.setnchannels(1) wf.setsampwidth(4) wf.setframerate(44100) wf.writeframes(b''.join(frames)) wf.close() ``` Progress Bar ------------ #### Basic: ```python import sys class Bar(): @staticmethod def range(*args): bar = Bar(len(list(range(*args)))) for i in range(*args): yield i bar.tick() def __init__(s, steps, width=40): s.st, s.wi, s.fl, s.i = steps, width, 0, 0 s.th = s.fl * s.st / s.wi s.p(f"[{' ' * s.wi}]") s.p('\b' * (s.wi + 1)) def tick(s): s.i += 1 while s.i > s.th: s.fl += 1 s.th = s.fl * s.st / s.wi s.p('-') if s.i == s.st: s.p('\n') def p(s, t): sys.stdout.write(t) sys.stdout.flush() ``` ```python from time import sleep for i in Bar.range(100): sleep(0.02) ``` #### Progress: ```python # pip3 install progress from progress.bar import Bar from time import sleep STEPS = 100 bar = Bar('Processing', max=STEPS) for i in range(STEPS): sleep(0.02) bar.next() bar.finish() ```