Jump to content

Fortran

From Wikipedia, the free encyclopedia

This is an old revision of this page, as edited by Rwwww (talk | contribs) at 07:09, 2 August 2006 (→‎High Performance Fortran). The present address (URL) is a permanent link to this revision, which may differ significantly from the current revision.

The Fortran Automatic Coding System for the IBM 704 (October 15, 1956), the first Programmer's Reference Manual for FORTRAN

"In the beginning, there was FORTRAN." (Jim Gray)

FORTRAN is a general-purpose[1], procedural[2], imperative programming language that is especially suited to numeric computation and scientific computing. Originally developed by International Business Machines Corporation (IBM) in the 1950s for scientific and engineering applications, FORTRAN came to dominate this area of programming early on and has been in continuous use in specialized applications such as climate modeling (more) and molecular dynamics modeling (more) for half a century.

FORTRAN (an acronym derived from its original official name, The IBM Mathematical Formula Translating System) encompasses a lineage of versions, each of which evolved to add extensions to the language while retaining compatibility with previous versions. Successive versions have added support for processing of character-based data (FORTRAN 77), array programming (Fortran 90), module-based programming (Fortran 90), object-based programming (Fortran 90), and object-oriented and generic programming (Fortran 2003).

The names of earlier versions of the language through FORTRAN 77 were conventionally spelled in all-caps (FORTRAN 77 was the last version in which the use of lowercase letters in keywords was strictly nonstandard). The capitalization has been dropped in referring to newer versions beginning with Fortran 90. The official language standards now refer to the language as "Fortran."[3]

History

An IBM 704 mainframe (image courtesy of LLNL)

In late 1953, John W. Backus submitted a proposal to his superiors at IBM to develop a more efficient alternative to assembly language for programming their IBM 704 mainframe computer. A draft specification, "Preliminary Report : Specifications for the IBM Mathematical FORmula TRANslating System, FORTRAN" (PDF). November 10, 1954. {{cite journal}}: Cite journal requires |journal= (help), was prepared. Although originally intended for internal use[4] only on the IBM 704 computer, the language was eventually made available to any customer purchasing an IBM 704. The first "official" manual for FORTRAN appeared in October 1956, with the first FORTRAN compiler delivered in April 1957. This was an optimizing compiler, because customers were reluctant to use a high-level programming language unless its compiler could generate code whose performance was comparable to that of hand-coded assembly language.

The language was widely adopted by scientists for writing numerically intensive programs, which encouraged compiler writers to produce compilers that could generate faster and more efficient code. The inclusion of a complex number data type in the language made Fortran especially suited to technical applications such as electrical engineering.

By 1960, versions of FORTRAN were available for the IBM 709, 650, 1620, and 7070 computers. Significantly, the increasing popularity of FORTRAN spurred competing computer manufacturers to provide FORTRAN compilers for their machines, so that by 1963 over 40 FORTRAN compilers existed. For these reasons, FORTRAN is considered to be the first, widely used cross-platform programming language[citation needed].

The development of FORTRAN paralleled the early evolution of compiler technology; indeed many advances in the theory and design of compilers were specifically motivated by the need to generate efficient code for FORTRAN programs.

FORTRAN initial release

The initial release of FORTRAN for the IBM 704 contained 33 statements. Thirty-two are described in the Programmer's Reference Manual (PDF)., including:

  • DIMENSION and EQUIVALENCE statements
  • Assignment statements
  • Three-way arithmetic IF statement. It is interesting to note that this statement, the primary branching construct, corresponds directly to the branch instruction on the IBM 704, which was a three-way branch.
  • IF statements for checking exceptions (ACCUMULATOR OVERFLOW, QUOTIENT OVERFLOW, and DIVIDE CHECK); and IF statements for manipulating sense switches and sense lights
  • GOTO, computed GOTO, ASSIGN, and assigned GOTO
  • DO loops
  • Formatted I/O: FORMAT, READ, READ INPUT TAPE, WRITE, WRITE OUTPUT TAPE, PRINT, and PUNCH
  • Unformatted I/O: READ TAPE, READ DRUM, WRITE TAPE, and WRITE DRUM
  • Other I/O: END FILE, REWIND, and BACKSPACE
  • PAUSE, STOP, and CONTINUE
  • FREQUENCY statement (for providing optimization hints to the compiler)

The function-statement "was added to the system after the manual was written and before the system was distributed" Backus (1978). "The History of Fortran I, II, and III". {{cite journal}}: Cite journal requires |journal= (help).


FORTRAN II

IBM's FORTRAN II appeared in 1958. The main enhancement was to support procedural programming by allowing user-written subroutines and functions. Six new statements were introduced:

  • SUBROUTINE, FUNCTION, and END
  • CALL and RETURN
  • COMMON

Over the next few years, FORTRAN II would also add support for the DOUBLE PRECISION and COMPLEX data types.

FORTRAN III

IBM also developed a FORTRAN III in 1958 that allowed for inline assembler code among other features; however, this version was never released as a product. Like the 704 FORTRAN and FORTRAN II, FORTRAN III included machine-dependent features that made code written in it unportable from machine to machine. Early versions of FORTRAN provided by other vendors suffered from the same disadvantage.

A pad of Fortran coding forms

FORTRAN IV

Starting in 1961, as a result of customer demands, IBM began development of a FORTRAN IV that removed the machine-dependent features of FORTRAN II (such as READ INPUT TAPE), while adding new features such as a LOGICAL data type, logical Boolean expressions and the logical IF statement as an alternative to the arithmetic IF statement. FORTRAN IV was eventually released in 1962, first for the IBM 7030 ("Stretch") computer, followed by versions for the IBM 7090 and 7094.

The increasing popularity of FORTRAN at this time led to the publication of the first book written by an independent author[citation needed] about a programming language, Daniel D. McCracken's A Guide to Fortran Programming (Wiley, 1961).


FORTRAN 66

Perhaps the most significant development in the early history of FORTRAN was the decision by the American Standards Association (now ANSI) to form a committee to develop an "American Standard Fortran." The resulting two standards, ratified in March 1966, defined two languages, FORTRAN (based on FORTRAN IV, which had served as a de facto standard), and Basic FORTRAN (based on FORTRAN II, but stripped of its machine-dependent features). The FORTRAN defined by the first standard became known as FORTRAN 66 (although many continued to refer to it as FORTRAN IV, the language upon which the standard was largely based). FORTRAN 66 effectively became the first "industry-standard" version of FORTRAN. FORTRAN 66 included:

  • Main program, SUBROUTINE, FUNCTION, and BLOCK DATA program units
  • INTEGER, REAL, DOUBLE PRECISION, COMPLEX, and LOGICAL data types
  • COMMON, DIMENSION, and EQUIVALENCE statements
  • DATA statement for specifying initial values
  • Intrinsic and EXTERNAL (e.g., library) functions
  • Assignment statement
  • GOTO, assigned GOTO, and computed GOTO statements
  • Logical IF and arithmetic (three-way) IF statements
  • DO loops
  • READ, WRITE, BACKSPACE, REWIND, and ENDFILE statements for sequential I/O
  • FORMAT statement
  • CALL, RETURN, PAUSE, and STOP statements
  • Hollerith constants in DATA and FORMAT statements, and as actual arguments to procedures
  • Identifiers of up to six characters in length
  • Comment lines

