{ "cells": [ { "cell_type": "markdown", "metadata": {}, "source": [ "# Trees\n", "\"Open\n", "\n", "http://openbookproject.net/thinkcs/python/english3e/trees.html\n", "- like linked lists, trees are made up of nodes\n", "\n", "## Binary Tree\n", " - is a commonly used tree in which each node contains a reference to atmost two other nodes (possibly None)\n", "- these references are referred to as the left and right subtrees\n", "- like the node of linked list, each node also contains data/cargo\n", "- like linked lists, trees are recursive data structures that are defined recursively:\n", " 1. the empty tree, represented by None, or\n", " 2. a node that contains a data and two tree references (left and right subtree)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Building trees\n", "- similar to building linked-list" ] }, { "cell_type": "code", "execution_count": 1, "metadata": {}, "outputs": [], "source": [ "class Tree:\n", " def __init__(self, data, left=None, right=None):\n", " self.cargo = data\n", " self.left = left\n", " self.right = right\n", " \n", " def __str__(self):\n", " return \"{}\".format(self.cargo)\n", " " ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### bottom-up way to build-trees\n", "- first create children and link them to the parent" ] }, { "cell_type": "code", "execution_count": 4, "metadata": {}, "outputs": [], "source": [ "left = Tree(2)\n", "right = Tree(3)\n", "tree = Tree(1, left, right)" ] }, { "cell_type": "code", "execution_count": 5, "metadata": {}, "outputs": [], "source": [ "tree1 = Tree(10, Tree(20), Tree(30))" ] }, { "cell_type": "code", "execution_count": 6, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "1\n" ] } ], "source": [ "print(tree)" ] }, { "cell_type": "code", "execution_count": 7, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "10\n" ] } ], "source": [ "print(tree1)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## traversing tree\n", "- natural way to traverse a tree is recursively!" ] }, { "cell_type": "code", "execution_count": 8, "metadata": {}, "outputs": [], "source": [ "def findSum(tree):\n", " if not tree:\n", " return 0\n", " return tree.cargo + findSum(tree.left) + findSum(tree.right)\n", " " ] }, { "cell_type": "code", "execution_count": 9, "metadata": {}, "outputs": [ { "data": { "text/plain": [ "6" ] }, "execution_count": 9, "metadata": {}, "output_type": "execute_result" } ], "source": [ "findSum(tree)" ] }, { "cell_type": "code", "execution_count": 10, "metadata": {}, "outputs": [ { "data": { "text/plain": [ "60" ] }, "execution_count": 10, "metadata": {}, "output_type": "execute_result" } ], "source": [ "findSum(tree1)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Expression trees\n", "- trees are natural way to represent the structure of an expression unambigigously.\n", "- infix expression 1 + 2 * 3 is ambigious unless we know the order of operation that * happens before +\n", "- we can use tree to represent the same expression\n", " - operands are leaf nodes\n", " - operator nodes contain references to their operands; operators are binary (two operands)\n", " \n", "\n", "\n", "- applications:\n", " - translate expressions to postfix, prefix, and infix\n", " - compilers use expression trees to parse, optimize, and translate programs\n", " \n", "- three ways to traverse trees: pre-order, in-order and post-order" ] }, { "cell_type": "code", "execution_count": 18, "metadata": {}, "outputs": [], "source": [ "expression = Tree('+', Tree(1), Tree('*', Tree(2), Tree(3)))" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### pre-order tree traversal\n", "- contents of the root appear before the contents of the children\n", "- recursive algorithm:\n", " - visit the node\n", " - visit left subtree\n", " - visit right subtree" ] }, { "cell_type": "code", "execution_count": 19, "metadata": {}, "outputs": [], "source": [ "def preorder(tree):\n", " if not tree:\n", " return\n", " print(tree.cargo, end=' ')\n", " preorder(tree.left)\n", " preorder(tree.right)\n", " " ] }, { "cell_type": "code", "execution_count": 20, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "+ 1 * 2 3 " ] } ], "source": [ "preorder(expression)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### in-order tree traversal\n", "- contents of the tree appear in order\n", "- recursive algorithm:\n", " - visit left subtree\n", " - visit node\n", " - visit right subtree" ] }, { "cell_type": "code", "execution_count": 21, "metadata": {}, "outputs": [], "source": [ "def inorder(tree):\n", " if not tree:\n", " return\n", " inorder(tree.left)\n", " print(tree.cargo, end=' ')\n", " inorder(tree.right)" ] }, { "cell_type": "code", "execution_count": 22, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "1 + 2 * 3 " ] } ], "source": [ "inorder(expression)" ] }, { "cell_type": "code", "execution_count": 29, "metadata": {}, "outputs": [], "source": [ "def inorderIndented(tree, level=0):\n", " if not tree:\n", " return\n", " inorderIndented(tree.right, level+1)\n", " print(' '*level + str(tree.cargo))\n", " inorderIndented(tree.left, level+1)" ] }, { "cell_type": "code", "execution_count": 30, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ " 3\n", " *\n", " 2\n", "+\n", " 1\n" ] } ], "source": [ "inorderIndented(expression)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### post-order traversal\n", "- recursive algorithm:\n", " 1. visit left subtree\n", " 2. visit right subtree\n", " 3. visit node" ] }, { "cell_type": "code", "execution_count": 33, "metadata": {}, "outputs": [], "source": [ "def postorder(tree):\n", " if not tree:\n", " return\n", " postorder(tree.left)\n", " postorder(tree.right)\n", " print(tree.cargo, end=' ')" ] }, { "cell_type": "code", "execution_count": 34, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "1 2 3 * + " ] } ], "source": [ "postorder(expression)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## building an expression tree\n", "- parse an infix expression and build the corresponding expression tree\n", "- e.g., (3 + 7) * 9 yields the following tree:\n", "" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "1. tokenize expression into python list? How (left as an exercise)\n", " - (3 + 7) * 9 = [\"(\", 3, \"+\", 7, \")\", \"*\", 9, \"end\"]\n", "2. \"end\" token is useful for preventing the parser from reading pas the end of the list" ] }, { "cell_type": "code", "execution_count": 31, "metadata": {}, "outputs": [], "source": [ "def get_token(token_list, expected):\n", " if token_list[0] == expected:\n", " del token_list[0]\n", " return True\n", " return False" ] }, { "cell_type": "code", "execution_count": 36, "metadata": {}, "outputs": [], "source": [ "# handles operands\n", "def get_number(token_list):\n", " x = token_list[0]\n", " if not isinstance(x, int):\n", " return None\n", " del token_list[0]\n", " return Tree(x, None, None) # leaf node" ] }, { "cell_type": "code", "execution_count": 37, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "9 " ] } ], "source": [ "token_list = [9, 11, 'end']\n", "x = get_number(token_list)\n", "postorder(x)" ] }, { "cell_type": "code", "execution_count": 38, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "[11, 'end']\n" ] } ], "source": [ "print(token_list)" ] }, { "cell_type": "code", "execution_count": 39, "metadata": {}, "outputs": [], "source": [ "def get_product(token_list):\n", " a = get_number(token_list)\n", " if get_token(token_list, '*'):\n", " b = get_number(token_list)\n", " return Tree('*', a, b)\n", " return a" ] }, { "cell_type": "code", "execution_count": 40, "metadata": {}, "outputs": [], "source": [ "token_list = [9, '*', 11, 'end']\n", "tree = get_product(token_list)" ] }, { "cell_type": "code", "execution_count": 42, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "9 11 * " ] } ], "source": [ "postorder(tree)" ] }, { "cell_type": "code", "execution_count": 44, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "9 " ] } ], "source": [ "token_list = [9, '+', 11, 'end']\n", "tree = get_product(token_list)\n", "postorder(tree)" ] }, { "cell_type": "code", "execution_count": 45, "metadata": {}, "outputs": [], "source": [ "# adapt the function for compound product such as 3 * (5 * (7 * 9))\n", "def get_product(token_list):\n", " a = get_number(token_list)\n", " if get_token(token_list, '*'):\n", " b = get_product(token_list)\n", " return Tree('*', a, b)\n", " return a" ] }, { "cell_type": "code", "execution_count": 46, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "2 3 5 7 * * * " ] } ], "source": [ "token_list = [2, \"*\", 3, \"*\", 5 , \"*\", 7, \"end\"]\n", "tree = get_product(token_list)\n", "postorder(tree)" ] }, { "cell_type": "code", "execution_count": 47, "metadata": {}, "outputs": [], "source": [ "# a sum can be a tree with + at the root, a product on the left, and a sum on the right. \n", "# Or, a sum can be just a product.\n", "def get_sum(token_list):\n", " a = get_product(token_list)\n", " if get_token(token_list, \"+\"):\n", " b = get_sum(token_list)\n", " return Tree(\"+\", a, b)\n", " return a" ] }, { "cell_type": "code", "execution_count": 48, "metadata": {}, "outputs": [], "source": [ "token_list = [9, \"*\", 11, \"+\", 5, \"*\", 7, \"end\"]\n", "tree = get_sum(token_list)" ] }, { "cell_type": "code", "execution_count": 49, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "9 11 * 5 7 * + " ] } ], "source": [ "postorder(tree)" ] }, { "cell_type": "code", "execution_count": 52, "metadata": {}, "outputs": [], "source": [ "# handle parenthesis\n", "def get_number(token_list):\n", " if get_token(token_list, \"(\"):\n", " x = get_sum(token_list) # Get the subexpression\n", " get_token(token_list, \")\") # Remove the closing parenthesis\n", " return x\n", " else:\n", " x = token_list[0]\n", " if not isinstance(x, int):\n", " return None\n", " del token_list[0]\n", " return Tree(x, None, None)" ] }, { "cell_type": "code", "execution_count": 53, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "9 11 5 + 7 * * " ] } ], "source": [ "# 9 * (11 + 5) * 7\n", "token_list = [9, \"*\", \"(\", 11, \"+\", 5, \")\", \"*\", 7, \"end\"]\n", "tree = get_sum(token_list)\n", "postorder(tree)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## handling errors on malformed expressions" ] }, { "cell_type": "code", "execution_count": 54, "metadata": {}, "outputs": [], "source": [ "# handle parenthesis\n", "def get_number(token_list):\n", " if get_token(token_list, \"(\"):\n", " x = get_sum(token_list) # Get the subexpression\n", " if not get_token(token_list, \")\"): # Remove the closing parenthesis\n", " raise ValueError('Missing close parenthesis!')\n", " return x\n", " else:\n", " x = token_list[0]\n", " if not isinstance(x, int):\n", " return None\n", " del token_list[0]\n", " return Tree(x, None, None)" ] }, { "cell_type": "code", "execution_count": 55, "metadata": {}, "outputs": [ { "ename": "ValueError", "evalue": "Missing close parenthesis!", "output_type": "error", "traceback": [ "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m", "\u001b[0;31mValueError\u001b[0m Traceback (most recent call last)", "\u001b[0;32m\u001b[0m in \u001b[0;36m\u001b[0;34m()\u001b[0m\n\u001b[1;32m 1\u001b[0m \u001b[0mtoken_list\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0;36m9\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\"*\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\"(\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;36m11\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\"+\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;36m5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\"*\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;36m7\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\"end\"\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 2\u001b[0;31m \u001b[0mtree\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mget_sum\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m 3\u001b[0m \u001b[0mpostorder\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtree\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n", "\u001b[0;32m\u001b[0m in \u001b[0;36mget_sum\u001b[0;34m(token_list)\u001b[0m\n\u001b[1;32m 2\u001b[0m \u001b[0;31m# Or, a sum can be just a product.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m 3\u001b[0m \u001b[0;32mdef\u001b[0m \u001b[0mget_sum\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 4\u001b[0;31m \u001b[0ma\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mget_product\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m 5\u001b[0m \u001b[0;32mif\u001b[0m \u001b[0mget_token\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\"+\"\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m 6\u001b[0m \u001b[0mb\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mget_sum\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n", "\u001b[0;32m\u001b[0m in \u001b[0;36mget_product\u001b[0;34m(token_list)\u001b[0m\n\u001b[1;32m 2\u001b[0m \u001b[0ma\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mget_number\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m 3\u001b[0m \u001b[0;32mif\u001b[0m \u001b[0mget_token\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'*'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 4\u001b[0;31m \u001b[0mb\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mget_product\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m 5\u001b[0m \u001b[0;32mreturn\u001b[0m \u001b[0mTree\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m'*'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0ma\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mb\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m 6\u001b[0m \u001b[0;32mreturn\u001b[0m \u001b[0ma\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n", "\u001b[0;32m\u001b[0m in \u001b[0;36mget_product\u001b[0;34m(token_list)\u001b[0m\n\u001b[1;32m 1\u001b[0m \u001b[0;32mdef\u001b[0m \u001b[0mget_product\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 2\u001b[0;31m \u001b[0ma\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mget_number\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m 3\u001b[0m \u001b[0;32mif\u001b[0m \u001b[0mget_token\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'*'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m 4\u001b[0m \u001b[0mb\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mget_product\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m 5\u001b[0m \u001b[0;32mreturn\u001b[0m \u001b[0mTree\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m'*'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0ma\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mb\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n", "\u001b[0;32m\u001b[0m in \u001b[0;36mget_number\u001b[0;34m(token_list)\u001b[0m\n\u001b[1;32m 4\u001b[0m \u001b[0mx\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mget_sum\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;31m# Get the subexpression\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m 5\u001b[0m \u001b[0;32mif\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mget_token\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtoken_list\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\")\"\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0;31m# Remove the closing parenthesis\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 6\u001b[0;31m \u001b[0;32mraise\u001b[0m \u001b[0mValueError\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m'Missing close parenthesis!'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m 7\u001b[0m \u001b[0;32mreturn\u001b[0m \u001b[0mx\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m 8\u001b[0m \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n", "\u001b[0;31mValueError\u001b[0m: Missing close parenthesis!" ] } ], "source": [ "token_list = [9, \"*\", \"(\", 11, \"+\", 5, \"*\", 7, \"end\"]\n", "tree = get_sum(token_list)\n", "postorder(tree)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## exercises:" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "1. Modify inorder function so that it puts parentheses around every operator and pair of operands. Is the output correct and unambiguous? Are the parentheses always necessary?" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "2. Write a function that takes an expression string and returns a token list." ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "3. Find other places in the expression tree functions where errors can occur and add appropriate raise statements. Test your code with improperly formed expressions." ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [] } ], "metadata": { "kernelspec": { "display_name": "Python 3", "language": "python", "name": "python3" }, "language_info": { "codemirror_mode": { "name": "ipython", "version": 3 }, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.7.3" } }, "nbformat": 4, "nbformat_minor": 2 }