模块和程序#

模块(module)是创建现代Fortran库和应用程序的首选方法。 按照惯例,一个源文件应始终只包含一个模块,而模块名称应与文件路径匹配,以便在较大的项目中可以轻松浏览。 此外还建议在模块名前加上库名,以避免在其它项目中作为依赖项使用时发生名称冲突。

这里给出了一个模块文件的示例

!> Interface to TOML processing library.
!>
!> ...
module fpm_toml
  use fpm_error, only : error_t, fatal_error, file_not_found_error
  use fpm_strings, only : string_t
  use tomlf, only : toml_table, toml_array, toml_key, toml_stat, get_value, &
    & set_value, toml_parse, toml_error, new_table, add_table, add_array, &
    & toml_serializer, len
  implicit none
  private

  public :: read_package_file
  public :: toml_table, toml_array, toml_key, toml_stat, get_value, set_value
  public :: new_table, add_table, add_array, len
  public :: toml_error, toml_serializer, toml_parse

contains

  !> Process the configuration file to a TOML data structure
  subroutine read_package_file(table, manifest, error)
    !> TOML data structure
    type(toml_table), allocatable, intent(out) :: table
    !> Name of the package configuration file
    character(len=*), intent(in) :: manifest
    !> Error status of the operation
    type(error_t), allocatable, intent(out) :: error
    ! ...
  end subroutine read_package_file

end module fpm_toml

本示例模块中有几点需要强调。 首先,每个模块都以注释开始,记录模块的目的和内容。 类似的,每个过程都以一条注释开始,简要描述其目的和虚参的意义。 无论使用何种语言,文档都是创建长期软件的最重要部分之一。

其次,明确给出了导入 (use) 和导出 (public),这允许查看模块源代码以检查已使用和可用的过程、常量和派生类型。导入通常仅限于模块范围,而不是在每个过程或接口范围内重新导入。类似地,导出是通过在单行上添加 private 语句并在 public 语句中显式列出所有导出的符号来显式进行的。

最后,implicit none 语句适用于整个模块,无需在每个过程中重复。

模块内的变量是静态的(隐式保存)。强烈建议将模块变量的使用限制为常量表达式,例如参数或枚举器,或者将它们导出为 protected 而不是 public

在 Fortran 程序中,子模块(submodule)可以用来打破长依赖链,缩短重新编译的时间。 它们使得我们在不使用预处理器的情况下,提供专用化和经过优化的程序实现。

如下示例是Fortran标准库中的积分模块, 此处只定义了模块过程的接口,而没有具体实现

!> Numerical integration
!>
!> ...
module stdlib_quadrature
  use stdlib_kinds, only: sp, dp, qp
  implicit none
  private

  public :: trapz
  ! ...

  !> Integrates sampled values using trapezoidal rule
  interface trapz
    pure module function trapz_dx_dp(y, dx) result(integral)
      real(dp), intent(in) :: y(:)
      real(dp), intent(in) :: dx
      real(dp) :: integral
    end function trapz_dx_dp
    module function trapz_x_dp(y, x) result(integral)
      real(dp), intent(in) :: y(:)
      real(dp), intent(in) :: x(:)
      real(dp) :: integral
    end function trapz_x_dp
  end interface trapz

  ! ...
end module stdlib_quadrature

具体实现是在单独的子模块中提供的,比如这里给出的梯形积分规则。

!> Actual implementation of the trapezoidal integration rule
!>
!> ...
submodule (stdlib_quadrature) stdlib_quadrature_trapz
  use stdlib_error, only: check
  implicit none

contains

  pure module function trapz_dx_dp(y, dx) result(integral)
    real(dp), intent(in) :: y(:)
    real(dp), intent(in) :: dx
    real(dp) :: integral
    integer :: n

    n = size(y)
    select case (n)
    case (0:1)
      integral = 0.0_dp
    case (2)
      integral = 0.5_dp*dx*(y(1) + y(2))
    case default
      integral = dx*(sum(y(2:n-1)) + 0.5_dp*(y(1) + y(n)))
    end select
  end function trapz_dx_dp

  ! ...
end submodule stdlib_quadrature_trapz

请注意,模块过程不必在同一个子模块中实现,可以使用多个子模块来减少大型模块的编译负载。

最后,在设置程序时,建议将程序主体中的实际实现保持在最低限度。重用模块中的实现允许你编写可重用的代码并将程序单元集中在将用户输入传送到相应的库函数和对象上。