In 1975-1976 Will Crowther and Don Woods using Fortran create the first computer adventure game, Colossal Cave Adventure (also known as ADVENT or Colossal Cave).

FORTRAN 77

After the release of the FORTRAN 66 standard, compiler vendors introduced a number of extensions to "Standard Fortran," prompting ANSI in 1969 to begin work on revising the 1966 standard. Final drafts of this revised standard circulated in 1977, leading to formal approval of the new Standard FORTRAN in April 1978. The new standard, known as FORTRAN 77, added a number of significant features to address many of the shortcomings of FORTRAN 66:

  • CHARACTER data type, with vastly expanded facilities for character input and output and processing of character-based data
  • IMPLICIT statement
  • Block IF statement, with optional ELSE and ELSE IF clauses, to provide improved language support for structured programming
  • OPEN, CLOSE, and INQUIRE statements for improved I/O capability
  • Direct-access file I/O
  • PARAMETER statement for specifying constants
  • SAVE statement for persistent local variables
  • Generic names for intrinsic functions

An important practical extension to FORTRAN 77 was the release of MIL-STD-1753 in 1978. This specification, developed by the U. S. Department of Defense, standardized a number of features implemented by most FORTRAN 77 compilers but not included in the ANSI FORTRAN 77 standard. These features would eventually be incorporated into the Fortran 90 standard.

  • DO WHILE and END DO statements
  • INCLUDE statement
  • IMPLICIT NONE variant of the IMPLICIT statement
  • Bit manipulation intrinsics, based on the Industrial Real-Time Fortran (IRTF) Standard

The development of a revised standard to succeed FORTRAN 77 would be repeatedly delayed as the standardization process struggled to keep up with rapid changes in computing and programming practice. In the meantime, as the "Standard FORTRAN" for nearly fifteen years, FORTRAN 77 would become the historically most important dialect.

Fortran 90

The much delayed (see: Meek, Brian (October 1990). "The Fortran (not the foresight) saga: the light and the dark". Fortran Forum. 9 (2).) successor to FORTRAN 77, informally known as Fortran 90, was finally released as an ANSI Standard in 1992. This major revision added many new features to reflect the significant changes in programming practice that had evolved since the 1978 standard:

  • Free-form source input, also with lowercase Fortran keywords
  • Modules, to group related procedures and data together
  • RECURSIVE procedures
  • A vastly improved argument-passing mechanism, allowing interfaces to be checked at compile time
  • User-written interfaces for generic procedures
  • Operator overloading
  • Derived/abstract data types
  • New data type declaration syntax, to specify the data type and other attributes of variables
  • Ability to operate on arrays as a whole in expressions and assignment statements, thus greatly simplifying math and engineering computations
  • Dynamic memory allocation by means of the ALLOCATABLE attribute and the ALLOCATE and DEALLOCATE statements
  • POINTER attribute, pointer assignment, and NULLIFY statement to facilitate the creation and manipulation of dynamic data structures
  • CASE construct for multi-way selection
  • EXIT and CYCLE statements for "breaking out" of normal DO loop iterations in an orderly way
  • Identifiers up to 31 characters in length
  • Inline comments
  • Portable specification of numerical precision under the user's control

High Performance Fortran

About 1993 the High Performance Fortran Forum (HPFF) was organized. From http://www.netlib.org/hpf: "HPF is a set of extensions to Fortran 90 that provide access to high-performance architecture features while maintaining portability across platforms. HPFF is a coalition of industry, academic and laboratory representatives that works to define the HPF language specification." HPF uses a SIMD model of computation to support spreading the work of a single array computation over multiple processors. HPFF produced a version 1 document in 1993-94, a version 2 in 1997.

Fortran 95

Fortran 95 was a minor revision, mostly to resolve some outstanding issues from the Fortran 90 standard. Nevertheless, Fortran 95 also added a number of extensions, notably from the High Performance Fortran dialect:

  • FOR ALL and nested WHERE constructs to aid vectorization
  • User-defined PURE and ELEMENTAL procedures

An important supplement to Fortran 95 was the ISO technical report TR-15581: Enhanced Data Type Facilities, also known informally as the "Allocatable TR." This specification defined enhanced use of ALLOCATABLE arrays, prior to the availability of fully Fortran 2003-compliant Fortran compilers. Such uses include ALLOCATABLE arrays as derived type components, in procedure dummy argument lists, and as function return values. (ALLOCATABLE arrays are preferable to POINTER-based arrays because ALLOCATABLE arrays are guaranteed by Fortran 95 to be deallocated automatically when they go out of scope, eliminating the possibility of memory leakage. In addition, aliasing is not an issue for optimization of array references, allowing compilers to generate faster code than in the case of pointers.)

Fortran 2003

The most recent standard, Fortran 2003, is a major revision introducing a number of extensions:

A comprehensive summary of the new features of Fortran 2003 is available for download as a PDF file or gzipped PostScript file.

Current Fortran activities

See the ISO Fortran Working Group (WG5) web site for, as described there, "... a summary of the present state of standardization of Fortran and of current developments which will, in due course, lead to further extensions to the language, as well as a brief history of the standardization of the Fortran language and the names of the key individuals and organisations involved:

  • Overview of WG5 and the Standardization Process
  • WG5 Officers and Project Editors
  • WG5 Projects
  • WG5 Electronic Document Archive
  • WG5 Email List
  • Future WG5 Meetings"

Also see: Co-array Fortran.

There is a Fortran newsgroup, comp.lang.fortran on Usenet.

A mailing list for Fortran 90, COMP-FORTRAN-90 Discussion Group and Archives.

The Association for Computing Machinery (ACM) Special Interest Group for Programming Languages (SIGPLAN) publishes the Fortran Forum (subscription required). From their website "SIGPLAN FORTRAN Forum (not included in membership) Addresses the FORTRAN language, its uses, profitability, standardization, further evolution, and the implementation of FORTRAN processors. Published 3 times per year."

A number of national standards organizations have an active Fortran committee.

The legacy of Fortran

Fortran has been around since 1957 and there is now a vast body of Fortran in daily use throughout the scientific and engineering communities. It is the primary language for some of the most intensive supercomputing tasks, such as weather/climate modeling, computational chemistry, quantum chromodynamics (QCD), and simulation of automobile crash dynamics. Indeed, one finds that even today, half a century later, floating-point benchmarks to gauge the performance of new computer processors are still written in Fortran (e.g., CFP2000, the floating-point component of the SPEC CPU2000 benchmarks), an eloquent tribute to the longevity of the language.

Language features

The following subsections are not intended to be a comprehensive overview of the Fortran language; that role properly belongs to any of the excellent Fortran textbooks listed at the end of this article. Rather, the following describe some of the more salient features of the language, particularly those peculiar to Fortran. Although both old and new features are described, few of the historic features are used in modern programs. Still, most have been retained in the language to maintain backward compatibility.

Fixed-form source

Fortran punch card, with column markings

