La programación genérica es un paradigma de programación que consiste en tal descripción de datos y algoritmos que se pueden aplicar a varios tipos de datos sin cambiar la descripción misma. De una forma u otra, se apoya en diferentes lenguajes de programación . Las capacidades de programación genérica aparecieron por primera vez en forma de genéricos (funciones genéricas) en la década de 1970 en los lenguajes Clu y Ada , luego como polimorfismo paramétrico en ML y sus descendientes, y luego en muchos lenguajes orientados a objetos como C++ , Python [ 1] , Java , Object Pascal [2] , D , Eiffel , lenguajes para la plataforma .NET y otros.
La programación genérica se considera como una metodología de programación basada en la separación de estructuras de datos y algoritmos mediante el uso de descripciones de requisitos abstractos [3] . Las declaraciones de requisitos abstractos son una extensión del concepto de un tipo de datos abstracto . En lugar de describir un solo tipo en la programación genérica, se utiliza una descripción de una familia de tipos que tienen una interfaz y un comportamiento semántico comunes . Un conjunto de requisitos que describe una interfaz y un comportamiento semántico se denomina concepto . Así, un algoritmo escrito en un estilo generalizado se puede aplicar a cualquier tipo que lo satisfaga con sus conceptos. Esta posibilidad se llama polimorfismo .
Se dice que un tipo modela un concepto (es un modelo de un concepto) si satisface sus requisitos. Un concepto es un refinamiento de otro concepto si lo complementa. Los requisitos del concepto contienen la siguiente información: [4]
En C++ , OOP se implementa a través de funciones virtuales y herencia, mientras que OP (programación genérica) se implementa a través de plantillas de clases y funciones. Sin embargo, la esencia de ambas metodologías está solo indirectamente relacionada con tecnologías de implementación específicas. Más formalmente, OOP se basa en polimorfismo de subtipo , mientras que OP se basa en polimorfismo paramétrico . En otros idiomas, ambos pueden implementarse de manera diferente. Por ejemplo, los métodos múltiples en CLOS tienen una semántica similar al polimorfismo paramétrico.
Masser y Stepanov distinguen las siguientes etapas en la resolución del problema según la metodología OP:
La minimización y el enmarcado tienen como objetivo crear una estructura tal que los algoritmos sean independientes de tipos de datos específicos. Este enfoque se refleja en la estructura de la biblioteca STL . [5]
Richard Bird y Lambert Meertens propusieron un enfoque alternativo para definir la programación genérica, que puede llamarse programación genérica de tipo de datos . En él, las estructuras de tipos de datos son parámetros de programas genéricos. Para ello, se introduce un nuevo nivel de abstracción en el lenguaje de programación, a saber, la parametrización respecto a clases de álgebras con signatura variable . Aunque las teorías de ambos enfoques son independientes del lenguaje de programación, el enfoque de Musser-Stepanov, que enfatiza el análisis de conceptos, ha hecho de C++ su principal plataforma, mientras que Haskell y sus variantes utilizan casi exclusivamente la programación de tipos de datos genéricos [6] .
Las herramientas de programación genéricas se implementan en los lenguajes de programación en forma de ciertos medios sintácticos que permiten describir datos (tipos de datos) y algoritmos (procedimientos, funciones, métodos) parametrizados por tipos de datos. Para una función o tipo de datos, los parámetros de tipo formal se describen explícitamente . Esta descripción es generalizada y no puede usarse directamente en su forma original.
En aquellos lugares del programa donde se usa un tipo o función genéricos, el programador debe especificar explícitamente el parámetro de tipo real que especifica la declaración. Por ejemplo, un procedimiento genérico para intercambiar dos valores puede tener un parámetro de tipo que especifique el tipo de valores que intercambia. Cuando el programador necesita intercambiar dos valores enteros, llama al procedimiento con el parámetro de tipo " entero " y dos parámetros - enteros, cuando dos cadenas - con el parámetro de tipo " cadena " y dos parámetros - cadenas. En el caso de los datos, un programador puede, por ejemplo, describir un tipo genérico " lista " con un parámetro de tipo que especifica el tipo de valores almacenados en la lista. Luego, al describir listas reales, el programador debe especificar un tipo genérico y un parámetro de tipo, obteniendo así cualquier lista deseada utilizando la misma declaración.
Cuando un compilador encuentra una llamada a un tipo o función genéricos, realiza los procedimientos de verificación de tipos estáticos necesarios , evalúa la posibilidad de una instanciación determinada y, si es positivo, genera código, sustituyendo el parámetro de tipo real en lugar del parámetro de tipo formal. en la descripción genérica. Naturalmente, para el uso exitoso de descripciones genéricas, los tipos de parámetros reales deben cumplir ciertas condiciones. Si una función genérica compara valores de un parámetro de tipo, cualquier tipo concreto que se utilice en ella debe soportar operaciones de comparación, si asigna valores de un parámetro de tipo a variables, el tipo concreto debe garantizar la asignación correcta.
En C++, la programación genérica se basa en el concepto de una "plantilla", denotada por la palabra clave plantilla . Es ampliamente utilizado en la biblioteca estándar de C++ (consulte STL ), así como en bibliotecas de terceros boost , Loki . Alexander Stepanov hizo una gran contribución al surgimiento de herramientas de programación genéricas avanzadas en C++ .
Como ejemplo, demos una plantilla (generalización) de una función que devuelve el valor mayor de dos.
// Plantilla de descripción de plantilla de función plantilla < nombre de tipo T > T max ( T x , T y ) { si ( x < y ) devolver y ; más devuelve x ; } ... // Aplicando la función dada por la plantilla int a = máx ( 10 , 15 ); ... doble f = máx ( 123.11 , 123.12 ); ...o una plantilla (generalización) de una clase de lista enlazada:
plantilla < claseT > _ lista de clases { /* ... */ público : void Add ( const T & Element ); bool Buscar ( const T & Elemento ); /* ... */ };Haskell proporciona programación de tipos de datos genéricos. En el siguiente ejemplo a , una variable de tipo paramétrico.
lista de datos a = cero | Contras a ( Lista a ) longitud :: Lista a -> Int longitud Nil = 0 longitud ( Cons _ tl ) = 1 + longitud tlEjemplo de cálculo:
longitud ( Cons 1 ( Cons 2 Nil )) == 2Java ha proporcionado genéricos que se basan sintácticamente en C++ desde J2SE 5.0. Este lenguaje tiene genéricos o "contenedores de tipo T", un subconjunto de la programación genérica.
En la plataforma .NET aparecieron herramientas de programación genéricas en la versión 2.0.
// Declaración de una clase genérica. public class GenericList < T > { void Add ( T input ) { } } class TestGenericList { private class ExampleClass { } static void Main () { GenericList < int > list1 = new GenericList < int >(); GenericList < cadena > list2 = new GenericList < cadena >(); GenericList < EjemploClase > lista3 = new GenericList < EjemploClase >(); } }Un ejemplo de generación recursiva basada en plantillas D :
// http://digitalmars.com/d/2.0/template.html template Foo ( T , R ...) // T es un tipo, R es un conjunto de tipos { void Foo ( T t , R r ) { escribirln ( t ); static if ( r . length ) // si hay más argumentos Foo ( r ); // hacer el resto de los argumentos } } vacío principal () { Foo ( 1 , 'a' , 6.8 ); } /++++++++++++++++ imprime: 1 a 6.8 ++++++++++++++++/El soporte para la programación genérica del compilador Free Pascal ha estado disponible desde la versión 2.2 en 2007 [7] . En Delphi - desde octubre de 2008 . El soporte básico para clases genéricas apareció por primera vez en Delphi 2007 .NET en 2006 , pero solo afectó a .NET Framework . En Delphi 2009 se ha agregado un soporte más completo para la programación genérica . Las clases genéricas también son compatibles con Object Pascal en el sistema PascalABC.NET .