Lenguajes de Programación Funcional

Adentrándonos en el ámbito de la informática, el concepto de Lenguajes de Programación Funcional avanza en una liga propia, irradiando una profunda influencia en la forma en que los programadores tratan los datos y ejecutan el código. Esta lectura informativa te ofrecerá una comprensión exhaustiva de los lenguajes de Programación Funcional, definiendo sus elementos centrales, principios y los distintos tipos de lenguajes en uso. Tendiendo un puente entre la teoría y la aplicación, la última parte del texto arroja luz sobre los aspectos tangibles de la implementación de la Programación Funcional en varios lenguajes, elucidando enfoques y técnicas clave al tiempo que destaca ejemplos en diversos lenguajes de programación funcional. Permitiéndote recorrer desde los fundamentos hasta las teorías complejas, esta obra informativa pretende proporcionar una guía que lo abarque todo sobre el mundo de los lenguajes de Programación Funcional, garantizando una comprensión en profundidad de este elemento vital de la informática moderna. Tanto si eres un estudiante de informática como un programador experimentado, esta guía promete enriquecer tu base de conocimientos y mejorar tus habilidades de programación. Embarquémonos en este viaje de descubrimiento y dominio de los lenguajes de Programación Funcional.

Pruéablo tú mismo

Millones de tarjetas didácticas para ayudarte a sobresalir en tus estudios.

Regístrate gratis

Millones de tarjetas didácticas para ayudarte a sobresalir en tus estudios.
Millones de tarjetas didácticas para ayudarte a sobresalir en tus estudios.

Upload Icon

Create flashcards automatically from your own documents.

   Upload Documents
Upload Dots

FC Phone Screen

Need help with
Lenguajes de Programación Funcional?
Ask our AI Assistant

Review generated flashcards

Regístrate gratis
Has alcanzado el límite diario de IA

Comienza a aprender o crea tus propias tarjetas de aprendizaje con IA

Tarjetas de estudio
Tarjetas de estudio