Early FORTRAN programs were handwritten on coding forms specially designed for the purpose or on plain paper and then punched on punch cards. Each punch card had 80 columns, numbered 1–80, and represented a single line of source input. Formatting rules were rigid:

  • If a C (or asterisk, as permitted by FORTRAN 77) was punched in column 1, the card was a comment line, ignored by the compiler
  • A numeric statement label (in the range 199999) could be punched in columns 1–5 if needed
  • If any (non-blank) character other than zero was punched in column 6, the card was taken to be a continuation of the previous statement. The maximum number of continuation lines was standardized to 19 with FORTRAN 77
  • The contents of columns 7–72 were treated as a statement (or a continuation of the previous statement in the case of a continuation card)
  • Columns 73–80 were completely ignored by the compiler. These columns were often used for sequence numbers, as a precaution in case the card deck should get dropped or out of order. (With programs consisting of hundreds to thousands of lines, this was a serious issue.) Some source code tools would also use these columns to record modification history. It might seem those eight blank columns reflected intentional design. Not so. Those 8 blank columns were an artifact of the IBM 704. A machine with a 36 bit word, the commonly used card reader control panel read a punched card row by row, two words to a row. Two times 36 is 72; columns 73-80 could not be read.

As programs moved to magnetic media, and interactive terminals replaced keypunches and card readers, fixed-form source became increasingly restrictive. Fixed form also allowed certain errors to be silently accepted by compilers, leading to hidden bugs. Fortran 90 introduced free-form source to resolve these problems. As of Fortran 95, fixed-form source is now considered a deprecated feature of the language.

Treatment of blanks

Blanks, in general, were not significant in fixed-form FORTRAN programs. For example, the GOTO statement could equivalently be (and very often was) punched as GO TO. Likewise, a statement such as[5]

DO 1 I = 1, 10

could equivalently be punched as:

DO1I=1,10

Minor typos could lead to wildly different semantics; for example, if the comma above were inadvertently punched as a period, the above statement (intended to implement a DO loop) would be interpreted as the assignment statement

DO1I = 1.10

This would not be flagged by the compiler as an error, due to the implicit typing mechanism (see below) which allows variables to be declared by their first use. (As of Fortran 90, implicit typing can be disabled.). Computer folklore has attributed, incorrectly, the loss of an early U.S.A. rocket to such an error. See http://catless.ncl.ac.uk/Risks/9.54.html#subj1.

In addition, in order to save the keypunch operator the trouble of punching lots of zeroes in input data cards, blanks in numeric data fields were, depending on the context and on the particular compiler, interpreted by FORTRAN input statements as zeroes.

As part of the Fortran 90 free-form source form, blank significance is now required. The older rules were retained for the now-deprecated fixed-form source.

Statement numbers and GOTO statements

Early FORTRAN programs (prior to Fortran 90) relied heavily on statement numbers and the GOTO statement to implement control structures. For example, the number immediately following the DO keyword above specified the statement number of the last statement enclosed in the body of the DO loop.

Using the IF statement combined with the GOTO statement, it was possible to implement looping constructs such as a while (condition) do ... construct (with the test at the top, illustrated here using the block IF statement introduced with FORTRAN 77):

    1 IF (condition) THEN
        executable_statements
        GOTO 1
      END IF

as well as a repeat ... until (condition) construct (with the test at the bottom, illustrated here using the logical IF statement introduced with FORTRAN 66):

    1 executable_statements
      IF (.NOT.condition) GOTO 1

Edsger W. Dijkstra wrote a classic paper on the GOTO statement, "GoTo Considered Harmful". 11 (3). CACM. March 1968. {{cite journal}}: Cite journal requires |journal= (help).

Statement numbers were also required as targets to handle I/O exceptions, such as those handled by the END= and ERR= specifiers in I/O statements such as READ and WRITE, among others.

Free-form source

Fortran 90 introduced free-form source input, which removed almost all the column-based restrictions on the input layout discussed above. Lines could be as many as 132 characters long. Blanks were now significant as token delimiters. Statements could be continued on following lines by ending incomplete lines with &.

As of Fortran 95, the older fixed-form source is considered deprecated.

Identifiers

Names of identifiers in early compilers were usually limited to the number of alphanumeric characters that fit in one machine word; six characters was typical (the original IBM 704 used 36-bit machine words, and alphanumeric characters were encoded in 6-bit binary-coded decimal, packed six characters per word). However, the maximum could be as few as three characters, depending on the machine.

Both the FORTRAN 66 and FORTRAN 77 standards required compilers to support at least six characters for identifiers, though many compilers supported longer identifiers. Fortran 90 guaranteed support for identifiers of up to 31 characters, while Fortran 2003 extended this support to 63 characters.

Implicit typing

By default, numeric variables did not need to be declared before use; their declaration was implicit with their first appearance in a program. Furthermore, variables whose names began with any of the letters from I through N were implicitly assumed to be INTEGER variables, whereas all other variables were assumed to be REALs. (In mathematics, the symbols i through n are typically used as (integer) subscripts and superscripts.) This implicit typing mechanism could be overridden using an IMPLICIT statement to change the default types associated with variables based on their initial letter.

Alternatively, explicit variable declarations could be used to declare the types of variables. Explicit declarations were also necessary for variables of type other than REAL and INTEGER (namely, for the DOUBLE PRECISION, COMPLEX, LOGICAL, and CHARACTER data types).

Because simply mistyping a variable name effectively caused a new variable to be declared according to the implicit typing rules above (assuming the mistyped name was not previously declared), later compilers provided an IMPLICIT NONE statement to allow turning off the implicit typing mechanism, requiring all variables to be declared before use. The IMPLICIT NONE statement became standard with Fortran 90. Most Fortran coding standards mandate the use of IMPLICIT NONE.

The arithmetic IF statement

In the earliest versions of Fortran, the only conditional branching statement provided was the three-way arithmetic IF statement:

      IF (expression) n1, n2, n3

The conditional expression was an INTEGER- or REAL-valued numeric expression, and control would be passed to statement number n1, n2, or n3 depending on whether the expression was negative, zero, or positive, respectively.[6]

Although seemingly arcane, the arithmetic IF statement was convenient shorthand for branching based on the three possible conditions of , , or , in the case where the test expression was of the form X - Y. Thus, one was often taught mentally to view a statement of the form

      IF (X - Y) 100, 200, 300

as FORTRAN idiom for the following group of equivalent statements (here, expressed as logical IF statements, which were not introduced until FORTRAN 66):

      IF (X .LT. Y) GO TO 100
      IF (X .EQ. Y) GO TO 200
      IF (X .GT. Y) GO TO 300

Obviously, in the case of "less-than-or-equal-to" or "greater-than-or-equal-to," the first two or last two, respectively, of the GOTO targets would be the same.

(With the introduction of the more modern forms of IF statements described below, the arithmetic IF statement is very seldom seen in modern Fortran programs.)

More IF statements

As alluded to previously, FORTRAN 66 introduced the logical IF statement, which allows a single Fortran statement, including the GOTO statement, to be conditionally executed:

      IF (expression) executable_statement

FORTRAN 77 introduced the block IF statement with optional ELSE IF and ELSE clauses to support structured programming:

      IF (expression) THEN
          executable_statements
      ELSE IF (expression) THEN
          executable_statements
      ELSE
          executable_statements
      END IF

The CASE statement

Fortran 90 introduced the case selection structure to implement multi-way branches. The following artificial example mimics the arithmetic IF statement above (but only in the case of an integer-valued test expression, as values such as or would fall through the cracks in this example):

      select case(i - j)
          case(:-1)
              ! Former code jumped to by GOTO 100
          case(0)
              ! Former code jumped to by GOTO 200
          case(1:)
              ! Former code jumped to by GOTO 300
      end select

The ENTRY statement

