{ "cells": [ { "cell_type": "markdown", "metadata": { "id": "CItwGnEDMlCR" }, "source": [ "# **Pauta Auxiliar 3: Recursión y Diagramas de Estado**\n", "\n", "### Profesores: Nelson Baloian, Benjamín Bustos, Sebastian Ferrada, Patricio Poblete\n", "### Auxiliares: Sebastián Acuña, Valentina Aravena, Vicente Olivares, Ricardo Valdivia" ] }, { "cell_type": "markdown", "metadata": { "id": "B5CogT3yCCg2" }, "source": [ "## P1. Triángulo de Pascal" ] }, { "cell_type": "markdown", "metadata": { "id": "2HN4N7BFi4Nh" }, "source": [ "La idea del ejercicio es usar como caso recursivo lo siguiente:\n", "\\begin{equation*}\n", "\\binom{n}{k} = \\binom{n-1}{k-1} + \\binom{n−1}{k}\n", "\\end{equation*}\n", "hasta llegar a los casos base:\n", "\\begin{equation*}\n", "\\binom{n}{n} = \\binom{n}{0} = 1\n", "\\end{equation*}" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "b3zjjeq0Cq5q" }, "outputs": [], "source": [ "# calcula la combinatoria de n sobre k\n", "# n, k enteros no negativos, n >= k\n", "# Ejemplo: combinatoria(5, 2) devuelve 10\n", "\n", "def combinatoria(n, k):\n", " # Casos base (n = k y n = 0)\n", " if n==k or k==0:\n", " return 1\n", "\n", " # Casos recursivos\n", " else:\n", " # Las variables comb1 y comb2 representan las combinatorias en las que se separa\n", " comb1 = combinatoria(n-1, k-1)\n", " comb2 = combinatoria(n-1, k)\n", " return comb1 + comb2\n" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "colab": { "base_uri": "https://localhost:8080/" }, "id": "40rOHXgzDX3X", "outputId": "04ca7917-6cec-4146-d916-3994af86520c" }, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Todo funciona!!\n", "Estrellita dorada para ti\n" ] } ], "source": [ "assert combinatoria(5,2) == 10\n", "assert combinatoria(10,6) == 210\n", "assert combinatoria(4,4) == 1\n", "assert combinatoria(5,0) == 1\n", "assert combinatoria(0,0) == 1\n", "assert combinatoria(101,2) == 5050\n", "print(\"Todo funciona!!\")\n", "print(\"Estrellita dorada para ti\")" ] }, { "cell_type": "markdown", "metadata": { "id": "d84QBFB2DQTr" }, "source": [ "## P2. Un juego con monedas" ] }, { "cell_type": "markdown", "metadata": { "id": "vKZg6LJbZVzl" }, "source": [ "Alicia y Roberto, son dos niños que les encanta jugar con monedas. Un día Alicia llega a la casa de Roberto con un nuevo juego que acaba de inventar. Usando una moneda no cargada (es decir, P(sello) = P(cara) = 1/2), se lanza la moneda al aire una cantidad indeterminada de veces, hasta que se cumpla alguna de las siguientes situaciones:\n", "\n", "• Si sale de corrido cara, cara, sello. Alicia gana el juego.\n", "\n", "• Si sale de corrido cara, sello, sello. Roberto gana el juego.\n", "\n", "Roberto cree que este juego no es justo, por esto te pide ayuda a ti, su amigo/a computín. ¿Puedes ayudar a Roberto a decidir si es justo o no el juego?" ] }, { "cell_type": "markdown", "metadata": { "id": "N6X680FaZafl" }, "source": [ "### Diagrama de Estado del problema\n", "![Monedas_color.jpg]()" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "QchL71PTZcGy" }, "outputs": [], "source": [ "import random\n", "def lanzarMoneda():\n", " if (random.random() < 0.5):\n", " return True\n", " return False\n", "\n", "def juegoConMonedas():\n", " estado = \"Morado\"\n", " while(estado!=\"Naranja\" and estado!=\"Azul\"):\n", " if lanzarMoneda(): # Salio Cara\n", " if estado==\"Morado\":\n", " estado=\"Verde\"\n", " elif estado==\"Verde\":\n", " estado=\"Amarillo\"\n", " elif estado==\"Amarillo\":\n", " estado=\"Amarillo\"\n", " else:\n", " estado=\"Verde\"\n", " else: # Salio Sello\n", " if estado==\"Morado\":\n", " estado=\"Morado\"\n", " elif estado==\"Verde\":\n", " estado=\"Gris\"\n", " elif estado==\"Amarillo\":\n", " estado=\"Naranja\"\n", " else:\n", " estado=\"Azul\"\n", " if estado==\"Naranja\":\n", " return False\n", " else:\n", " return True\n" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "colab": { "background_save": true }, "id": "9HerMoqjcQsW", "outputId": "83bf4c66-46ba-41e9-e16e-f7ff6fbf407c" }, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Alicia: 666676262\n", "Roberto: 333323738\n" ] } ], "source": [ "n = 1000000000\n", "A = 0\n", "R = 0\n", "while(n > 0):\n", " n -= 1\n", " if juegoConMonedas():\n", " R += 1\n", " else:\n", " A += 1\n", "\n", "print(\"Alicia: \",A)\n", "print(\"Roberto: \",R)" ] }, { "cell_type": "markdown", "metadata": { "id": "vsYm5N-5Cs5K" }, "source": [ "## P3. Factores Primos" ] }, { "cell_type": "markdown", "metadata": { "id": "AG-zJlKOY4yO" }, "source": [ "Como se menciona en el enunciado de la pregunta, los números enteros positivos se pueden representar separándolo en sus factores primos. En particular, nos interesa que se puede ir separando factor por factor. Por ejemplo:\n", "\n", "\\begin{align*}\n", "90 &= 2 * 45 \\\\\n", "&= 2 * 3 * 15 \\\\\n", "&= 2 * 3 * 3 * 5\n", "\\end{align*}\n", "\n", "Podemos aprovechar lo anterior para la recursión de nuestra función, esta funcionaría de la siguiente manera:\n", "\n", "\n", "* Si n es primo, retorna **str(n)**.\n", "* En caso contrario, se busca el factor primo *p* más pequeño y se retorna **str(p) + \"*\" + factorPrimo(n//p)**.\n", "\n", "\n", "Siguiendo el ejemplo de arriba, la función para n = 90 funcionaría de la siguiente manera:\n", "\n", "\\begin{align*}\n", "factoresPrimos(90) &= 2 * factoresPrimos(45)\\\\\n", "&= 2 * 3 * factoresPrimos(15) \\\\\n", "&= 2 * 3 * 3 * factoresPrimos(5)\\\\\n", "&= 2 * 3 * 3 * 5\n", "\\end{align*}\n", "\n", "Ahora, ¿cómo determinamos si n es primo o cual es su factor primo más pequeño?\n", "\n", "\n", "Primero tenemos que tener en cuenta que el divisor d más pequeño distinto de 1 de un número entero positivo n siempre es primo. Esto se puede demostrar por contradicción: Si d no fuera primo se podría descomponer en dos factores que serían divisores de n y menores que d, contradiciendo que d es el divisor más pequeño de n.\n", "\n", "(Demostración no necesaria para el curso, solo para que decir que todavía nos acordamos de cómo demostrar 😀)\n", "\n", "\n", "Teniendo lo anterior en cuenta, podemos buscar el número más pequeño que divida a n. Si no encontramos uno, n es primo.\n", "\n", "Pero no es necesario probar con todos los enteros desde 2 hasta $n-1$, con llegar hasta $\\sqrt{n}$ basta. Esto se debe a que si existiese un divisor $d_1$ de n mayor que $\\sqrt{n}$, también debe existir un divisor $d_2$ menor a $\\sqrt{n}$, tal que $d_1*d_2=n$. Por esa razón no es necesario buscar divisores superiores a $\\sqrt{n}$.\n", "\n", "El siguiente código representa al texto anterior:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "gbBBSYk8DQDf" }, "outputs": [], "source": [ "import numpy as np\n", "\n", "def factoresPrimos(n):\n", " limite = int(np.sqrt(n))\n", " div = 2\n", "\n", " # div aumenta hasta que sea divisor de n o hasta que el divisor sea mayor que sqrt(n)\n", " while n % div != 0 and div <= limite:\n", " div += 1\n", "\n", " # Caso base\n", " # Si div es mayor que limite, n es primo\n", " if div > limite:\n", " return str(n)\n", " # Llamada recursiva\n", " else:\n", " return str(div)+\"*\"+factoresPrimos(n//div)" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "colab": { "base_uri": "https://localhost:8080/" }, "id": "gKMktVXQ4VoA", "outputId": "96a4a96b-7807-46b8-ac81-0e638e7b484d" }, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Todo de maravilla\n" ] } ], "source": [ "assert factoresPrimos(2) == \"2\"\n", "assert factoresPrimos(23) == \"23\"\n", "assert factoresPrimos(10) == \"2*5\"\n", "assert factoresPrimos(90) == \"2*3*3*5\"\n", "print(\"Todo de maravilla\")" ] } ], "metadata": { "colab": { "provenance": [] }, "kernelspec": { "display_name": "Python 3", "name": "python3" }, "language_info": { "name": "python" } }, "nbformat": 4, "nbformat_minor": 0 }