Fortran | |
---|---|
Semántica | imperativo , paralelo |
clase de idioma | procedimental , modular , con elementos de programación orientada a objetos |
tipo de ejecución | compilado |
Apareció en | 1957 |
Autor | John Backus , IBM |
Desarrollador | IBM [1] y John Backus [1] |
extensión de archivo | .for .f .f90 .f95 |
Liberar | Fortran 2018 (ISO/IEC 1539-1:2018) (28 de noviembre de 2018) |
sistema de tipos | estricto , estático |
Implementaciones principales | GFortran , Open Watcom , Sun Studio , XL Fortran , Intel Fortran |
Dialectos | Fortran de alto rendimiento |
sido influenciado | Codificación rápida [d] |
influenciado | ALGOL 68 , BÁSICO , PL/I |
Sitio web | fortran-lang.org _ |
Archivos multimedia en Wikimedia Commons |
Fortran ( ing. Fortran ) es el primer lenguaje de programación de alto nivel que ha recibido una aplicación práctica, tiene un traductor y ha experimentado un mayor desarrollo [2] . Creado entre 1954 y 1957 por un grupo de programadores liderados por John Backus en IBM Corporation [3] . El nombre Fortran es la abreviatura de FOR mula TRAN slator (traductor de fórmulas) [4] . Fortran es ampliamente utilizado principalmente para computación científica y de ingeniería. Una de las ventajas del Fortran moderno es una gran cantidad de programas y bibliotecas de subrutinas escritas en él [5] .
Hay una gran cantidad de varias bibliotecas matemáticas escritas en Fortran (principalmente en versiones anteriores del lenguaje) para álgebra matricial y resolución de sistemas de ecuaciones lineales , bibliotecas para resolver ecuaciones diferenciales , ecuaciones integrales y sus sistemas, aproximación de funciones , funciones especiales , transformadas rápidas de Fourier , estadística matemática y otras disciplinas matemáticas. Estas bibliotecas generalmente se suministran con el compilador. Varios de estos paquetes se han creado a lo largo de las décadas y todavía son populares en la comunidad científica hasta el día de hoy, por ejemplo, la Biblioteca internacional de subrutinas matemáticas (IMSL) [6] [7] .
La mayoría de estas bibliotecas son de hecho propiedad de la humanidad: están disponibles en código fuente, bien documentadas, depuradas y muy efectivas.
Modern Fortran (Fortran 95 y Fortran 2003) ha adquirido las características necesarias para una programación eficiente para las nuevas arquitecturas informáticas; le permite aplicar tecnologías de programación modernas, en particular, programación genérica y modular, programación orientada a objetos , manteniendo la continuidad con versiones anteriores. Uno de los conceptos principales para el desarrollo de Fortran moderno es el soporte de paralelismo y las operaciones vectoriales [8] .
Fortran es un lenguaje altamente estandarizado, por lo que se puede portar fácilmente a diferentes plataformas. Los nuevos estándares del lenguaje conservan en gran medida la continuidad con los más antiguos, lo que hace posible utilizar los códigos de programas previamente escritos y modificarlos [8] . Al mismo tiempo, a medida que se desarrolla el lenguaje, se anuncian con anticipación construcciones obsoletas, que pueden ser eliminadas en el futuro [9] .
Fortran tiene un gran conjunto de funciones matemáticas integradas, admite el trabajo con números enteros, reales y complejos de precisión doble y alta (se usa para operaciones binarias real(10)), tiene un rico conjunto de herramientas para trabajar con matrices y archivos externos. Modern Fortran (2003 y posteriores) tiene un juego de herramientas completo para trabajar con datos de caracteres.
Inicialmente, los medios expresivos del lenguaje no eran muy grandes, ya que Fortran fue el primer lenguaje de alto nivel en implementarse. Posteriormente, se agregaron muchas construcciones léxicas a Fortran, que son características de la programación estructural, modular, generalizada y orientada a objetos.
La estructura de los programas se centró originalmente en la entrada de tarjetas perforadas y tenía una serie de propiedades que eran convenientes para este caso particular. Así, de la 1ª a la 5ª columna había un área de etiqueta, la 6ª servía para marcar el texto como continuación de la línea anterior (cualquier carácter excepto un espacio y un "0"), y de la 7ª a la 72ª la el texto real fue localizado operador o comentario. Las columnas 73 a 80 podían servir para la numeración de cartas (para restaurar una baraja desparramada accidentalmente) o para un breve comentario, fueron ignoradas por el traductor. Si el texto del operador no encajaba en el espacio asignado (de la columna 7 a la 72), se colocaba un signo de continuación en la columna 6 de la siguiente línea y luego el operador continuaba. Era imposible colocar dos o más operadores en una línea (mapa). Cuando las tarjetas perforadas pasaron a la historia, estas ventajas se convirtieron en serios inconvenientes.
Es por eso que al estándar Fortran, comenzando con Fortran 90, manteniendo el formato fijo del texto fuente, se agregó el formato libre , que no regula la posición de la línea y permite escribir más de un enunciado por línea. La introducción del formato libre y los métodos modernos de programación estructurada [8] ha hecho posible crear código cuya legibilidad y claridad no es inferior al código creado utilizando otros lenguajes de programación modernos como Pascal , C o Java . Los entornos de desarrollo modernos le permiten combinar formatos: por ejemplo, expandir la longitud de una cadena a un formato libre (generalmente 132 caracteres), le permiten escribir varias declaraciones por línea, pero al mismo tiempo también le permiten mantener la sangría izquierda (hacer un margen izquierdo), característico del antiguo formato fijo, dejando la columna más dedicada a marcas de error y formatos, así como una columna de continuación de línea.
Una especie de "tarjeta de presentación" del antiguo Fortran es una gran cantidad de etiquetas que se usaban en operadores de salto incondicional GOTO, operadores de bucle, operadores condicionales y operadores de descripción de entrada-salida de formato FORMAT. La gran cantidad de etiquetas y declaraciones GOTOa menudo dificultaban la comprensión de los programas Fortran.
Es esta experiencia negativa la que se ha convertido en la razón por la que en varios lenguajes de programación modernos (por ejemplo, en el lenguaje Java ), las etiquetas y sus operadores de salto incondicional asociados se han modificado en gran medida.
Sin embargo, el Fortran moderno (principalmente a partir de la versión Fortran'90) está libre de un exceso de etiquetas debido a la introducción de operadores como DO... END DO, DO WHILE, SELECT CASE, construcciones IF THEN- ELSEIF THEN- ELSE- END IF, etc. Además, en los estándares del lenguaje moderno, solo el clásico operador GOTOutilizado en muchos idiomas hasta el día de hoy. El operador calculado GOTO, así como la ENTRY construcción de múltiples entradas a procedimientos, se eliminaron del estándar, aunque generalmente aún son compatibles con los compiladores.
Hola Mundo!Formato fijo (los espacios en las posiciones de línea 1 a 6 están marcados con caracteres "␣"):
␣␣␣␣␣␣ PRINT * , '¡Hola, mundo!' ␣␣␣␣␣␣ FINFormato libre:
print * , "¡Hola, mundo!" final Comentarios.Fortran admite 5 tipos de datos integrados elementales básicos: real ( REAL) , complejo ( COMPLEX) , entero ( INTEGER) con o sin signo, booleano ( LOGICAL) y carácter ( CHARACTER) . También es posible crear tipos de datos derivados usando TYPE. Desde el comienzo de la creación del lenguaje, hubo 4 tipos de datos: real, complejo, entero y booleano.
Existe una noción de un tipo de tipo de datos o parámetros de tipo. Esto permite parametrizar datos reales y complejos (es decir, especificar la precisión y el rango del orden decimal) y otros datos, aumentando la portabilidad de las aplicaciones.
Para todos los datos numéricos, se definen las operaciones y asignaciones aritméticas habituales, y hay funciones integradas. Una función incorporada solo puede tomar un número real o complejo ( no un número entero ) como argumento .Log(x)x
Como regla general, para números reales de precisión "simple", se asignan 4 bytes ( REAL(4)o parámetro de variación de tipo KIND=4), "doble" - 8 bytes. Para números complejos, el número de bytes se duplica.
Los enteros pueden tomar de 1 a 4 bytes. Los compiladores modernos permiten al programador operar con números y precisión "cuádruple".
En las operaciones aritméticas, el tipo se cambia por defecto de un número entero a un número real y luego a un número complejo, o mediante el uso de una función numérica especial incorporada. Entonces, las siguientes expresiones son equivalentes ( i es un número entero): y . Log(i*1.)Log(real(i))
Los datos de cadena (carácter) se especifican con una longitud entre paréntesis, ya sea después del atributo de tipo o después del nombre de la cadena. Para especificar una cadena en el cuerpo del programa, se utilizan comillas simples o dobles. Entonces, las entradas son equivalentes: A='It is hot' o A="It is hot" . Esto es útil en los casos en que hay comillas en la propia cadena: B="It is not hot" .
Para las cadenas, hay una operación de concatenación (suma) de cadenas integrada: //. Hay 17 funciones integradas especializadas para datos de caracteres (además de las genéricas que manejan todo tipo de datos).
Las etiquetas son números enteros con no más de 5 dígitos; No se permiten caracteres. Las etiquetas se utilizan en instrucciones GO TO, instrucciones de lectura, escritura y formato, así como en el manejo de errores y excepciones. Además de las etiquetas, Fortran tiene la capacidad de nombrar estructuras de control (bucles, condiciones lógicas, estructuras FORALL... END FORALL, WHERE... END WHERE, SELECT CASE... END SELECT, TYPE... END TYPE, etc.), y el nombre de la estructura puede contener cualquier carácter. permitido en nombres de variables.
Funciones integradas para datos de caracteresPara la conversión de números a caracteres : CHAR(i)y ACHAR(i). Convierte un número entero a su carácter de sistema operativo correspondiente o carácter de tabla ASCII .
Para la conversión de caracteres a números : ICHAR(i)y IACHAR(i). Realiza transformaciones inversas.
Funciones de comparación de cadenas : LGE(A,B), LGT(A,B)y . El resultado de las funciones es "verdadero" si las longitudes de las cadenas ( L(A) y L(B) ), expresadas en caracteres ASCII, satisfacen las siguientes desigualdades, respectivamente: L(A) L(B), L(A) L(B), L(A) L(B) y L(A) L(B) . LLE(A,B)LLT(A,B)
Funciones de longitud : LEN(A)y LEN_TRIM(A). El primero devuelve la longitud de la cadena A (número de caracteres), el segundo devuelve la longitud de la cadena sin espacios finales, si los hay.
Funciones de conversión : TRIM(A), ADJUSTL(A)y ADJUSTR(A), REPEAT(A,N). La función TRIM(A)devuelve la cadena A sin espacios finales. Las funciones ADJUSTL(A)y ADJUSTR(A)alinean una cadena (eliminan espacios) a la izquierda y a la derecha, respectivamente. La función REPEAT(A,N)devuelve N copias de la cadena A.
Funciones de búsqueda de línea: , , . El parámetro opcional atrás especifica la dirección de búsqueda: por defecto, izquierda (para ) derecha (para ). SCAN(A,B,[back])INDEX(A,B,[back])VERIFY(A,B,[back])back=.false.back=.true.
La función SCANdetermina el número de posición en la cadena A (izquierda o derecha) del primer carácter encontrado de la lista de cadenas B. Si el resultado es negativo, la función devolverá un número entero 0. La función INDEXdetermina el número de la posición desde la que comienza por primera vez la aparición completa de la cadena B en la cadena A. Además, la búsqueda se puede realizar tanto a la izquierda como a la derecha, pero el número de posición siempre se calcula a la izquierda, desde el principio de la línea. Si la búsqueda falla, la función devolverá 0. La función es VERIFY inversa a la función INDEX. Por lo tanto, VERIFYdevuelve el número de posición de dicho carácter en la cadena A , que no está en la cadena de máscara B. Si todos los caracteres (diferentes) de la cadena A están presentes en la cadena de máscara B, la función devolverá 0.
Todas estas funciones son elementales y su argumento puede ser una matriz de caracteres o números enteros. El resultado será una matriz numérica, de caracteres o lógica coincidente.
Los estándares de Fortran, a partir de la versión 2003, brindan la capacidad de trabajar con caracteres Unicode .
Además de estas características, Fortran le permite procesar datos simbólicos utilizando su análisis de matriz (vector) incorporado , lo que aumenta significativamente la flexibilidad del procesamiento de datos simbólicos.
En Fortran, por compatibilidad con programas escritos en lenguaje C, existe el concepto de C-string , que se especifica agregando un carácter después de la comilla: A='This is a C-string'c . La cadena nula se dará así: A='\0'c .
Las subrutinas en Fortran han existido desde el primer estándar y siguen siendo una de las principales herramientas de programación [8] .
En Fortran, llamar a subrutinas, funciones y pasar sus parámetros ocurre exclusivamente por referencia (y no por valor ). Por lo tanto, la subrutina puede cambiar el argumento que se le pasa en el programa principal, si esto no se previene específicamente. Este mecanismo hace posible que la notación sea natural al escribir fórmulas matemáticas y, al mismo tiempo, mantener un alto rendimiento cuando se trabaja con grandes matrices de datos [24] .
Las subrutinas de Fortran pueden contener en la lista de parámetros (llamados parámetros formales) y parámetros opcionales (opcionales), o pueden no contener ningún parámetro.
El lenguaje estándar permite que los procedimientos y las operaciones se sobrecarguen a través de una interfaz genérica, combinando diferentes procedimientos (cada uno operando, por ejemplo, con números enteros, reales, números complejos y variables de caracteres) bajo un solo nombre (genérico). En este caso, basta con referirse al procedimiento genérico en el programa principal, y la naturaleza de las operaciones realizadas dependerá del tipo de datos ofrecidos al procedimiento para su procesamiento. Todas las funciones y subrutinas integradas se realizan de acuerdo con este principio, por ejemplo, COS(x). La sobrecarga de procedimientos, funciones y operadores (además, el programador puede ofrecer sus propios símbolos de operadores sobrecargados, además de los incorporados) se aplica no solo a los tipos de datos incorporados, sino también a los tipos definidos por el programador [12] .
Tipos de subrutinasLos procedimientos se dividen en subrutinas y funciones . Las subrutinas son más convenientes cuando necesita devolver una gran cantidad de resultados heterogéneos; funciones: al devolver un resultado de un tipo (incluida una matriz).
Un subprograma se define mediante el operador de descripción Subroutine subprogram_name (lista de argumentos formales) , una función se define mediante el operador Function function_name (lista de argumentos formales) .
Una subrutina es llamada por la declaración Call subroutine_name (lista de argumentos reales) . Una función se llama por su nombre, con una lista de argumentos reales y sin usar ningún operador especial.
A partir del estándar F'90, se admiten procedimientos recursivos (no disponibles en versiones anteriores debido a la memoria limitada de la máquina) que requieren un especificador explícito para declarar recursive. En este caso, el resultado de la función debe diferir del nombre de la función misma.
Los procedimientos y funciones puros ( pure subroutine [function]) son procedimientos y funciones introducidos por el estándar F'95 que no tienen efectos secundarios. Una función pura debe devolver un valor y no debe cambiar ninguno de sus parámetros de entrada y/o datos globales; un procedimiento puro debe cambiar solo aquellos parámetros que se especifican explícitamente como resultado (salida) por medio del atributo intent(outo inout)). La posibilidad de efectos secundarios en Fortran (es decir, eventualmente la posibilidad de cambiar variables en el programa principal a través de una subrutina) es un efecto secundario del método rápido de pasar a una dirección.
Las unidades de programa puras no pueden contener sentencias de E/S ( WRITEy READ) para archivos y dispositivos externos, incluidos el teclado y la pantalla, así como sentencias de pausa y detención del programa.
Todas las funciones y subrutinas integradas de Fortran, incluidas las matemáticas (excepto las que acceden al sistema operativo, las funciones de fecha y hora y los generadores de números aleatorios) son puras, es decir, no crean efectos secundarios. Se introdujeron funciones puras para mejorar la cultura de la programación y aumentar la eficiencia de la paralelización de algoritmos [25] [9] .
Argumentos del subprogramaLos argumentos de subprograma pueden ser cualquier tipo de datos incorporado, incluidas matrices y sus secciones, punteros y tipos definidos por el programador. Los argumentos de subrutina también pueden ser funciones y otras subrutinas, excepto las subrutinas internas, funciones de operador, procedimientos genéricos (genéricos) (solo se permiten nombres específicos) y algunos otros tipos incorporados.
Los argumentos se dividen en formales y fácticos . Los argumentos están entre paréntesis después del nombre de la subrutina y separados por comas. Los nombres de los argumentos reales y formales pueden ser los mismos.
Los argumentos formales son los argumentos de una subrutina (función) especificada en su descripción. Si la subrutina no tiene argumentos, se pueden omitir los paréntesis. La función tiene paréntesis incluso si no hay argumentos formales. Un procedimiento de parámetro formal se denomina procedimiento formal .
Los argumentos reales son argumentos que se pasan a una subrutina o función para que se ejecuten cuando se la llame. Una función sin argumentos se llama con una lista vacía entre paréntesis, una subrutina sin paréntesis.
Los argumentos formales y reales deben ser consistentes . Los tipos de argumentos y las variedades de su apariencia deben ser los mismos, la matriz debe corresponder a una matriz (o sección de una matriz) de la misma configuración.
Hay matrices que aceptan la configuración y el tamaño como argumentos formales para los procedimientos. Una matriz de toma de configuración es un argumento de matriz formal que hereda la configuración de su matriz real correspondiente. Para una matriz de este tipo, cuando se declara, se establece la dimensión (coincidiendo con la dimensión del argumento de la matriz real) y se omiten los límites superiores. De forma predeterminada, los límites inferiores son 1, pero se pueden establecer de forma arbitraria. El número y el valor de los elementos de la matriz que se hace cargo de la configuración se heredan exactamente del argumento de la matriz real. Una matriz que adquiere un tamaño es una forma anterior, nativa del lenguaje Fortran'77, de describir las matrices heredadas, retenida por compatibilidad. Para dichas matrices, solo se hereda la última dimensión, cuyo límite superior se describe con un asterisco ( *). En este caso, los arreglos-argumentos formales y reales pueden tener diferentes dimensiones. Los arreglos que toman una configuración y tamaño no pueden ser dinámicos o punteros. Los procedimientos que heredan arreglos deben tener una interfaz explícita.
Las cadenas de argumentos formales también pueden heredar (asumir) la longitud del argumento de cadena real correspondiente. Las cadenas que adquieren una longitud se describen con el símbolo *: Character (Len = *) string_name . Cuando se especifica explícitamente la longitud de una cadena, la longitud del argumento formal de cadena no puede ser mayor que el argumento de cadena real correspondiente.
Los argumentos son posicionales y clave . Los argumentos posicionales formales y reales se asocian entre sí en el orden en que aparecen en la lista de argumentos, que deben coincidir. Palabras clave : por el nombre de la clave, que coincide con el nombre del argumento formal. Las palabras clave le permiten romper el orden de los argumentos u omitir algunos de ellos. Entonces, para una subrutina con el encabezado SubroutineUNO (A, B, C, D), la llamada podría ser: CallUNO (D= Z , C= Y , B= X , A= W ), donde W, X, Y, Z son los argumentos reales.
Los argumentos clave le permiten tener argumentos opcionales que se pueden omitir. En este caso, los argumentos opcionales deben tener la extensión Optional. Por ejemplo, si se especifica Optional C, DCall , entonces se puede llamar UNO (B= X , A= W ) en este caso .
Los procedimientos con parámetros opcionales deben tener una interfaz explícita.
Las matrices son fundamentales para la filosofía de Fortran. Todas las construcciones de lenguaje, datos, unidades de programa, operadores, funciones integradas, bucles se han creado y se crean para un procesamiento eficiente, en primer lugar, de matrices. Fortran, a medida que se desarrolla, sigue el principio de evitar la descripción detallada (elemento por elemento) y el procesamiento de matrices tanto como sea posible. Esto es especialmente efectivo cuando se procesan arreglos multidimensionales (la dimensión máxima de los arreglos en el estándar F2008 es 15). Esta vista de matrices no era común en las primeras versiones del lenguaje; los primeros elementos de un enfoque generalizado de las matrices aparecieron en FORTRAN77; todavía se están desarrollando.
Los arreglos son estáticos o dinámicos . Los dinámicos se dividen en colocados y automáticos (se forman cuando se llama a una subrutina). Los elementos de un arreglo bidimensional en Fortran se organizan por columnas , no por filas, como, por ejemplo, en C. Por lo tanto, el primer índice de la matriz cambia más rápido. Por lo tanto, para trabajar efectivamente con matrices en bucles anidados, debe indexar los bucles internos con índices a la izquierda y los bucles externos con índices a la derecha. De forma predeterminada, las matrices se asignan, entrada, salida e inicialización por columnas.
hacer k = 1 , 10 hacer j = 1 , 20 hacer yo = 1 , 100 arr ( yo , j , k ) = 25 ! derecha brr ( k , j , yo ) = 0 ! viable, pero varias veces más lento fin hacer ; fin hacer ; terminar hacerLas matrices pueden tener un tamaño cero (incluso si el límite inferior supera al superior). Los índices de límite de matriz pueden ser cualquier número entero. El límite inferior predeterminado es 1.
Real , asignable :: ArR (:,:,:) ! declaración de una matriz real dinámica asignada Integer , asignable :: ArI (:), ArSI ( 2 , 5 ) ! matrices dinámicas y estáticas de enteros Carácter ( 32 ), asignable :: ArC (:), ArC2 ( 20 ) ! una matriz dinámica de cadenas con una longitud de 32 caracteres y una matriz estática de cadenas Asignar ( ArR ( - 74 : 0 , 8 , 1 : 3 ), ArI ( 0 ), ArC ( 1 : - 1 )) ! colocación de arreglos dinámicos print * , size ( ArR ), size ( ArI ), size ( ArC ), size ( ArSI ) ! 1800 0 0 10 ArC2 ( 17 )( 5 : 27 ) = 'Esta es una asignación de cadena' ! La línea número 17 se escribirá ␣ ␣ ␣ ␣ Esta es la␣ asignación␣ de la cadena␣ ␣ ␣ ␣ ␣ ... Secciones de arreglos, operadores FORALLy WHEREFortran permite la asignación eficiente de arreglos sin bucles al enmascarar la asignación usando los operadores WHEREy FORALL, así como el corte de arreglos y los índices vectoriales . En todos los casos, se evalúa inicialmente el lado derecho de la expresión completa (para todos los índices de matriz) y solo entonces se realiza la asignación para los índices que satisfacen la matriz de máscara. Los cálculos que utilizan estas herramientas le permiten aumentar el rendimiento y facilitar que el compilador seleccione secciones del programa que se pueden ejecutar de forma independiente, es decir, en paralelo.
Real :: arr ( I1 : I2 , J1 : J2 , K1 : K2 ), arr1 ( I1 : I2 , J1 : J2 , K1 : K2 ), arr2 ( I1 : I2 , J1 : J2 , K1 : K2 ) Real :: frr ( 100 ), frr1 ( 10 ) / 1 , 2 , 3 , 3 * 4 , 4 * 5 / ! o Real :: frr1 ( 10 ) = ( / 1 , 2 , 3 , 4 , 4 , 4 , 5 , 5 , 5 , 5 / ) ... ar = 1. ! asignación de matriz (sobrecarga incorporada para el operador de asignación) arr1 = Sin ( arr ) + arr ! la función de pecado elemental se aplica a cada elemento de la matriz arr2 ( I1 : I2 : 1 , J1 : J2 : 2 , K2 : K1 : - 4 ) = arr1 ( I1 : I2 : 1 , J1 : J2 : 2 , K2 : K1 : - 4 ) ! asignación de elementos en incrementos de 1, 2 y -4 (hacia atrás) dada por un triplete de índice fr = ( / ( J , J = 1 , 100 ) / ) ! asignar una matriz unidimensional a través de una lista circular Para todos ( i = I1 : I2 , j = J1 : J2 , k = K1 : K2 , arr ( i , j , k ) > 0. ) brr ( i , j , k ) = Log ( arr ( i , j , k )) ! sustitución de ciclos y sentencias y construcciones condicionales. Enmascaramiento de asignaciones (máscara — arr(i,j,k)>0.) Para todos ( i = 1 : N , j = 1 : N , k = 1 : N ) crr ( i , j , k ) = Sin ( 0.5 * ( i + j ) - k ) ! extensión de secciones transversales Para todos ( i = 1 : 100 ) ! Construcción forall para sentencias de asignación múltiple drr ( i , i ) = 0. ! acceder a la diagonal de la matriz err ( i , i , i ) = 1. ! y diagonales de un arreglo tridimensional End ForallOperaciones menos obvias son posibles:
Entero V ( - 2 : 2 , 1 : 5 ) V = remodelar ( fuente = ( / ( i , i = 1 , 25 ) / ), forma = ( / 5 , 5 / )) ! inicialización de una matriz con números de serie usando el constructor de matrices y la función de remodelación print * , V ! ¡ La salida a la ventana de DOS se hará línea por línea ! 1 2 3 4 5 - 1ra columna ! 6 7 8 9 10 - ¡2do ! 11 12 13 14 15 - ¡3ro ! 16 17 18 19 20 - ¡4to ! 21 22 23 24 25 - 5to V ( 2 , 3 : 4 ) = V ( - 1 : 0 , 1 ) ! Gire una parte de la matriz print * , V ! ¡ La salida a la ventana de DOS se hará línea por línea ! 1 2 3 4 5 ! 6 7 8 9 10 ! 11 12 13 14 2 cambio en la 3ra columna 15 a 2 ! 16 17 18 19 3 cambio en la 4ª columna 20 a 3 ! 21 22 23 24 25Las capacidades del operador y la construcción FORALLintroducidas por el estándar F'95 son más amplias que las del operador y la construcción WHERE, sin embargo, este último, en algunos casos de bifurcación lógica, le permite simplificar el código debido a la presencia de una alternativa ELSEWHERE. evitando operadores condicionales anidados y matrices de máscaras complejas.
El operador y la construcción FORALLsólo permiten el uso de funciones y procedimientos puros . Al enmascarar la asignación en los operadores WHERE, FORALL, así como en funciones integradas especiales para matrices (por ejemplo, SUM), la máscara de matriz lógica se calcula antes de la asignación y le permite reemplazar bucles con condiciones lógicas dentro de ellos, lo que evita trabajo adicional para el predictor de rama del microprocesador .
Un índice vectorial es un arreglo unidimensional de enteros cuyos valores son los índices de algún otro arreglo. Los índices vectoriales se utilizan para crear secciones arbitrarias de matrices multidimensionales y son sus generalizaciones. Al usar índices vectoriales, debe tener cuidado con los valores de índice repetidos en el lado izquierdo del operador de asignación, ya que en este caso se intentará escribir valores posiblemente diferentes en una celda de memoria. El orden de los índices es arbitrario (aunque esto no debe usarse en exceso para evitar la degradación del rendimiento).
Entero vi ( 5 ) / 7 , 7 , 7 , 3 , 8 / , vj ( 4 ) / 1 , 2 , 3 , 10 / ! inicialización de matrices - índices vectoriales Real arr ( 20 , 20 ), brr ( 10 , 10 ) brr = 0. ; ar = 1. ! los índices vectoriales también se pueden especificar dentro de la matriz que los usa brr (( / 8 , 6 , 2 , 1 , 4 / ), vj ) = arr ( vi , vj ) ! ¡ Las dimensiones de los índices vectoriales deben coincidir a la izquierda y a la derecha, y sus valores no deben ir más allá de los límites de las matrices que los utilizan ! el tamaño de los índices vectoriales puede ser menor que el tamaño de las matrices de trabajo Funciones integradas para arreglosModern Fortran tiene una gran cantidad de funciones integradas especializadas para trabajar con matrices numéricas y de caracteres (además de los métodos generales discutidos anteriormente). Los argumentos de las funciones son el array numérico y/o de caracteres array , la máscara de array lógico (que es, por ejemplo, el array de condición>0 ) y la dimensión dim del array array , que forma (si se da el argumento dim ) una sección de la matriz a lo largo de una de las dimensiones con el número dim . La matriz matriz puede, a menos que se especifique lo contrario, ser un número entero, contener números reales o complejos. Si no se especifica la matriz de máscaras, su valor se considera igualmente verdadero. La máscara de matriz booleana , si se proporciona, debe tener la misma forma que matriz o un valor escalar .TRUE..
La mayor parte de las características introducidas por el estándar F'90.
ALL(mask[, dim]) es una función lógica; devuelve "verdadero" si todos los elementos de la máscara de matriz lógica son verdaderos (junto con la dimensión opcional dim ) y viceversa en caso contrario.
ANY(mask[, dim]) es una función lógica; es verdadero si al menos un elemento de la máscara de matriz lógica es verdadero (junto con la dimensión opcional dim ).
COUNT(mask[, dim]) es una función entera; el resultado es igual al número de elementos verdaderos de la matriz de máscaras (a lo largo de la dimensión opcional dim ).
MAXLOC(array[, mask][, dim]), MINLOC(array[, mask][, dim]),
son funciones enteras que devuelven respectivamente el índice del elemento máximo y mínimo (o índices de los elementos máximo y mínimo) a lo largo de la dimensión opcional dim para elementos que satisfacen la matriz de máscara. Las funciones devuelven el índice del primer elemento de la matriz matriz . Si no se proporciona el argumento de la función dim , o si la matriz es una matriz unidimensional, el resultado se escribe en una matriz unidimensional.
Para matrices multidimensionales matriz , el resultado se escribe en una matriz con un rango uno menor que el rango de la matriz matriz (se excluye dim ).
Los valores de índice se cuentan en orden desde los límites inferiores de la matriz. Es decir, si el número del límite inferior del arreglo arreglo difiere de uno, entonces para acceder al elemento máximo o mínimo del arreglo, se debe sumar al resultado de las funciones la MAXLOCdiferencia MINLOCentre el índice del límite inferior y uno .
MAXVAL(array[, mask][, dim]), MINVAL(array[, mask][, dim]) — funciones de búsqueda para el elemento máximo y mínimo, respectivamente, en la matriz matriz para elementos que satisfagan la máscara lógica matriz máscara a lo largo de la dimensión opcional dim . El resultado de la función es del mismo tipo y variedad que array . La matriz matriz solo puede ser real o entera. Para una matriz
unidimensional , o si no hay un argumento dim , el resultado es un escalar; de lo contrario, una matriz con un rango menor que el rango de la matriz .
FINDLOC(array, value[, dim][, mask]) es una función de número entero que devuelve el índice del elemento de matriz igual a valor . Introducido por el estándar F2008. Los elementos de matriz buscados satisfacen la máscara de matriz de máscara lógica a lo largo de la dimensión opcional dim . El tipo del argumento de valor debe coincidir con el tipo de matriz y puede ser de cualquier tipo integrado (incluidos complejos, booleanos o de caracteres). Las propiedades restantes de la función son similares a las propiedades de las funciones y . MAXLOCMINLOC
A partir del estándar F2003, las funciones MAXLOCy MINLOC, al igual que la función FINDLOC, también operan con datos de caracteres.
SUM(array[, mask][, dim])y PRODUCT(array[, mask][, dim])realizar, respectivamente, la suma y la multiplicación de los elementos del arreglo. El significado de los argumentos de la función es SUMel PRODUCTmismo que el de las funciones anteriores.
La función PRODUCTopera en datos complejos a partir del estándar F2003.
DOT_PRODUCT(vector_1, vector_2)realiza un producto interno según las reglas del álgebra lineal de los vectores vector_1 y vector_2 (matrices unidimensionales) del mismo tamaño. Los arreglos unidimensionales vector_1 y vector_2 pueden contener datos de cualquier tipo numérico y booleano. Los vectores vector_1 y vector_2 pueden ser tanto numéricos como booleanos.
MATMUL(matrix_a, matrix_b) - Función de multiplicación de matrices incorporada. Multiplica dos matrices, una matriz por un vector, un vector por una matriz de acuerdo con las reglas del álgebra lineal. Los argumentos de las funciones matrix_a y matrix_b son matrices lógicas o numéricas bidimensionales o unidimensionales (de cualquier tipo numérico incorporado). Los argumentos de una función no pueden ser dos vectores al mismo tiempo: uno de los argumentos debe ser una matriz (arreglo bidimensional). El número de elementos en la primera (o única) dimensión del arreglo matrix_b debe ser igual al número de elementos en la última dimensión del arreglo matrix_a . Introducido por el estándar F'90.
En algunos casos, al calcular el producto de un vector columna por un vector fila, lo que requiere MATMULuna transformación adicional de vectores en matrices de la forma (/m,1/)y al utilizar la función (/1,n/), la eficiencia, MATMULsegún Barteniev [12] , es notablemente inferior a un bucle anidado convencional.
Según las pruebas de la NASA [26] para el producto de matrices (arreglos bidimensionales), el rendimiento MATMULdel compilador Intel cuando se usa la optimización completa -O3 significativamente (en algunos casos, en un orden de magnitud) supera el rendimiento de los bucles anidados. , aunque para matrices con un tamaño de ~1000 × 1000 y mayores, es algo inferior a las subrutinas de rendimiento DGEMM de la biblioteca LAPAK . Al mismo tiempo, para matrices ~100×100 y menos , MATMULsupera a DGEMM en términos de velocidad. El compilador IBM Fortran, a partir de la versión F'90, utiliza para ello MATMUL el algoritmo Winograd-Strassen con complejidad [27] . Tenga en cuenta que las implementaciones algorítmicas de las funciones matemáticas generalmente no están especificadas por el estándar y quedan a discreción del desarrollador del compilador.
MERGE(t_source, f_source, mask) — una función que crea una nueva matriz bajo el control de la máscara mask-array a partir de los elementos de las matrices t_source y f_source de la misma forma y tamaño que las matrices originales. Las matrices de argumentos y la matriz de resultados pueden ser de cualquier tipo integrado y coincidir en el tipo, el tamaño y la forma de los datos.
Si el elemento de máscara es verdadero ( .ТRUE.), entonces el elemento correspondiente de la matriz de resultados es igual al elemento correspondiente de la matriz t_source ; si es falso ( .FALSE.), entonces al elemento de la matriz f_source .
Los argumentos de función pueden ser escalares; en este caso, por ejemplo, MERGE(a,0,c>=0)=a· Θ(с) , donde Θ(с) es una función de Heaviside entera .
MOVE_ALLOC(from, to) es una subrutina integrada que le permite reasignar dinámicamente una matriz dinámica previamente asignada con nuevos límites y tamaño, como la matriz dinámica de . Los datos de la matriz from se copian en la matriz to . El tipo de datos y el rango de las matrices desde y hasta deben coincidir. Después de reasignar la matriz to , la matriz from se libera y no se asigna. Puede ser útil en métodos numéricos con discretización variable del problema ( métodos multired y adaptativos ).
Introducido por el estándar F2003.
TRANSPOSE(matrix) es una función que transpone (intercambia filas y columnas) una matriz bidimensional.
Modern Fortran proporciona funciones integradas para empaquetar y desempaquetar una matriz multidimensional en una matriz unidimensional (y desde una matriz unidimensional, respectivamente) bajo el control de una condición lógica para mejorar el rendimiento y ahorrar memoria.
PACK(array, mask [, vector]) - función; empaqueta una matriz multidimensional de cualquier tipo de matriz en una matriz vectorial unidimensional controlada por la máscara de matriz lógica . El vector de matriz unidimensional opcional debe ser del mismo tipo de datos que array , y el número de elementos en vector , si se proporciona, debe ser al menos tanto como el número de elementos verdaderos en mask . Si máscara es un escalar con valor , entonces el número de elementos en la .TRUE.matriz vectorial , si se proporciona, debe ser al menos tan grande como el número total de elementos en la matriz .
El resultado de la función será un arreglo unidimensional del mismo tipo que arreglo . La longitud del resultado será igual a la longitud del vector , si se da uno; si no, entonces el número de elementos verdaderos en la matriz de máscaras . Si vector no se especifica y mask es un escalar con el valor true , entonces la longitud del vector unidimensional resultante es igual al número de elementos en array .
La matriz resultante se llena secuencialmente con elementos de la matriz ( en el orden en que se colocan en la memoria de la computadora) que satisfacen los valores verdaderos de la matriz de máscaras . En este caso, si se proporciona un vector , los elementos que faltan (posiblemente) en la matriz de resultados se seleccionan a partir de él y comienzan desde el índice que sigue al último elemento verdadero de la matriz de matriz en orden .
Entonces, para una matriz, el
resultado de la función será una matriz unidimensional . Si además se da un vector , entonces el resultado será .
PACK(A, mask=A.NE.0)PACK(A, mask=A.NE.0, V)
Fortran tiene ricas herramientas integradas para operaciones de entrada y salida, incluso para grandes arreglos de datos. Los archivos en Fortran son internos y externos.
Un archivo interno es cualquier matriz, cadena de caracteres o subcadena. Los archivos internos siempre están abiertos de forma predeterminada. Un archivo externo es cualquier archivo que es externo al programa que se está ejecutando.
Ambos tipos de archivos utilizan los mismos operadores de escritura WRITEy lectura READ. Los archivos internos se utilizan para conversiones de número-cadena-número y para crear entradas mixtas de números y caracteres.
Carácter ( 15 ) cadena Real :: xyz =- 12 3.456 Integer intg Escribe ( cadena , * ) xyz ! Escriba el número -123.456 en la cadena Print * , 'string=' , string ! cadena=␣ -123.4560␣ ␣ Leer ( cadena , '(I6)' ) intg ! Leer un entero de una cadena Print * , 'intg=' , intg ! intg=␣ -123 !...Los archivos externos se subdividen en formateados (texto, secuencias CR y LF), binarios (binarios), que muestran directamente la RAM y sin formato (no binarios). Además, pueden ser archivos de acceso directo y secuencial con registros de longitud fija y variable (variable solo para archivos secuenciales), así como registros segmentados (para archivos secuenciales muy grandes sin formato). Por lo tanto, Fortran le permite crear y procesar una cantidad bastante grande de tipos de archivos: 15 formas de organizar. El tamaño máximo de un registro es de 2,14 bytes.
Los archivos de acceso directo le permiten realizar operaciones de E/S en registros con un número determinado (sin sobrescribir las entradas superiores o inferiores del archivo).
Al abrir un archivo secuencial, puede colocarse al principio, al final (antes de escribir "fin de archivo"), lo que le permite agregar datos a medida que se acumulan sin sobrescribir los ingresados anteriormente, así como al principio o al final, dependiendo del estado del archivo previamente definido (estaba abierto antes o no).
En los estándares del lenguaje moderno (comenzando con Fortran'90 o Fortran'95) es posible trabajar con tres tipos de matrices dinámicas (matrices automáticas, asignadas y de referencia), punteros , enlaces ; hay procedimientos integrados para trabajar directamente con la memoria y procedimientos para operaciones bit a bit.
Los estándares del lenguaje asumen la liberación automática de RAM ocupada por cualquier tipo de matriz dinámica después de la finalización del programa o subrutina (procedimiento, función), incluso si el programador no realizó explícitamente la liberación. Esto le permite evitar fugas de memoria cuando trabaja con arreglos dinámicos usando Fortran (cuando se usan punteros y en otros casos, las fugas son posibles) con una programación descuidada [12] .
Las matrices automáticas (colocadas por el compilador) se crean cuando se llama a una subrutina y son sus objetos locales. Sus límites se definen cada vez que se invoca el procedimiento; al mismo tiempo, cuando sales, se destruyen y se borra la memoria. Las matrices automáticas están en la pila durante la ejecución del programa , asignadas (usando el operador ALLOCATE) - en el montón [25] .
Los punteros Fortran son similares a los punteros C [12] , sin embargo, cuando se resuelven problemas computacionales y se desarrollan algoritmos matemáticos, en la mayoría de los casos son reemplazados con éxito por otras herramientas Fortran.
Comenzando con el estándar F'90, la construcción de rama condicional IF THEN - ELSEIF THEN - END IFno difiere de construcciones similares en otros idiomas modernos y reemplazó a los llamados. "aritmética" IFcon etiquetas, en desuso [9] . También existe una forma más simple del operador condicional: , donde el operador ejecutable debe ser el único después del operador , por ejemplo, . IF(логическое условие) операторIFGoto метка
La construcción de elección SELECT CASE - CASE - CASE DEFAULT - END SELECT, por el contrario, difiere de la construcción SWITCH - CASEen lenguajes similares a C [28] , lenguaje Java [29] [30] y se parece un poco al operador CASEen Pascal [31] en términos de sus capacidades .
Entero N ( 74 ) ! select expression es un número entero, lógico, expresión de caracteres o ... ! matriz de enteros o matriz de caracteres con cualquier número de elementos distinto de cero nameSC : Seleccione Caso ( N ( i )) ! nameSC — nombre de la construcción, N(i) — elemento de matriz Caso (: - 5 ) ! se realiza para todo N(i) menor o igual a -5 con un paso de +1 ! Bloque 1 Caso ( -3 , -1 , 0 , 2 ) ! _ _ para variable N(i) igual a -3, -1, 0, 2 ! Bloque 2 Caso ( 50 : 100 ) ! para N(i) en el rango de 50 a 100 inclusive (paso +1) ! Bloque 3 Caso ( 400 ) ! para N(i)=400 ! Bloque 4 Caso ( 1 , 20 : 30 , 35 ) ! para N(i)=1, N(i) en el rango de 20 a 30 inclusive y N(i)=35 ! Bloque 5 Caso por defecto ! para todas las demás situaciones. Predeterminado del caso: declaración opcional opcional ! bloque por defecto Finalizar Seleccionar nombreSCSi el valor de la variable N, llamada expresión de selección, coincide con la lista de selectores (lista de valores o rango) en cualquier sentencia CASE, por ejemplo, en la tercera para N=70 , entonces, luego de ejecutar el bloque correspondiente de sentencias Block-3 , la construcción sale SELECT CASE[ 12] [25] , y las sentencias de interrupción (como BREAK) no son requeridas . El operador CASE DEFAULT, así como el nombre de la construcción, no son necesarios. Los rangos en las listas de selectores de diferentes operadores CASEno deben superponerse o tener al menos un elemento común.
La expresión de elección (N) puede ser un elemento de una matriz de enteros.
Los rangos en las listas de selección se aplican solo a números enteros o caracteres, en orden ascendente de abajo hacia arriba; para caracteres, en orden ascendente de sus códigos .
En Fortran moderno, hay dos formas de bucles con un contador de iteraciones: la construcción tradicional y DOel ENDDObucle etiquetado. Este último ha sido reconocido como un diseño obsoleto desde el estándar F'90, pero aún está incluido en los estándares. Para bucles anidados con más anidamiento, la última forma puede ser más concisa:
! Sintaxis de la construcción de bucle en desuso hacer 1 k = 1 , 10 ! 1 es el final del ciclo hacer 1 j = 1 , 20 ! la etiqueta puede ser la misma para bucles anidados hacer 1 i = 1 , 100 arr ( yo , j , k ) = 25 1 ¡ Continuar ! la etiqueta solo puede aparecer ante cualquier operador L1 : hacer k = 1 , 10 ! sin embargo, la notación moderna le permite nombrar ciclos, L2 : hacer j = 1 , 20 ! que está más en línea con el concepto de programación estructurada L3 : i = 1 , 100 ! _ y hace que sea más fácil evitar errores arr ( yo , j , k ) = 25 terminar hacer L3 terminar hacer L2 terminar hacer L1El encabezado del bucle con un contador tiene la siguiente forma completa:
nombre : DO I = N1 , N2 , dN ! ¡ el nombre de la construcción es opcional ... ! N1 es el valor inicial del contador, N2 es el valor final, dN es el paso (parámetros de bucle) END DO nombre ! Los parámetros del bucle son números enteros de signo arbitrario. dN es distinto de cero.El número de ejecución del cuerpo del bucle sin interrupción es N c = max( int((N 2 -N 1 +dN)/dN), 0 ) y puede ser igual a cero.
Por ejemplo, el bucle se ejecutará cero veces si N 2 <N 1 y el paso es positivo: dN>0 . Si el paso es negativo, dN<0, entonces el ciclo irá en orden inverso, mientras que para completar el ciclo es necesario que N 2 <N 1 . Si se omite el paso dN , entonces por defecto se considera igual a uno: dN=1 .
El valor de la variable del bucle I después de salir siempre es igual a N s +1 , es decir , uno más que el número de iteraciones del bucle y no menos de uno: I≥1.
También es posible tener un ciclo condicional - , y un ciclo infinito - , que tienen una forma estándar. DO WHILE(логическое условие)END DODOEND DO
El operador CYCLE имя циклаinterrumpe la iteración actual del bucle y continúa con la siguiente iteración del mismo bucle. CYCLESi no se especifica el nombre del bucle, se interrumpe la iteración del bucle actual (en el que se encuentra el operador ).
El operador EXIT имя циклаinterrumpe la ejecución del bucle con el nombre especificado y transfiere más control, y si no hay nombre, interrumpe el bucle actual (en el que está anidado el operador EXIT).
Los operadores CYCLEy son EXITlógicamente idénticos a un operador GOTO(en las circunstancias apropiadas), pero hacen que el código sea mucho más fácil de entender y mantener para el programador.
Desde el desarrollo inicial del lenguaje , los compiladores de Fortran han sido fabricados por IBM. Actualmente, IBM envía el compilador de optimización VS Fortran [32] para mainframes IBM System z , cuya historia de desarrollo de varias versiones se remonta a 1964, así como el compilador XL Fortran [33] para plataformas basadas en la arquitectura PowerPC : AIX , Linux y la supercomputadora Blue Gene (también hubo una versión para Mac OS X , cuando las computadoras Macintosh usaban procesadores PowerPC). Ambos compiladores contienen optimizadores muy sofisticados, resultado de medio siglo de trabajo científico continuo por parte de los especialistas de IBM. Sobre la base del compilador IBM Fortran XL, Absoft, un socio comercial de IBM, ha creado y suministra el compilador Absoft Pro Fortran para sistemas basados en procesadores PowerPC (Linux, Mac OS X) e Intel (Linux, Mac OS X, Windows) [34] .
Hasta 1997 , Microsoft Corporation fue uno de los principales fabricantes del compilador Fortran para el sistema operativo Windows . Posteriormente, abandonó su desarrollo debido a la baja rentabilidad. Posteriormente, el compilador fue suministrado por DEC , que pasó a formar parte de Compaq en 1998 y, junto con esta última, se fusionó con HP en 2002 . Intel desarrolló aún más esta versión del compilador , y el compilador se llama Intel Fortran Compiler , que le permite optimizar el código para las plataformas Intel IA-32, x86_64 e IA-64.
DEC proporciona un compilador integrado en el entorno de desarrollo de Digital Visual Fortran basado en Microsoft Visual Studio . Los productos más famosos de esta línea son FPS 4.0 (Microsoft Fortran Power Station), DVF 5.0 y 6.0. Cada compilador puede admitir múltiples estándares de Fortran. Las fusiones hicieron que aparecieran productos posteriores en el mercado con las marcas Compaq y HP. HP vende actualmente un entorno de desarrollo versión 6.6 para Intel/win32. La compatibilidad con Fortran también se implementa para todas las plataformas de alto rendimiento de HP.
Otro importante proveedor de sistemas de desarrollo de Fortran es Lahey , que ofrece soluciones integradas para Windows y Linux.
Durante mucho tiempo, el compilador de Watcom se consideró el mejor compilador de Fortran para PC , que se separó en un proyecto separado de Open Watcom , que desarrolla el compilador de forma abierta.
Entre los compiladores gratuitos de Fortran, cabe destacar el compilador de la ex Sun Microsystems (ahora Oracle), que forma parte de Sun Studio , que genera código eficiente bajo SPARC , x86 y x86-64 [35] y está disponible para Solaris , OpenSolaris y GNU/Linux .
La GNU Free Software Foundation lanzó el compilador g77 Fortran 77, que es de código abierto y está disponible para casi cualquier plataforma y es totalmente compatible con GCC . Ahora ha sido reemplazado por el compilador GFortran , que implementa casi todas las construcciones del estándar Fortran-95 y muchas de las construcciones de los estándares Fortran-2003, Fortran-2008 y Fortran-2018. También es totalmente compatible con versiones anteriores de Fortran-77. También hay un proyecto g95 independiente para crear un compilador Fortran-95 basado en GCC .
Muchos sistemas de programación le permiten vincular archivos objeto obtenidos como resultado de la traducción de un programa Fortran con archivos objeto obtenidos de compiladores de otros lenguajes, lo que le permite crear aplicaciones más flexibles y multifuncionales. También hay disponible una gran cantidad de bibliotecas para el lenguaje Fortran, que contienen rutinas para resolver problemas computacionales clásicos ( LAPACK , IMSL , BLAS ), tareas para organizar la computación distribuida ( MPI , PVM ) y tareas para construir interfaces gráficas ( Quickwin , FORTRAN / TK ) o acceso a DBMS ( Oracle ).
Fortran apareció en la URSS más tarde que en Occidente, ya que al principio en la URSS el algol se consideraba un lenguaje más prometedor . La comunicación de los físicos soviéticos con sus colegas del CERN , donde en la década de 1960 casi todos los cálculos se realizaban utilizando programas Fortran, desempeñó un papel importante en la introducción de Fortran [36] .
El primer compilador Fortran soviético se creó en 1967 para la máquina Minsk-2 , pero no recibió mucha fama. La introducción generalizada de Fortran comenzó después de la creación en 1968 del compilador FORTRAN-DUBNA para la máquina BESM-6 . Fortran es el lenguaje principal para las computadoras ASVT y SM , a menudo se usa junto con el preprocesador RATFOR . Las computadoras ES que aparecieron en 1972 ya tenían inicialmente un traductor Fortran ("prestado" del IBM/360 junto con otro software).
En la década de 1970, el IPM desarrolló la biblioteca gráfica GRAFOR (“Graphic Extension of FORtran”) [37] .
A fines de la década de 1980 y principios de la de 1990, el físico Andrei Zaretsky creó una serie de libros para niños, uno de los personajes principales de los cuales era el profesor Fortran , quien explicaba los conceptos básicos de alfabetización y programación informática a los niños en un lenguaje accesible [38] [39] .
en redes sociales | ||||
---|---|---|---|---|
sitios temáticos | ||||
diccionarios y enciclopedias | ||||
|
Lenguajes de programación | |
---|---|
|