The earliest versions of FORTRAN were developed before modern recommended practices on topics such as control flow had matured. One example is the ENTRY statement, which allows a program to jump to an arbitrary point within a subprogram, violating the single-entry control flow philosophy of modern programming practice. In the example below, FUN is a function expecting three variable arguments, while FUNALT is an alternate entry point invoked with only two arguments; for illustration, the third argument is given a default value for the alternate entry point:

      FUNCTION FUN(A, B, OPTARG)
          statements specific to FUN
          GOTO 1
      ENTRY FUNALT(A, B)
          OPTARG = 0
          statements specific to FUNALT
    1     statements common to both
          RETURN
      END

Equally as interesting, FORTRAN also supports alternate RETURN, where a subprogram can specify an alternate point to which to return, rather than the original point at which the subprogram was invoked.

As of Fortran 95, both the ENTRY statement and alternate RETURN are now considered deprecated features.

The assigned GOTO statement

In this feature of historic Fortran, a variable may be assigned a statement label (typically implemented as a code pointer) for subsequent use as branch destination or to specify a format statement.

      INTEGER NEXT
      ASSIGN 45 TO NEXT
   10 GOTO NEXT
   15 ASSIGN 90 TO NEXT
      GOTO 10
   25 ASSIGN 15 TO NEXT
      GOTO 10
   35 ASSIGN 25 TO NEXT
      GOTO 10
   45 ASSIGN 35 TO NEXT
      GOTO 10
   90 CONTINUE

This contrived example jumps to statement 45 (after the first ASSIGN), which reassigns the variable and jumps to 10 to execute the indirect jump to 35, and so on until statement 15 breaks out to the CONTINUE at statement 90.

More usefully, FORMAT statements may be assigned:

      SUBROUTINE OCCURR(N)
        INTEGER IFMT
   10   FORMAT ('No occurrences found.')
   11   FORMAT ('One occurrence found.')
   12   FORMAT (I4, ' occurrences found.')
*       If N < 1, goto 20; if N = 1, goto 21; if N > 1, goto 22
        IF (N - 1) 20, 21, 22
   20   ASSIGN 10 TO IFMT
        GOTO 30
   21   ASSIGN 11 TO IFMT
        GOTO 30
   22   ASSIGN 12 TO IFMT
   30   PRINT IFMT, N
        RETURN
      END

Many implementations did not protect against invalid assignments or the use of assigned variables, even allowing arithmetic. The following code fragment illustrates some possible constructs that could potentially be found acceptable to a Fortran compiler, but would almost certainly lead to a program crash when executed.

      SUBROUTINE CRASH4
        INTEGER IDEST

*       Jump to random location pointed to by uninitialized var. IDEST
        GOTO IDEST
   10   FORMAT ('Message')
        ASSIGN 10 TO IDEST

*       Invalid jump to data
        GOTO IDEST
   20   CONTINUE
        ASSIGN 20 TO IDEST

*       Attempt to use machine code at label 20 as format specification
*         (almost certain to give an "invalid format" error at runtime)
        PRINT IDEST
   30   ASSIGN 30 TO IDEST
        IDEST = 3 * IDEST

*       A random jump likely to fault on illegal instructions or data
        GOTO IDEST
      END

As of Fortran 95, the assigned GOTO statement has become obsolete.[7]

Variants of Fortran

Specific variants

Vendors of high-performance scientific computers (e.g., Burroughs, CDC, Cray, Honeywell, IBM, Texas Instruments, and UNIVAC) added extensions to Fortran to take advantage of special hardware features such as instruction cache, CPU pipelines, and vector arrays. For example, one of IBM's Fortran compilers (H Extended IUP) had a level of optimization which reordered the machine language instructions to keep multiple internal arithmetic units busy simultaneously. Another example is CFD, a special variant of Fortran designed specifically for the ILLIAC IV supercomputer, running at NASA's Ames Research Center.

Such machine-specific extensions have either disappeared over time or have had elements incorporated into the main standards; the major remaining extension is OpenMP, which is a cross-platform extension for shared memory programming. One new extension, CoArray Fortran, is intended to support parallel programming.

Fortran-based languages

Prior to FORTRAN 77, a number of preprocessors were commonly used to provide a friendlier language, while retaining the advantage that the preprocessed code could still be compiled on any machine with a standard FORTRAN compiler. Popular preprocessors included FLECS, MORTRAN, Ratfor, and Ratfiv. (Ratfor and Ratfiv, for example, implemented a remarkably C-like language, outputting preprocessed code in standard FORTRAN 66.[8])

Among the more popular specialized Fortran-based languages that evolved over time are SAS, for generating statistical reports, and SIMSCRIPT, for modeling and simulating large discrete systems.

F was designed to be a clean subset of Fortran 95 that attempted to remove the redundant, unstructured, and deprecated features of Fortran, such as the EQUIVALENCE statement.

Criticisms and rebuttals

Arcane syntax

As what was essentially a first attempt at designing a high-level language, Fortran's syntax is sometimes regarded as archaic by programmers familiar with subsequently developed languages such as C. Fortran has stayed abreast of such advances, however, and contemporary versions have attempted to supersede and deprecate such syntax in favor of more robust and transparent syntax.

Early FORTRAN syntax (prior to Fortran 90) did not in general treat blanks as significant, which made writing robust and efficient lexical analyzers for Fortran very difficult. Subsequent developments such as free-form source have largely resolved these issues.

A language tailored to "specialists"

Historical versions of Fortran could rightly be criticized for arbitrary conventions such as implicit typing and fixed-form source input, which together with a paucity of block structures could allow careless or obfuscated programming practices. However, modern Fortran has overcome such issues. Since its introduction, Fortran has been tuned to scientific and numerical work. Fortran 95 provides concise statements for applying mathematical operations directly to entire arrays, which not only improves program readability but also assists the compiler in vectorizing operations. As another example, from its earliest days Fortran provided full support for complex numbers (used, e.g., in the computation of Fourier transforms and in power engineering calculations), a feature not supported by ANSI C until the advent of the ISO/IEC 9899:1999 ("C99") standard.

For these reasons, while Fortran is seldom used outside of scientific and engineering numerical work, it remains the language of choice for high-performance numerical computing. It is also simple for non-programmers, particularly those with a mathematical or technical background, to learn how to write efficient, numerically robust code in Fortran.

C/C++ is 'just as fast'

Fortran was designed to create fast executable code for scientific programming. One reason for this is that compiled Fortran programs were intended to be comparable in performance to hand-written assembly language from the start. In scientific computing, this means that efficient array processing is of critical concern. Few other languages treat arrays, especially multi-dimensional arrays, as cleanly and simply as Fortran does.

In particular, Fortran data items are not allowed to alias one another, except in very well-defined and explicit situations. This assists Fortran compilers in generating very efficient code, by default, compared to pointer-intensive languages. On the other hand, explicit use of certain C features, such as the register attribute and the C99 restrict attribute can mitigate some of the performance problems of C.

Fortran 95 is slower than FORTRAN 77

Most programs in FORTRAN 77 are valid in Fortran 95, or require only minor changes. The new Fortran features often allow the programmer to express his or her intentions more precisely and explicitly, affording the compiler greater opportunity for optimization. Any observed slowdown resulting from rewriting a FORTRAN 77 program in Fortran 95 is usually caused by improper usage of Fortran 95 features.

Fortran programs require static storage allocation