Saltar a un capítulo clave

    Entender los lenguajes de programación funcionales

    Los lenguajes de programación funcionales, un tema central en el campo de la informática, desempeñan un papel fundamental en los paradigmas de programación, que simplifican el desarrollo y el mantenimiento de los programas. Con la comprensión de estos lenguajes funcionales, tendrás una base sólida en un enfoque único del desarrollo de software que contrasta con la programación más tradicional orientada a procedimientos.

    Definición de programación funcional

    La programación funcional es un paradigma de programación que trata los cálculos como la evaluación de funciones matemáticas y evita los datos mutables y de estado cambiante.

    Observablemente, el enfoque principal de los lenguajes de programación funcional se centra en la expresión de la lógica en lugar del flujo de control, donde las funciones representan relaciones entre conjuntos bien definidos de entradas y salidas. Al ser "funciones de orden superior", pueden asignarse a variables, almacenarse en estructuras de datos, devolverse desde otras funciones y pasarse como argumentos a otras funciones.

    Por ejemplo, en el lenguaje funcional "Haskell", la función para calcular el factorial de un número "n" se definiría como: factoriales n = if n<2 then 1 else n* factorial (n-1)

    En esencia, los lenguajes de programación funcionales están dotados de una serie de características únicas para optimizar el desarrollo de software, como las funciones anónimas, los cierres y las comprensiones de listas.

    Hasta la fecha, algunos de los lenguajes de programación funcional más populares son Haskell, Lisp, Scala y Erlang, entre otros.

    Principios de los lenguajes de programación funcionales

    Los lenguajes de programación funcionales se basan en algunos principios fundamentales que contribuyen a su eficacia en el desarrollo de software. Algunos de estos principios significativos son:
    • Funciones puras: Funciones que sólo dependen de la entrada suministrada y no producen efectos secundarios.
    • Datos inmutables: Una vez creado un objeto de datos, no se puede modificar.
    • Funciones de primera clase: Funciones que se tratan como cualquier otra variable. Pueden pasarse como argumentos a otras funciones, devolverse como valor desde otras funciones o asignarse como valor a una variable.

    Por ejemplo, en el lenguaje funcional Scala, una función de orden superior que toma un número entero y devuelve una función se define como def addIt(more : Int) = (x : Int) => x + more En este ejemplo, addIt es una función que produce otra función como resultado.

    Otro principio digno de mención es el concepto de recursividad como estructura primaria de control. En los lenguajes de programación funcionales, en lugar de una construcción de bucle tradicional, la recursividad se utiliza para realizar cálculos que deben ejecutarse repetidamente. Esto está en consonancia con la filosofía de "funciones hasta el final". Para encapsular estos principios, he aquí una sencilla tabla comparativa que contrasta el paradigma de programación funcional con el paradigma procedimental más común.
    Aspecto Programación funcional Programación procedimental
    Perspectiva Hace hincapié en la evaluación de funciones matemáticas Se centra en la secuencia de actividades a realizar
    Flujo de control Utiliza la recursividad como estructura de control principal Utiliza estructuras de bucle y condicionales
    Manejo de datos Datos inmutables Datos mutables
    Características de las funciones Funciones puras sin efectos secundarios Rutinas con posibles efectos secundarios
    Al comprender los principios básicos de la programación funcional, estarás bien equipado para recorrer tu camino en la programación con eficacia y utilizar un valioso conjunto de herramientas en diversos escenarios de desarrollo de software.

    Descubrir la lista de lenguajes de programación funcional

    Sumergirse en el mundo de los lenguajes de programación funcionales abre un vasto horizonte a los programadores. Existen múltiples lenguajes funcionales, cada uno con sus características y habilidades únicas que los hacen ideales para casos de uso específicos y crean un profundo impacto en el desarrollo de software. Esto refuerza la idea de que aprender programación funcional puede ser una valiosa adición al conjunto de herramientas de cualquier programador.

    Lenguajes populares de programación funcional

    En la actualidad, muchos lenguajes de programación funcional se utilizan ampliamente en el mundo académico y en la industria. Son populares debido a sus características distintivas y a su facilidad para manejar tareas complejas con un rendimiento superior. Aquí tienes una lista completa de estos lenguajes:
    • Lisp: Creado en 1958, Lisp es uno de los lenguajes de programación funcional más antiguos. Utilizado predominantemente en la industria de la inteligencia artificial (IA), sus dialectos como Scheme y Clojure son apreciados por los desarrolladores por su diseño minimalista.
    • Haskell: Conocido por su estricto sistema de tipos, Haskell es un lenguaje de programación de propósito general, estáticamente tipado y puramente funcional. Se utiliza en áreas en las que la corrección del código es primordial, como el análisis de datos, el desarrollo de compiladores y la criptografía.
    • Scala: Este lenguaje combina la programación funcional y la programación orientada a objetos, aprovechando lo mejor de ambos mundos. Resulta especialmente útil en tareas de computación distribuida y procesamiento de big data.
    • Erlang: Creado para sistemas en tiempo real, Erlang se centra en la concurrencia, la tolerancia a fallos y el rendimiento en tiempo real. Es popular para aplicaciones de telecomunicaciones, banca y comercio electrónico.
    • F#: Este lenguaje desarrollado por Microsoft se emplea en diversas aplicaciones. Puede integrarse perfectamente con otros lenguajes .NET, lo que lo hace perfecto para aplicaciones de nivel empresarial.
    • Clojure: Clojure, un lenguaje de programación dinámico y de propósito general que se centra en ser sencillo, robusto y cohesivo, es el preferido por los desarrolladores por su concurrencia y robustez.
    Cada uno de estos lenguajes, aunque arraigados en los principios de la programación funcional, ofrece un sabor único y aborda necesidades de programación específicas de forma única y eficaz.

    Definición de los lenguajes de programación funcional pura

    Un lenguaje de programación funcional puro es aquel en el que cada función es una "función pura". Tales funciones, al tomar cierta entrada, siempre producen la misma salida sin crear ningún efecto secundario. Es decir, la salida de una función depende únicamente de su entrada y no modifica ningún estado o dato externo.

    Esto se encapsula con la propiedad de "transparencia referencial", que indica que cualquier función, dada la misma entrada, siempre producirá la misma salida. Esta propiedad contribuye a una ventaja clave de los lenguajes de programación funcionales puros, que es una mayor previsibilidad y una depuración más fácil, ya que la ausencia de cambio de estado externo elimina la posibilidad de que efectos secundarios desconocidos alteren la salida del programa. Como analogía, considera una función matemática \( f(x) = x^2 \). Llamar a esta función con la misma entrada siempre produce la misma salida, independientemente de otras ocurrencias en el sistema. Del mismo modo, los lenguajes funcionales puros siguen la misma regla, lo que los hace excepcionalmente fiables. Además, los lenguajes funcionales puros tienden a facilitar un estilo de programación más declarativo. En lugar de especificar "cómo" conseguir algo (como es típico en la programación imperativa), los lenguajes de programación funcionales se centran en "qué" resultado se desea. Esto puede dar lugar a un código más legible y fácil de mantener.

    Ejemplos de lenguajes de programación funcional puros

    Aunque existen numerosos lenguajes de programación funcional, un subconjunto de ellos se clasifica como lenguajes de programación funcional "puros". Estos lenguajes aplican los principios de la programación funcional de forma más estricta que sus homólogos. He aquí algunos ejemplos:
    • Haskell: Sin duda, Haskell viene a la mente cuando se reflexiona sobre la programación funcional pura. Su robusta arquitectura y su potente sistema de tipos mantienen las funciones puras por defecto, lo que lo convierte en una opción excelente para la resolución de problemas complejos.
    • Clean: Similar a Haskell en muchos aspectos, pero se diferencia en su enfoque único para gestionar la E/S y su sintaxis limpia. Separa estrictamente las funciones puras de las tareas interactivas (E/S), garantizando que no se filtren datos no puros en el sistema.
    • Mercurio: Conocido por sus robustos y expresivos sistemas de tipos y modos, Mercury aplica la metodología de la programación lógica a la programación funcional pura. Se utiliza principalmente para crear software complejo en campos como la inteligencia artificial.
    Cada uno de estos lenguajes de programación funcional pura se adhiere estrechamente a los principios de la programación funcional y ofrece capacidades únicas que satisfacen diferentes requisitos de programación. Por tanto, son testimonio del potencial de la programación funcional en una serie de áreas de aplicación.

    Profundizando: Ejemplos de lenguajes de programación funcional

    Avanzar en tu comprensión de los lenguajes de programación funcionales implica comprender su implementación. Estudiando y practicando la implementación de la programación funcional en varios lenguajes, puedes obtener una visión amplia y matizada de este paradigma. Es intrigante ver cómo los distintos lenguajes dan vida a los principios de la programación funcional, ofreciendo una base de código estructurada y reutilizable.

    Implementación de la programación funcional en varios lenguajes

    Los principios de la programación funcional ofrecen una forma alternativa de pensar sobre la construcción de software que puede parecer radical a quienes están familiarizados con la programación procedimental y orientada a objetos. Este paradigma se centra más en "qué" debe lograr el sistema que en "cómo" debe lograr el resultado. Profundicemos en la aplicación de la programación funcional en algunos lenguajes populares:

    En JavaScript, un enfoque de programación funcional implica tratar las funciones como valores y manipularlas del mismo modo que otros tipos de datos. Un ejemplo sencillo es utilizar la función map de Array, que aplica una función a cada elemento de un array y devuelve un nuevo array con los resultados: ```html ``` En este ejemplo, la función map, cuando se combina con la función arrow, crea un elegante mecanismo para aplicar un cálculo (elevar al cuadrado) a cada elemento de un array.

    Java 8 añadió capacidades de programación funcional a su ya madura plataforma. Utilizando las expresiones lambda y las interfaces funcionales, los desarrolladores Java pueden ahora escribir código de estilo funcional.

    Por ejemplo, en Java se puede utilizar el método `forEach` para iterar sobre una lista y aplicar una función lambda, como por ejemplo: ```java ArrayList list = new ArrayList(Arrays.asList(1, 2, 3, 4, 5)); list.forEach( n -> System.out.println(n * n) ); ``` Este programa eleva al cuadrado cada elemento de la lista y lo imprime. La expresión lambda bajo el método `forEach` es una demostración clásica de programación funcional en Java.

    Por último, veamos un paradigma funcional en Python, un lenguaje dinámico que admite multitud de estilos de programación.

    En Python, puedes utilizar la función incorporada `map` para aplicar una función a una lista de elementos. ``python números = [1, 2, 3, 4, 5] al cuadrado = map(lambda x: x**2, números) print(list(al cuadrado)) #Salida: [1, 4, 9, 16, 25] ``` Hemos definido una función lambda que eleva al cuadrado la entrada, y luego hemos asignado esta función a cada elemento de nuestra lista.

    Estos ejemplos dan fe de la flexibilidad de la programación funcional, adaptable a distintos lenguajes.

    Explicación de la implementación de los lenguajes de programación funcional

    La implementación de los lenguajes de programación funcional, puros o impuros, consiste en componer funciones puras, evitando el estado compartido, los datos mutables y los efectos secundarios. Este estilo no elimina el estado, pero hace que los cambios de estado sean más controlados y predecibles. En esencia, la programación funcional se centra en unos pocos conceptos básicos:
    • Funciones de primera clasey de alto orden: En los lenguajes funcionales, las funciones son ciudadanos de primera clase, lo que significa que pueden utilizarse como cualquier otro valor. Pueden pasarse como argumentos, devolverse desde funciones y asignarse a variables. Las funciones de alto orden son las que pueden tomar funciones como parámetros o devolverlas como resultados.
    • Funcionespuras: Son funciones en las que el valor de retorno sólo está determinado por sus valores de entrada, sin efectos secundarios observables. Este aspecto se demuestra con esta función hipotética: \[ f(x) = x + 2 \] Se trata de una función pura en la que la salida (\(y\)) es únicamente un factor de la entrada (\(x\)). Tras repetidas llamadas con las mismas entradas, una función pura siempre produce los mismos resultados.
    • Inmutabilidad y apatridia: En la programación funcional, el estado y los datos son inmutables, lo que significa que no pueden modificarse tras su creación. Adoptar este enfoque elimina los problemas que surgen a causa del estado mutable y ayuda en el razonamiento de la aplicación.
    • Composición de funciones: En los paradigmas de codificación funcional, los desarrolladores de software crean funciones complejas mediante la composición de funciones más pequeñas y sencillas. La composición de funciones es similar a una composición matemática, que puede representarse como: \[ (g \circ f)(x) = g(f(x)) \] Aquí, el resultado de la función\( f(x) \) se pasa a la función \( g \), y la salida final es el resultado de \( g \).

    En los estilos profundamente funcionales, los códigos tienden a ser concisos, y las funciones son predecibles, lo que favorece intrínsecamente las pruebas y la programación concurrente.

    Aprender programación funcional no sólo mejora tu comprensión del tema, sino que también entrena a tu cerebro para pensar funcionalmente. Este proceso de pensamiento, a su vez, puede ayudar a aumentar la productividad y a mantener el código libre de errores. Así, comprender los detalles de la implementación fomenta una adopción más significativa de este poderoso paradigma en los proyectos del mundo real.

    Desvelar la implementación de los lenguajes de programación funcional

    Comprender la implementación práctica de los lenguajes de programación funcionales es un paso clave para dominar el arte de la codificación funcional. Colmar esta brecha entre teoría y práctica proporciona importantes conocimientos sobre cómo construir software que sea más lógico, comprobable y mantenible.

    Enfoques clave en la implementación de lenguajes de programación funcionales

    Dada la naturaleza diversa de los lenguajes de programación funcional, se puede acceder a diversas metodologías y herramientas para implementar código utilizando este paradigma. Sin embargo, a pesar de sus aparentes discrepancias, la esencia de estas metodologías sigue estando profundamente arraigada en los principios básicos de la programación funcional. A continuación, te presentamos algunos enfoques cruciales que debes tener en cuenta al profundizar en los aspectos prácticos:
    • Datos Inmutables: La inmutabilidad es una característica fundamental de los lenguajes de programación funcional. Significa que una vez inicializada una variable, su valor nunca puede cambiar. Este enfoque desaconseja el uso de bucles que suelen implicar la modificación de la información. En su lugar, la atención se centra en las llamadas a funciones, en particular las llamadas a funciones recursivas.
    • Funcionespuras: Las funciones puras son otro pilar de los lenguajes de programación funcional. Esto significa que la salida de las funciones se basa únicamente en su entrada, sin ninguna dependencia del estado externo. En consecuencia, restringe los efectos secundarios, lo que hace que la función sea más predecible y más fácil de probar o depurar.
    • Funcionesde orden superior: Una función de orden superior permite funciones como parámetros y devuelve una función como salida, llevando la idea de manipular funciones como valores a otro nivel. El resultado es un código más versátil y reutilizable.
    • Recursión: La recursión desempeña un papel crucial en los lenguajes de programación funcionales para repetir operaciones, a diferencia de las estructuras de bucle tradicionales. Fundamentalmente, la recursión consiste en que una función se llame a sí misma hasta alcanzar un caso base. Este método se presta bien a determinados algoritmos y estructuras de datos (por ejemplo, árboles).
    • Composición funcional: Es el proceso de combinar dos o más funciones para crear una nueva función. La salida de una función actúa como entrada de la siguiente, lo que ofrece flexibilidad para crear funciones complejas. La fórmula para representar esto es \((g \circ f)(x) = g(f(x))\), donde el resultado de la función \(f(x)\) se da como entrada a \(g\), y la salida final es el cálculo de \(g\).
    Es crucial comprender y aplicar estos principios fundamentales al implementar lenguajes de programación funcional, ya que construye una base sólida para crear código probabilístico, fácil de depurar y de mantener.

    Implementación de lenguajes de programación funcionales puros

    Los lenguajes funcionales puros llevan los conceptos de la programación funcional un paso más allá, garantizando que todas las funciones sean "puras". Es decir, cada función de estos lenguajes es autónoma e independiente del contexto externo, por lo que produce la misma salida para entradas idénticas y no tiene efectos secundarios.

    Implementar un lenguaje funcional puro implica adherirse a los principios de la programación funcional de forma aún más estricta. Esto implica
    • Preferir la recursividad a los bucles.
    • Utilizar ampliamente funciones de orden superior.
    • Componer funciones para una lógica más compleja.
    • Restringir los efectos secundarios y mantener la inmutabilidad del estado.
    Esta pureza e inmutabilidad de estado hacen que los lenguajes de programación funcionales puros sean eminentemente adecuados para manejar retos de programación complejos y desplazar el centro de atención de "cómo" se realizan los cálculos a "qué" cálculos se realizan. Así, el código es más intuitivo, sencillo y fácil de cuantificar.

    Ejemplos de implementación en varios lenguajes de programación funcional

    Estudiar los ejemplos concretos de los lenguajes funcionales puros puede ser útil para comprender sus implicaciones prácticas. Exploremos el manejo de las funciones de orden superior y el uso de la recursividad en Haskell y Erlang:

    Haskell es un lenguaje de programación funcional estrictamente puro, y la aplicación de funciones es su objetivo central. Por ejemplo, el cálculo recursivo del factorial en Haskell se expresa en un estilo sencillo y natural: ```haskell factorial n = if n == 0 then 1 else n * factorial (n-1) ``` Aquí, la función "factorial" se llama a sí misma repetidamente hasta que llega al caso base de \(n=0\), demostrando el principio de recursividad en Haskell.

    Otro ejemplo, Erlang, un lenguaje desarrollado para sistemas altamente distribuidos, tiene sus raíces en los principios de la programación funcional. El código Erlang utiliza a menudo funciones de orden superior y recursividad. Un ejemplo de recursión en Erlang utilizando la conocida secuencia de Fibonacci podría tener este aspecto: ```erlang -module(fibonacci). -export([fib/1]). fib(0) -> 0; fib(1) -> 1; fib(N) cuando N > 0, is_integer(N) -> fib(N-1) + fib(N-2). ``` Este programa muestra cómo resolver la secuencia de Fibonacci utilizando la recursión, con "0" y "1" como casos base.

    Aunque aparentemente diferentes, estos ejemplos demuestran cómo se mantienen los principios fundamentales de la programación funcional en diversos lenguajes de programación funcional pura. De hecho, estos principios básicos constituyen la base sobre la que estos lenguajes construyen y hacen evolucionar sus características más complejas y únicas. La ventaja de estos lenguajes es que los programas escritos en ellos son más fáciles de entender, probar y depurar, lo que hace que el mantenimiento del código sea una tarea más sencilla.

    Lenguajes de Programación Funcionales - Puntos clave

      • Los lenguajes de programación funcional son un componente básico de la informática y ofrecen un enfoque único para tratar los datos y ejecutar el código.
      • La programación funcional es un paradigma de programación que trata los cálculos como la evaluación de funciones matemáticas y evita los datos mutables y de estado cambiante.
      • Los lenguajes de programación funcional se centran en la expresión de la lógica más que en controlar el flujo del código, proponiendo así funciones de orden superior que se pueden asignar, almacenar, devolver y pasar como argumentos.
      • Algunos lenguajes de programación funcional populares son Haskell, Lisp, Scala y Erlang.
      • Los principios básicos de los lenguajes de programación funcional incluyen funciones puras, datos inmutables y funciones de primera clase.
      • Un lenguaje de programación funcional puro es aquel en el que cada función es una "función pura", lo que hace que la salida del programa sea predecible y más fácil de depurar.
      • La implementación de la programación funcional en varios lenguajes incluye JavaScript, Java y Python, y cada lenguaje ofrece su metodología única para estructurar y ejecutar código.
      • Los principios de los lenguajes de programación funcional se centran en las funciones de primera clase y de alto orden, las funciones puras, la inmutabilidad y la ausencia de estado, y la composición de funciones. Estos principios fomentan un código conciso que promueve las pruebas y la programación concurrente.
    Lenguajes de Programación Funcional Lenguajes de Programación Funcional
    Aprende con 16 tarjetas de Lenguajes de Programación Funcional en la aplicación StudySmarter gratis
    Regístrate con email

    ¿Ya tienes una cuenta? Iniciar sesión

    Preguntas frecuentes sobre Lenguajes de Programación Funcional
    ¿Qué son los lenguajes de programación funcional?
    Los lenguajes de programación funcional se centran en el uso de funciones matemáticas puras para el procesamiento de datos.
    ¿Cuáles son ejemplos de lenguajes de programación funcional?
    Ejemplos incluyen Haskell, Lisp, Erlang y Scala.
    ¿Cuál es la diferencia entre programación funcional y programación orientada a objetos?
    La programación funcional se enfoca en funciones puras y evita estados mutables, mientras que la orientada a objetos se basa en objetos y estados mutables.
    ¿Cuáles son las ventajas de usar lenguajes de programación funcional?
    Fomenta código más predecible, concurrente y menos propenso a errores.
    Guardar explicación

    Pon a prueba tus conocimientos con tarjetas de opción múltiple

    ¿Cuál es la característica definitoria de los paradigmas de programación funcional?

    ¿Cuáles son algunos de los principios básicos de los lenguajes de programación funcionales?

    ¿Nombra algunas características exclusivas de los lenguajes de programación funcionales?

    Siguiente

    Descubre materiales de aprendizaje con la aplicación gratuita StudySmarter

    Regístrate gratis
    1
    Acerca de StudySmarter

    StudySmarter es una compañía de tecnología educativa reconocida a nivel mundial, que ofrece una plataforma de aprendizaje integral diseñada para estudiantes de todas las edades y niveles educativos. Nuestra plataforma proporciona apoyo en el aprendizaje para una amplia gama de asignaturas, incluidas las STEM, Ciencias Sociales e Idiomas, y también ayuda a los estudiantes a dominar con éxito diversos exámenes y pruebas en todo el mundo, como GCSE, A Level, SAT, ACT, Abitur y más. Ofrecemos una extensa biblioteca de materiales de aprendizaje, incluidas tarjetas didácticas interactivas, soluciones completas de libros de texto y explicaciones detalladas. La tecnología avanzada y las herramientas que proporcionamos ayudan a los estudiantes a crear sus propios materiales de aprendizaje. El contenido de StudySmarter no solo es verificado por expertos, sino que también se actualiza regularmente para garantizar su precisión y relevancia.

    Aprende más
    Equipo editorial StudySmarter

    Equipo de profesores de Ciencias de la Computación

    • Tiempo de lectura de 22 minutos
    • Revisado por el equipo editorial de StudySmarter
    Guardar explicación Guardar explicación

    Guardar explicación

    Sign-up for free

    Regístrate para poder subrayar y tomar apuntes. Es 100% gratis.

    Únete a más de 22 millones de estudiantes que aprenden con nuestra app StudySmarter.

    La primera app de aprendizaje que realmente tiene todo lo que necesitas para superar tus exámenes en un solo lugar.

    • Tarjetas y cuestionarios
    • Asistente de Estudio con IA
    • Planificador de estudio
    • Exámenes simulados
    • Toma de notas inteligente
    Únete a más de 22 millones de estudiantes que aprenden con nuestra app StudySmarter.