Hola! 👨💻. Somos estudiantes del curso de Programación III de UTEC. Hemos aplicado conocimientos aprendidos durante el curso de Programación para desarrollar el juego Maze en C++.
Tabla de contenidos:
Maze es un juego adictivo que tiene una fórmula clásica, debes deslizar y guiar al personaje a través del laberinto pero, ¡Cuidado! El tiempo puede estar corriendo. Tendrás que ser ágil al resolverlo para encontrar el camino adecuado para que llegue a su punto objetivo.
- Nuestro tablero es modificable.
- Tenemos dos opciones de juego: Bot y Human.
- Tiene timer.
- 4 Algoritmos distintos para el bot (DFS, BFS, GBFS, A*)
- Lenguaje de programación C++20 o posterior
- Librería raylib
- Raylib-cpp (header only, fork)
- Doxyegn (documentación)
- El uso de raylib como interfaz gráfica.
- Algoritmos de busqueda
- Algoritmos de generación de laberintos perfectos (arboles de expansión minima)
Este algoritmo emplea en su implementación el concepto de stack. Lo que hace es recorrer en su totalidad una estructura. Cada vez que se encuentra con dos o más caminos posibles, este recorrerá cada uno lo más profundamente posible, a la vez que almacena los nodos recorridos en un stack. Si alcanza un camino sin salida antes que el objetivo, este retrocede a la bifurcación anterior y repite el proceso. Este algoritmo, aunque puede cumplir con el objetivo, no es eficiente en lo absoluto, considerando que existen alternativas más eficientes.
Una búsqueda en anchura (BFS) es un algoritmo de búsqueda, recorre los nodos de un grafo, comenzando en la raíz para luego explorar todos los vecinos de este nodo. Además, para cada uno de los vecinos se exploran sus respectivos vecinos adyacentes, y así hasta que se recorra todo el grafo.
BFS va formando un árbol a medida que va recorriendo un grafo y se usa para algoritmos en donde resulta crítico elegir el mejor camino posible en cada momento del recorrido.
Este es un algoritmo de búsqueda heurística, es decir, necesita información adicional específica relacionada con el problema a resolver. En el caso de un laberinto, lo que requiere es el número de casillas de distancia a la que se encuentra cada casilla del objetivo. Lo que el algoritmo GBFS busca es reducir este número conforme se mueve de casilla en casilla. Si alcanza un punto sin salida, retrocede de nuevo al último cruce. A pesar de siempre buscar la casilla más cercana al objetivo, no puede garantizar que el camino elegido será el más eficiente.
Tenemos una celda inicial y una celda objetivo. Queremos llegar a la celda objetivo (si es posible) desde la celda inicial lo más rápido posible. Aquí A* Search Algorithm lo que hace es que, en cada paso, selecciona el nodo que le acorte o le permita llegar de manera más rapida u optimizada al punto de objetivo. Esto a menudo se denomina heurística, que no es más que una especie de suposición inteligente. Realmente no sabemos la distancia real hasta que encontramos el camino, porque todo tipo de cosas pueden estar en el camino.
El algoritmo de Wilson utiliza caminatas aleatorias borradas en bucle para generar un árbol de expansión uniforme, una muestra imparcial de todos los árboles de expansión posibles. Este inicializa el laberinto con una celda de inicio arbitraria. Luego, se agrega una nueva celda al laberinto, iniciando una caminata aleatoria. La caminata aleatoria continúa hasta que se vuelve a conectar con el laberinto existente. Sin embargo, si la caminata aleatoria se cruza a sí misma, el bucle resultante se borra antes de que continúe la caminata aleatoria.
- Compare: función heuristica para GBFS/A*
- Solve: función para resolver el laberinto
- Class Game/GameBase: template en función del modo de juego
- Utils::RandomNum: función robusta para generar aleatorios
- DrawAll: Función para dibujar caulquier tipo T con {T.Draw()}
- Compilador compatible con c++20 o posterior
- Administrador de paquetes Cmake v3 o posterior
- Libreria raylib (previamente no incluida)
- Clonación de repositorio con
git clone https://github.com/CS1103/proyecto-final-2023_0-grupo-4.git
cd proyecto-final-2023_0-grupo-4
mkdir build
cmake -B build
cd build
sudo make install
- ejecutable Maze
Maze
- Crear entrada de escritorio
cd .. # Carpeta principal del repositorio cp Maze.desktop /usr/share/applications/
- Elegir tamaño del tablero
- Elegir modo de juego (Humano o computador)
- Elegir limite de tiempo prendido o apagado
-
El laberinto tiene un punto inicial.
-
Luego, debes elegir un cuadrado adyacente, ya sea hacia adelante o hacia un lado.
-
Continuas creando el camino que creas adecuado.
-
En caso llegues a un callejon sin salida, no podras avanzar, pero si retroceder.
-
Finalmente, al crear el camino correcto, llegarás al punto objetivo.
Distribuido bajo la licencia MIT. Ver LICENSE
para más información.
Decidimos usar una matriz como estructura de datos debido a que el laberinto es cuadrado y cada una de las casillas contiene información relevante.
- Camila Villasana Boggiano
- Diana Carolina Ñañez Andres
- Jaime Alfonso Ramos Talla
- Kevin Jonás Zevallos López
- Enrique Francisco Flores Teniente
- Luis Enrique Cortijo
-
CS50 (2020). Search - Lecture 0 - CS50's Introduction to Artificial Intelligence with Python 2020 https://cs50.harvard.edu/ai/2020/notes/0/
Utilizamos recursos provistos por un curso libre de harvard para aprender a fondo el funcionamiento de los distintos algoritmos de busqueda. -
Thee Buckblock (2011) - Jamis Buck https://weblog.jamisbuck.org/2011/1/20/maze-generation-wilson-s-algorithm
Utilizamos el block del desarrollador Jamis Buck para comparar y entender los diversos algorittmos de generación de laberintos existentes, en su block tiene multiples articulos para cada uno de los algoritmos junto con una comparación de ellos. -
Meyer's Singleto https://laristra.github.io/flecsi/src/developer-guide/patterns/meyers_singleton.html
En esta pagina encontramos un snipet para un singleto que nos llevo a entender más sobre el fiasco de inicialización y construir las funciónes estaticas de las texturas. -
Mejores prácticas para seguir con el proyecto y patrones de diseño para analizar. Refactoring.Guru. (s. f.-b). Refactoring and Design Patterns. https://refactoring.guru/
En esta pagina encontramos amplia información sobre refactoring, mejores practicas y, sobre todo, patrones de diseño. La utilizamos principalemente para el análisis de patrones enpatrones.md
. -
Documentación de Raylib y Raylib-cpp
https://www.raylib.com/cheatsheet/cheatsheet.html
https://robloach.github.io/raylib-cpp/
Documentación utilizada para la parte grafica del proyecto.