It is a common misconception that Fortran requires static storage allocation for all local variables. Both the FORTRAN 66 and FORTRAN 77 Standards allowed, but did not require, a fully static implementation. FORTRAN 77 introduced the SAVE statement to allow a programmer to state explicitly which variables are required to retain their values when they go out of scope. Likewise, Fortran 90 introduced RECURSIVE procedures. In a recursive procedure, non-SAVEd local variables must be unique for each active instance of the procedure.

Code examples

The following sample programs can be compiled and run with any standard Fortran compiler (see the end of this article for lists of open-source and proprietary compilers). Most modern Fortran compilers expect a file with a .f extension (for FORTRAN 66 or FORTRAN 77 source, although the FORTRAN 66 dialect may have to be selected specifically with a command-line option) or .f90/.f95 extension (for Fortran 90/95 source, respectively).

"Retro" FORTRAN IV

A retro example of a FORTRAN IV (later evolved into FORTRAN 66) program deck is available on the IBM 1130 page, including the IBM 1130 DM2 JCL required for compilation and execution. An IBM 1130 emulator is available at IBM 1130.org that will allow the FORTRAN IV program to be compiled and run on a PC.

The Snoopy Calendar program is the classic Fortran program often referenced in computer history folklore (e.g., Real Programmers Don't Use Pascal). This version can also be run using the IBM 1130 emulator above.

Hello, world

In keeping with computing tradition, the first example presented is a simple program to display the words "Hello, world" on the screen (or printer).

FORTRAN 66

In FORTRAN 66 (also FORTRAN IV):

C     FORTRAN IV WAS ONE OF THE FIRST PROGRAMMING
C     LANGUAGES TO SUPPORT SOURCE COMMENTS
      WRITE (6,7)
    7 FORMAT(13H HELLO, WORLD)
      STOP
      END

This program prints "HELLO, WORLD" to Fortran unit number 6, which on most machines was the line printer or terminal. (The card reader or keyboard was usually connected as unit 5). The number 7 in the WRITE statement refers to the statement number of the corresponding FORMAT statement. FORMAT statements may be placed anywhere in the same program or function/subroutine block as the WRITE statements which reference them. Typically a FORMAT statement is placed immediately following the WRITE statement which invokes it; alternatively, FORMAT statements are grouped together at the end of the program or subprogram block. If execution flows into a FORMAT statement, it is a no-op; thus, the example above has only two executable statements, WRITE and STOP.

The initial 13H in the FORMAT statement in the above example defines a Hollerith constant, here meaning that the 13 characters immediately following are to be taken as a character constant (note that the Hollerith constant is not surrounded by delimiters). (Some compilers also supported character literals enclosed in single quotes, a practice that came to be standard with FORTRAN 77.)

The space immediately following the 13H is a carriage control character, telling the I/O system to advance to a new line on the output. A zero in this position advances two lines (double space), a 1 advances to the top of a new page and + character will not advance to a new line, allowing overprinting.

FORTRAN 77

As of FORTRAN 77, single quotes are used to delimit character literals, and inline character strings may be used instead of references to FORMAT statements. Comment lines may be indicated with either a C or an asterisk (*) in column 1.

      PROGRAM HELLOW
*     The PRINT statement is like WRITE,
*     but prints to the standard output unit
        PRINT '(A)', 'Hello, world'
        STOP
      END

Fortran 90

As of Fortran 90, double quotes are allowed in addition to single quotes. An updated version of the Hello, world example (which here makes use of list-directed I/O, supported as of FORTRAN 77) could be written in Fortran 90 as follows:

program HelloWorld
  write (*,*) 'Hello, world!'   ! This is an inline comment
end program HelloWorld

Greatest common divisor (FORTRAN 77)

The following introductory example in FORTRAN 77 finds the greatest common divisor for two numbers and using a verbatim implementation of Euclid's algorithm.

*     euclid.f (FORTRAN 77)
*     Find greatest common divisor using the Euclidean algorithm

      PROGRAM EUCLID
        PRINT *, 'A?'
        READ *, NA
        IF (NA.LE.0) THEN
          PRINT *, 'A must be a positive integer.'
          STOP
        END IF
        PRINT *, 'B?'
        READ *, NB
        IF (NB.LE.0) THEN
          PRINT *, 'B must be a positive integer.'
          STOP
        END IF
        PRINT *, 'The GCD of', NA, ' and', NB, ' is', NGCD(NA, NB), '.'
        STOP
      END

      FUNCTION NGCD(NA, NB)
        IA = NA
        IB = NB
    1   IF (IB.NE.0) THEN
          ITEMP = IA
          IA = IB
          IB = MOD(ITEMP, IB)
          GOTO 1
        END IF
        NGCD = IA
        RETURN
      END

The above example is intended to illustrate the following:

  • The PRINT and READ statements in the above use '*' as a format, specifying list-directed formatting. List-directed formatting instructs the compiler to make an educated guess about the required input or output format based on the following arguments.
  • As the earliest machines running Fortran had restricted character sets, FORTRAN 77 uses abbreviations such as .EQ., .NE., .LT., .GT., .LE., and .GE. to represent the relational operators =, ≠, <, >, ≤, and ≥, respectively.
  • This example relies on the implicit typing mechanism described previously to specify the INTEGER types of NA, NB, IA, IB, and ITEMP.
  • In the function NGCD(NA, NB), the values of the function arguments NA and NB are copied into the local variables IA and IB respectively. This is necessary as the values of IA and IB are altered within the function. Because argument passing in Fortran functions and subroutines utilize call by reference by default (rather than call by value, as is the default in languages such as C), modifying NA and NB from within the function would effectively have modified the corresponding actual arguments in the main PROGRAM unit which called the function.

The following shows the results of compiling and running the program.

mac:~/Desktop $ g77 -o euclid euclid.f
mac:~/Desktop $ euclid
 A?
24
 B?
36
 The GCD of 24 and 36 is 12.

Complex numbers (FORTRAN 77)

The following FORTRAN 77 example prints out the values of (where ) for values of .

*     cmplxd.f (FORTRAN 77)
*     Demonstration of COMPLEX numbers
*
*     Prints the values of e ** (j * i * pi / 4) for i = 0, 1, 2, ..., 7
*         where j is the imaginary number sqrt(-1)

      PROGRAM CMPLXD
        IMPLICIT COMPLEX(X)
        PARAMETER (PI = 3.141592653589793, XJ = (0, 1))
        DO 1, I = 0, 7
          X = EXP(XJ * I * PI / 4)
          IF (AIMAG(X).LT.0) THEN
            PRINT 2, 'e**(j*', I, '*pi/4) = ', REAL(X), ' - j',-AIMAG(X)
          ELSE
            PRINT 2, 'e**(j*', I, '*pi/4) = ', REAL(X), ' + j', AIMAG(X)
          END IF
    2     FORMAT (A, I1, A, F10.7, A, F9.7)
    1     CONTINUE
        STOP
      END

The above example is intended to illustrate the following:

  • The IMPLICIT statement can be used to specify the implicit type of variables based on their initial letter if different from the default implicit typing scheme described above. In this example, this statement specifies that the implicit type of variables beginning with the letter X shall be COMPLEX.
  • The PARAMETER statement may be used to specify constants. The second constant in this example (XJ) is given the complex-valued value , where is the imaginary unit .
  • The first number in the DO statement specifies the number of the last statement considered to be within the body of the DO loop. In this example, as neither the END IF nor the FORMAT is a single executable statement, the CONTINUE statement (which does nothing) is used simply in order for there to be some statement to denote as the final statement of the loop.
  • EXP() corresponds to the exponential function . In FORTRAN 77, this is a generic function, meaning that it accepts arguments of multiple types (such as REAL and, in this example, COMPLEX). In FORTRAN 66, a specific function would have to be called by name depending on the type of the function arguments (for this example, CEXP() for a COMPLEX-valued argument).
  • When applied to a COMPLEX-valued argument, REAL() and AIMAG() return the values of the argument's real and imaginary components, respectively.

Incidentally, the output of the above program is as follows (see the article on Euler's formula for the geometric interpretation of these values as eight points spaced evenly about a unit circle in the complex plane).

mac:~/Desktop $ cmplxd
e**(j*0*pi/4) =  1.0000000 + j0.0000000
e**(j*1*pi/4) =  0.7071068 + j0.7071068
e**(j*2*pi/4) =  0.0000000 + j1.0000000
e**(j*3*pi/4) = -0.7071068 + j0.7071068
e**(j*4*pi/4) = -1.0000000 - j0.0000001
e**(j*5*pi/4) = -0.7071066 - j0.7071069
e**(j*6*pi/4) =  0.0000000 - j1.0000000
e**(j*7*pi/4) =  0.7071070 - j0.7071065

Error can be seen occurring in the last decimal place in some of the numbers above, a result of the COMPLEX data type representing its real and imaginary components in single precision. Incidentally, Fortran 90 also made standard a double-precision complex-number data type (although several compilers provided such a type even earlier).

Calculating cylinder area (Fortran 90)

The following program, which calculates the surface area of a cylinder, illustrates free-form source input and other features introduced by Fortran 90.

program cylinder

! Calculate the surface area of a cylinder.
!
! Declare variables and constants.
! constants=pi
! variables=radius squared and height

  implicit none    ! Require all variables to be explicitly declared

  integer :: ierr
  character :: yn
  real :: radius, height, area
  real, parameter :: pi = 3.141592653589793

  interactive_loop: do

!   Prompt the user for radius and height
!   and read them.

    write (*,*) 'Enter radius and height.'
    read (*,*,iostat=ierr) radius,height

!   If radius and height could not be read from input,
!   then cycle through the loop.

    if (ierr /= 0) then
      write(*,*) 'Error, invalid input.'
      cycle interactive_loop
    end if

!   Compute area.  The ** means "raise to a power."

    area = 2 * pi * (radius**2 + radius*height)

!   Write the input variables (radius, height)
!   and output (area) to the screen.

    write (*,'(1x,a7,f6.2,5x,a7,f6.2,5x,a5,f6.2)') &
      'radius=',radius,'height=',height,'area=',area

    yn = ' '
    yn_loop: do
      write(*,*) 'Perform another calculation? y[n]'
      read(*,'(a1)') yn
      if (yn=='y' .or. yn=='Y') exit yn_loop
      if (yn=='n' .or. yn=='N' .or. yn==' ') exit interactive_loop
    end do yn_loop

  end do interactive_loop

end program cylinder

Dynamic memory allocation and arrays (Fortran 90)

The following program illustrates dynamic memory allocation and array-based operations, two features introduced with Fortran 90. Particularly noteworthy is the absence of DO loops and IF/THEN statements in manipulating the array; mathematical operations are applied to the array as a whole. Also apparent is the use of descriptive variable names and general code formatting that comport with contemporary programming style. This example computes an average over data entered interactively.

program average

! read in some numbers and take the average
! As written, if there are no data points (or no positive/negative points)
!     an average of zero is returned.
! While this may not be desired behavior, it keeps this example simple.

  implicit none
  integer :: NumberOfPoints
  real, dimension(:), allocatable :: Points
  real :: AveragePoints=0., PositiveAverage=0., NegativeAverage=0.

  write (*,*) 'Input number of points to average:'
  read (*,*) NumberOfPoints

  allocate (Points(NumberOfPoints))

  write (*,*) 'Enter the Points to average:'
  read (*,*) Points

! take the average by summing Points and dividing by NumberOfPoints
  if (NumberOfPoints>0) AveragePoints = sum(Points)/NumberOfPoints

! now form average over positive and negative points only
  if (count(Points>0.)>0) PositiveAverage = sum(Points, Points>0.)/count(Points>0.)
  if (count(Points<0.)>0) NegativeAverage = sum(Points, Points<0.)/count(Points<0.)

  deallocate (Points)

! print result to terminal
  write (*,'(''Average = '', 1g12.4)') AveragePoints
  write (*,'(''Average of positive points = '', 1g12.4)') PositiveAverage
  write (*,'(''Average of negative points = '', 1g12.4)') NegativeAverage

end program average

Writing functions (Fortran 90)

Modern Fortran features available for use with procedures, including deferred-shape, protected, and optional arguments, are illustrated in the following example, a function to solve a system of linear equations.

function GaussSparse(NumIter, Tol, b, A, X, ActualIter)

!  This function solves a system of equations (Ax = b) by using the Gauss-Jordan Method

   implicit none

   real GaussSparse

!  Input: its value cannot be modified from within the function
   integer, intent(in) :: NumIter
   real, intent(in) :: Tol
   real, intent(in), dimension(1:) :: b
   real, intent(in), dimension(1:,1:) :: A

!  Input/Output: its input value is used within the function, and can be modified
   real, intent(inout), dimension(1:) :: X

!  Output: its value is modified from within the function, only if the argument is required
   integer, optional, intent(out) :: ActualIter

!  Locals
   integer i, n, Iter
   real TolMax, Xk

!  Initialize values
   n = ubound(b, dim = 1)  ! Size of array, obtained using ubound intrinsic function
   TolMax = 2. * Tol
   Iter = 0

!  Compute solution until convergence
   convergence_loop: do while (TolMax >= Tol.AND.Iter < NumIter); Iter = Iter + 1

      TolMax = -1.  ! Reset the tolerance value

!     Compute solution for the k-th iteration
      iteration_loop: do i = 1, n

!        Compute the current x-value
         Xk = (b(i) - sum(A(i,1:i-1) * X(1:i-1)) - sum(A(i,i+1:n) * X(i+1:n))) / A(i, i)

!        Compute the error of the solution
         TolMax = max((abs(X(i) - Xk)/(1. + abs(Xk))) ** 2, abs(A(i, i) * (X(i) - Xk)), TolMax)
         X(i) = Xk
      enddo iteration_loop
   enddo convergence_loop

   if (present(ActualIter)) ActualIter = Iter
   GaussSparse = TolMax

end function GaussSparse

Note that an explicit interface to this routine must be available to its caller so that the type signature is known. This is preferably done by placing the function in a MODULE and then USEing the module in the calling routine. An alternative is to use an INTERFACE block.

Writing subroutines (Fortran 90)

In those cases where it is desired to return values via a procedure's arguments, a subroutine is preferred over a function; this is illustrated by the following subroutine to swap the contents of two arrays (this example assumes that the arrays are of equal size):

subroutine Swap_Real(a1, a2)

   implicit none

!  Input/Output
   real, intent(inout) :: a1(:), a2(:)

!  Locals
   integer :: lb(1), & !Lower bound
              ub(1)    !Upper bound
   integer i
   real a

!  Get bounds
   lb = lbound(a1)
   ub = ubound(a1)

!  Swap
   do i = lb(1), ub(1)
      a = a1(i)
      a1(i) = a2(i)
      a2(i) = a
   enddo

end subroutine Swap_Real

As in the previous example, an explicit interface to this routine must be available to its caller so that the type signature is known. As before, this is preferably done by placing the function in a MODULE and then USEing the module in the calling routine. An alternative is to use a INTERFACE block.

Pointers and targets (Fortran 90)

In Fortran, the concept of pointers differs from that in C-like languages. A Fortran 90 pointer does not merely store the memory address of a target variable; it also contains additional descriptive information such as the target's rank, the upper and lower bounds of each dimension, and even strides through memory. This allows a Fortran 90 pointer to point at submatrices.

Fortran 90 pointers are "associated" with well-defined "target" variables, via either the pointer assignment operator (=>) or an ALLOCATE statement. When appearing in expressions, pointers are always dereferenced; no "pointer arithmetic" is possible.

The following example illustrates the concept:

program Test

!  NOTE: Variable expressions in format statements (e.g., <m> or <n>)
!            are compiler-dependent.
!        Also, array notation (e.g., [1,2,3]) is a Fortran 2003 feature;
!            for Fortran 95, use (/1,2,3/) instead.

   use FunctionsModule, only: DoSomething => A !This function performs some
                                               !operation on the integer
                                               !input and returns its result
   implicit none

   integer, parameter :: m = 3, n = 3
   integer, pointer :: p(:)=>null(), q(:,:)=>null()
   integer, allocatable, target :: A(:,:)
   integer ios = 0

   allocate(A(1:m, 1:n), q(1:m, 1:n), stat = ios)
   if (ios /= 0) stop 'Error during allocation of A and q'

!  Assign the matrix
!  A = [[1   4   7]
!       [2   5   8]
!       [3   6   9]]
   A = reshape([(i, i = 1, m*n)], [m, n])
   q = A

!  p will be associated with the first column of A
   p => A(:, 1)

!  This operation on p has a direct effect on matrix A
   p = p ** 2

!  This will end the association between p and the first column of A
   nullify(p)

!  Matrix A becomes:
!  A = [[1  4  7  ]
!       [4  5  8  ]
!       [9  6  9  ]]
   write(*, '(/, "Matrix A becomes:",/,"A = [",<m>("[",<n>(i1,2x),"]",/,5x)"]")') &
         ((A(i, j), j = 1, n), i = 1, m)

!  Perform some array operation
   q = q + A

!  Matrix q becomes:
!  q = [[ 2   8  14  ]
!       [ 6  10  16  ]
!       [12  12  18  ]]
   write(*, '(/, "Matrix q becomes:",/,"q = [",<m>("[",<n>(i2,2x),"]",/,5x)"]")') &
         ((q(i, j), j = 1, n), i = 1, m)

!  Use p as an ordinary array
   allocate (p(1:m*n), stat = ios)
   if (ios /= 0) stop 'Error during allocation of p'

!  Perform some array operation
   p = [((DoSomething(A(i, j) + A(i, j) ** 2), i = 1, m), j = 1, n)]

   write(*, '(<m*n>(i1,4x,"p[",i1,"] = ",i5))') (i, p(i), i = 1, m * n)

   deallocate(A, p, q, stat = ios)
   if (ios /= 0) stop 'Error during deallocation'

end program Test

Modules (Fortran 90)

A module is a program unit which contains data definitions, global data, and CONTAINed procedures. Unlike a simple INCLUDE file, a module is an independent program unit that can be compiled separately and linked in its binary form. Once compiled, a module's public contents can be made visible to a calling routine via the USE statement.

The module mechanism makes the explicit interface of procedures easily available to calling routines. In fact, modern Fortran encourages every SUBROUTINE and FUNCTION to be CONTAINed in a MODULE. This allows the programmer to use the newer argument passing options, and allows the compiler to perform full type checking on the interface.

The following example also illustrates derived types and overloading of procedures and operators.

module GlobalModule

!  Reference to a pair of procedures included in a previously compiled
!  module named PortabilityLibrary
   use PortabilityLibrary, only: GetLastError, &  ! Generic procedure
                                 Date             ! Specific procedure
!  Constants
   integer, parameter :: dp_k = kind (1.0d0)      ! Double precision kind
   real, parameter :: zero = (0.)
   real(dp_k), parameter :: pi = 3.141592653589793_dp_k

!  Variables
   integer :: n, m, retint
   logical :: status, retlog
   character(50) :: AppName

!  Arrays
   real, allocatable, dimension(:,:,:) :: a, b, c, d
   complex(dp_k), allocatable, dimension(:) :: z

!  Derived type definitions
   type ijk
      integer i
      integer j
      integer k
   end type ijk

   type matrix
     integer m, n
     real, allocatable :: a(:,:)  ! Fortran 2003 feature. For Fortran 95,
                                  ! use the pointer attribute instead
   end type matrix

!  All the variables and procedures from this module can be accessed
!  by other program units, except for AppName
   public
   private AppName

!  Procedure overloading
   interface swap
      module procedure swap_Integer, swap_Real
   end interface swap

   interface GetLastError  ! This adds a new, additional procedure to the
                           ! generic procedure GetLastError
      module procedure GetLastError_GlobalModule
   end interface GetLastError

!  Operator overloading
   interface operator(+)
      module procedure add_ijk
   end interface

!  Prototype for external procedure
   interface
      real function GaussSparse(NumIter, Tol, b, A, X)
         integer, intent(in) :: NumIter
         real, intent(in) :: Tol
         real, intent(in), dimension(1:) :: b
         real, intent(in), dimension(1:,1:) :: A
         real, intent(inout), dimension(1:) :: X
      end function GaussSparse
   end interface

!  Procedures included in the module
   contains

!  Internal function
   function add_ijk(ijk_1, ijk_2)
     type(ijk) add_ijk, ijk_1, ijk_2
     intent(in) :: ijk_1, ijk_2
     add_ijk = ijk(ijk_1%i + ijk_2%i, ijk_1%j + ijk_2%j, ijk_1%k + ijk_2%k)
   end function add_ijk

!  Include external files
   include 'Swap_Integer.f90' ! Comments SHOULDN'T be added on include lines
   include 'Swap_Real.f90'
end module GlobalModule

FORTRAN jokes

For a programming language with a half-century legacy, FORTRAN not surprisingly has accumulated its share of jokes and folklore. See also Wikiquote/Programing/Programming Language/Fortran.

Standard jokes

  • "GOD is REAL unless declared INTEGER."
  • Joke, circa 1980 (following the standardization of FORTRAN 77): "Q: What will the scientific programming language of the year 2000 look like? ... A: Nobody knows, but its name will be FORTRAN."
  • A good FORTRAN programmer can write FORTRAN code in any language.

See also Wikiquote/Programming/Programming languages/Fortran.

From the historical record

  • "The primary purpose of the DATA statement is to give names to constants; instead of referring to pi as 3.141592653589793 at every appearance, the variable PI can be given that value with a DATA statement and used instead of the longer form of the constant. This also simplifies modifying the program, should the value of pi change." —Early FORTRAN manual for Xerox Computers[9]
  • "Consistently separating words by spaces became a general custom about the tenth century A. D., and lasted until about 1957, when FORTRAN abandoned the practice."Sun FORTRAN Reference Manual
  • "People are very flexible and learn to adjust to strange surroundings — they can become accustomed to read Lisp and Fortran programs, for example."Art of PROLOG, Leon Sterling and Ehud Shapiro, MIT Press
  • In the pilot episode of the Futurama series, the robot Bender drinks a bottle of "Olde FORTRAN Malt Liquor" (alluding to "Olde English" malt liquor)
  • in San Jose, California, there are streets named Fortran Drive and Fortran Court, located near another street, Disk Drive.

References

History

Textbooks

  • Chapman, Stephen J. (2003). Fortran 90/95 for Scientists and Engineers (2nd ed. ed.). McGraw-Hill. ISBN 0072825758. {{cite book}}: |edition= has extra text (help)
  • Ellis, T. M. R. (1994). Fortran 90 Programming (1st ed. ed.). Addison-Wesley. ISBN 0201544466. {{cite book}}: |edition= has extra text (help); Unknown parameter |coauthors= ignored (|author= suggested) (help)
  • Kaufman, Roger Emanuel (1978). A FORTRAN Coloring Book]. MIT. ISBN 0262610264. {{cite book}}: External link in |last= (help)CS1 maint: numeric names: authors list (link) "The FORTRAN Coloring Book gave rise to a flood of other books done in the style it created." Kaufman
  • McCracken, Daniel D. (1961). A Guide to Fortran Programming. Wiley. {{cite book}}: External link in |last= (help)
  • McCracken, Daniel D. (1965). A Guide to Fortran IV Programming. Wiley. {{cite book}}: External link in |last= (help)
  • Metcalf, Michael (2004). Fortran 95/2003 Explained. Oxford University Press. {{cite book}}: Unknown parameter |coauthors= ignored (|author= suggested) (help)
  • Nyhoff, Larry (1995). FORTRAN 77 for Engineers and Scientists with an Introduction to Fortran 90 (4th ed. ed.). Prentice Hall. ISBN 013363003X. {{cite book}}: |edition= has extra text (help); Unknown parameter |coauthors= ignored (|author= suggested) (help)

"Core" language standards

  • ECMA 1965. ECMA Standard on FORTRAN (ECMA-9). European Computer Manufacturers Association.{{cite book}}: CS1 maint: numeric names: authors list (link)
  • ASA X3.9-1966. American Standard FORTRAN (PDF). American Standards Association.{{cite book}}: CS1 maint: numeric names: authors list (link) Informally known as FORTRAN 66.
  • ASA X3.10-1966. American Standard Basic FORTRAN. American Standards Association.{{cite book}}: CS1 maint: numeric names: authors list (link)
  • ISO 1539-1972. Recommendation: Programming Language FORTRAN. International Organization for Standardization.{{cite book}}: CS1 maint: numeric names: authors list (link)
  • ANSI X3.9-1978. American National Standard — Programming Language FORTRAN (PDF). American National Standards Institute.{{cite book}}: CS1 maint: numeric names: authors list (link) Also known as ISO 1539-1980, and informally known as FORTRAN 77.
  • MIL-STD-1753 (November 9, 1978). Fortran, DoD Supplement to American National Standard X3.9-1978. U. S. Government Printing Office.{{cite book}}: CS1 maint: numeric names: authors list (link)
  • ANSI X3.198-1992 (R1997). American National Standard — Programming Language Fortran Extended. American National Standards Institute.{{cite book}}: CS1 maint: numeric names: authors list (link) Informally known as Fortran 90.
  • ISO/IEC 1539-1:1997. Information technology — Programming languages — Fortran — Part 1: Base language.{{cite book}}: CS1 maint: numeric names: authors list (link) Informally known as Fortran 95. There are a further two parts to this standard. Part 1 has been formally adopted by ANSI.
  • ISO/IEC 1539-1:2004. Information technology — Programming languages — Fortran — Part 1: Base language.{{cite book}}: CS1 maint: numeric names: authors list (link) Informally known as Fortran 2003.

Notes

  1. ^ Since FORTRAN 77, which introduced the CHARACTER data type.
  2. ^ Since FORTRAN II (1958).
  3. ^ Because the capitalization (or lack thereof) of the word FORTRAN was never 100% consistent in actual usage, and because many hold impassioned beliefs on the issue, this article, rather than attempt to be normative, adopts the convention of using the all-caps FORTRAN in referring to versions of FORTRAN through FORTRAN 77 and the title-caps Fortran in referring to versions of Fortran from Fortran 90 onward. This convention is reflected in the capitalization of FORTRAN in the ANSI X3.9-1966 (FORTRAN 66) and ANSI X3.9-1978 (FORTRAN 77) standards and the title caps Fortran in the ANSI X3.198-1992 (Fortran 90) standard.
  4. ^ "The [FORTRAN] project was initially intended purely for use by IBM on a single computer." Ellis, T. M. R. (1982). A Structured Approach to FORTRAN 77 Programming. Addison-Wesley. Page 8.
  5. ^ With FORTRAN 77, it became standard practice to insert a comma between the first statement number and the DO variable; however, this comma could be omitted for compatibility with FORTRAN 66.
  6. ^ Note that in the case of a REAL-valued expression, the usual caveat regarding tests for floating-point equality applies, i.e., an expression mathematically expected to evaluate to zero may not exactly equal zero due to round-off error and the limited internal precision of floating-point arithmetic.
  7. ^ "GO TO (assigned)" page from the IBM XL Fortran Advanced Edition for Mac OS X Version 8.1 online documentation.
  8. ^ This is not altogether surprising, as Brian Kernighan, one of the co-creators of Ratfor, is also co-author of The C Programming Language.
  9. ^ Prior to FORTRAN 77, which introduced the PARAMETER statement (the preferred method of defining constants), the DATA statement was the conventional way to define constants, although this statement merely specified initial values and did not prevent the corresponding variables from being modified (as by regular assignment statements).

History

Tutorials

References

Code repositories

Free (open-source) compilers

  • g77: "g77 is a free Fortran 77 compiler...now integrated into the GNU Compiler Collection (GCC)." Note: GCC, as of version 4.0, now includes GFortran as the default Fortran compiler in lieu of g77.
  • g95: "The goal of the g95 project is to create a free, open source Fortran 95 compiler and runtime libraries." Note: Once an open, collaborative effort, the project has now reverted to the exclusive control of the original g95 project creator.
  • GFortran: "The GNU Fortran 95 compiler, part of GCC." Distributed as part of GCC as of GCC 4.0, GFortran is likely to become the de facto "standard" Fortran compiler (much as g77 was through GCC 3.4.6).
  • High Performance Computing for Mac OS X — Fortran compilers for Mac OS X.
  • Open Watcom: "A joint effort between SciTech Software Inc, Sybase and the Open Source development community to maintain and enhance the Watcom C/C++ and Fortran cross compilers and tools."

Proprietary compilers

Free graphical libraries/GUI

Proprietary/non-free graphical libraries/GUI

  • DISLIN: "DISLIN is a high-level plotting library for displaying data as curves, polar plots, bar graphs, pie charts, 3D-color plots, surfaces, contours and maps." For DOS, Linux, Mac OS X, OpenVMS, UNIX, Windows. "Free for non-commercial use" (but not free).
  • SansGUI: Requires Microsoft Visual C/C++ 6.0+ or Compaq Visual Fortran 6.1+. For Windows.
  • Winteracter: "The Fortran 9x GUI Toolset." For Linux, Mac OS X, Windows.