summaryrefslogtreecommitdiffhomepage
path: root/zh-cn/fortran95-cn.html.markdown
blob: bd77906c7a6cf155ff9f285ea3d2a3e4bec214af (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
---
language: Fortran
filename: learnfortran-cn.f95
contributors:
    - ["Robert Steed", "https://github.com/robochat"]
translators:
    - ["Corvusnest", "https://github.com/Corvusnest"]
lang: zh-cn
---

Fortran 是最古老的计算机语言之一。它由IBM开发于1950年用于数值运算(Fortran 为 "Formula
Translation" 的缩写)。虽然该语言的开发时间距离现在比较遥远,但目前仍用于高性能计算,例如
天气预报。该语言仍在持续发展,并且基本保持向下兼容。知名的版本为 FORTRAN 77, Fortran 90,
Fortran 95, Fortran 2003, Fortran 2008 与 Fortran 2015。

这篇概要将讨论 Fortran 95 的一些特征。因为它是目前所广泛采用的标准版本,并且与最新版本的内容
也基本相同(而 FORTRAN 77 则是一个非常不同的版本)。

```fortran

! 这是一行注释


program example   !声明一个叫做 example 的程序

    ! 代码只能放在程序、函数、子程序或者模块内部
    ! 推荐使用缩进,但不是必须的。

    ! 声明变量
    ! ===================

    ! 所有的声明必须放在语句与表达式之前

    implicit none    !阻止变量的隐式声明 (推荐!)
    ! Implicit none 必须在每一个 函数/程序/模块 中进行声明

    ! 重要  - Fortran 对大小写敏感
    real z
    REAL Z2

    real :: v,x    ! 警告: 默认值取决于编译器!    
    real :: a = 3, b=2E12, c = 0.01
    integer :: i, j, k=1, m
    real, parameter :: PI = 3.1415926535897931    !声明一个常量
    logical :: y = .TRUE. , n = .FALSE.    !布尔值
    complex :: w = (0,1)    !sqrt(-1) (译注: 定义复数,此为-1的平方根)
    character (len=3) :: month    !长度为3的字符串

    real :: array(6)     !声明长度为6的浮点数数组
    real, dimension(4) :: arrayb    !声明数组的另一种方法
    integer :: arrayc(-10:10)   !有着自定义索引的数组
    real :: array2d(3,2)    !多维数组

    ! 分隔符 '::' 并不总是必要的,但推荐使用

    ! many other variable attributes also exist:
    real, pointer :: p    !声明一个指针

    integer, parameter :: LP = selected_real_kind(20)
    real (kind = LP) :: d    !长精度变量

    ! 警告:在声明期间初始化变量将导致在函数内发生问题,因为这将自动具备了 “save” 属性,
    ! 因此变量的值在函数的多次调用期间将被存储。一般来说,除了常量,应分开声明与初始化!

    ! 字符串
    ! =======

    character :: a_char = 'i'
    character (len = 6) :: a_str = "qwerty"
    character (len = 30) :: str_b
    character (len = *), parameter :: a_long_str = "This is a long string."
    !可以通过使用 (len=*) 来自动判断长度,但只对常量有效

    str_b = a_str // " keyboard"    !通过 // 操作符来连接字符串


    ! 任务与计算
    ! =======================

    Z = 1    !向之前声明的变量 z 赋值 (大小写不敏感).
    j = 10 + 2 - 3
    a = 11.54  /  (2.3 * 3.1)
    b = 2**3    !幂


    ! 控制流程语句 与 操作符
    ! ===================================

    !单行 if 语句
    if (z == a) b = 4  !判别句永远需要放在圆括号内

    if (z /= a) then !z 不等于 a
    ! 其他的比较运算符: < > <= >= == /=
      b = 4
    else if (z .GT. a) then !z 大于(Greater) a
    ! 文本形式的比较运算符: .LT. .GT. .LE. .GE. .EQ. .NE.  
      b = 6
    else if (z < a) then !'then' 必须放在该行
      b = 5 !执行部分必须放在新的一行里
    else
      b = 10
    end if !结束语句需要 'if' (也可以用 'endif').


    if (.NOT. (x < c .AND. v >= a .OR. z == z)) then   !布尔操作符
      inner: if (.TRUE.) then    !可以为 if 结构命名
        b = 1
      endif inner    !接下来必须命名 endif 语句.
    endif


    i = 20
    select case (i)
      case (0)    !当 i == 0
        j=0
      case (1:10)    !当 i 为 1 到 10 之内 ( 1 <= i <= 10 )
        j=1
      case (11:)    !当 i>=11
        j=2
      case default
        j=3
    end select


    month = 'jan'
    ! 状态值可以为整数、布尔值或者字符类型
    ! Select 结构同样可以被命名
    monthly: select case (month)
      case ("jan")
         j = 0
      case default
         j = -1
    end select monthly

    do i=2,10,2    !从2到10(包含2和10)以2为步进值循环
      innerloop: do j=1,3    !循环同样可以被命名
        exit    !跳出循环
      end do innerloop
    cycle    !重复跳入下一次循环
    enddo


    ! Goto 语句是存在的,但强烈不建议使用
    goto 10    
    stop 1    !立即停止程序 (返回一个设定的状态码).
10  j = 201    !这一行被标注为 10 行 (line 10)


    ! 数组
    ! ======
    array = (/1,2,3,4,5,6/)
    array = [1,2,3,4,5,6]    !当使用 Fortran 2003 版本.
    arrayb = [10.2,3e3,0.41,4e-5]
    array2d =  reshape([1.0,2.0,3.0,4.0,5.0,6.0], [3,2])

    ! Fortran 数组索引起始于 1
    ! (默认下如此,也可以为数组定义不同的索引起始)
    v = array(1)    !获取数组的第一个元素
    v = array2d(2,2)

    print *, array(3:5)    !打印从第3到第五5之内的所有元素
    print *, array2d(1,:)    !打印2维数组的第一列

    array = array*3 + 2    !可为数组设置数学表达式
    array = array*array    !数组操作支持元素级(操作) (element-wise)
    !array = array*array2d    !这两类数组并不是同一个维度的

    ! 有很多内置的数组操作函数
    c = dot_product(array,array)    !点乘 (点积)
    ! 用 matmul() 来进行矩阵运算.
    c = sum(array)
    c = maxval(array)
    print *, minloc(array)
    c = size(array)
    print *, shape(array)
    m = count(array > 0)

    ! 遍历一个数组 (一般使用 Product() 函数).
    v = 1
    do i = 1, size(array)
        v = v*array(i)
    end do

    ! 有条件地执行元素级操作
    array = [1,2,3,4,5,6]
    where (array > 3)
        array = array + 1
    elsewhere (array == 2)
        array = 1
    elsewhere
        array = 0
    end where

    ! 隐式DO循环可以很方便地创建数组
    array = [ (i, i = 1,6) ]    !创建数组 [1,2,3,4,5,6]
    array = [ (i, i = 1,12,2) ]    !创建数组 [1,3,5,7,9,11]
    array = [ (i**2, i = 1,6) ]    !创建数组  [1,4,9,16,25,36]
    array = [ (4,5, i = 1,3) ]    !创建数组 [4,5,4,5,4,5]


    ! 输入/输出
    ! ============

    print *, b    !向命令行打印变量 'b'

    ! 我们可以格式化输出
    print "(I6)", 320    !显示 '   320'
    print "(I6.4)", 3    !显示 '  0003'
    print "(F6.3)", 4.32    !显示 ' 4.320'


    ! 该字母与数值规定了给定的数值与字符所用于打印输出的类型与格式
    ! 字母可为 I (整数), F (浮点数), E (工程格式),
    ! L (逻辑/布尔值), A (字符) ...
    print "(I3)", 3200    !如果数值无法符合格式将打印 '***'

    ! 可以同时设定多种格式
    print "(I5,F6.2,E6.2)", 120, 43.41, 43.41
    print "(3I5)", 10, 20, 30    !3 repeats of integers (field width = 5).
    print "(2(I5,F6.2))", 120, 43.42, 340, 65.3   !repeated grouping of formats.

    ! 我们也可以从终端读取输入
    read *, v
    read "(2F6.2)", v, x    !读取2个数值

    ! 读取文件
    open(unit=11, file="records.txt", status="old")
    ! 文件被引用带有一个单位数 'unit', 为一个取值范围在9-99的整数
    ! 'status' 可以为 {'old','replace','new'} 其中之一
    read(unit=11, fmt="(3F10.2)") a, b, c
    close(11)

    ! 写入一个文件
    open(unit=12, file="records.txt", status="replace")
    write(12, "(F10.2,F10.2,F10.2)") c, b, a
    close(12)
    ! 在讨论范围之外的还有更多的细节与可用功能,并于老版本的 Fortran 保持兼容


    ! 内置函数
    ! ==================

    ! Fortran 拥有大约 200 个内置函数/子程序
    ! 例子
    call cpu_time(v)    !以秒为单位设置时间
    k = ior(i,j)    !2个整数的位或运算
    v = log10(x)    !以10为底的log运算
    i = floor(b)    !返回一个最接近的整数小于或等于x (地板数)
    v = aimag(w)    !复数的虚数部分


    ! 函数与子程序
    ! =======================

    ! 一个子程序会根据输入值运行一些代码并会导致副作用 (side-effects) 或修改输入值
    ! (译者注: 副作用是指对子程序/函数外的环境产生影响,如修改变量)

    call routine(a,c,v)    !调用子程序

    ! 一个函数会根据输入的一系列数值来返回一个单独的值
    ! 但输入值仍然可能被修改以及产生副作用

    m = func(3,2,k)  !调用函数

    ! 函数可以在表达式内被调用
    Print *, func2(3,2,k)

    ! 一个纯函数不会去修改输入值或产生副作用
    m = func3(3,2,k)


contains ! 用于定义程序内部的副程序(sub-programs)的区域

    ! Fortran 拥有一些不同的方法去定义函数

    integer function func(a,b,c)    !一个返回一个整数的函数
        implicit none   !最好也在函数内将含蓄模式关闭 (implicit none)
        integer :: a,b,c !输入值类型定义在函数内部
        if (a >= 2) then
            func = a + b + c !返回值默认为函数名
            return !可以在函数内任意时间返回当前值
        endif
        func = a + c
        ! 在函数的结尾不需要返回语句
    end function func


    function func2(a,b,c) result(f)    !将返回值声明为 'f'
        implicit none
        integer, intent(in) :: a,b    !可以声明让变量无法被函数修改
        integer, intent(inout) :: c
        integer :: f     !函数的返回值类型在函数内声明
        integer :: cnt = 0    !注意 - 隐式的初始化变量将在函数的多次调用间被存储
        f = a + b - c
        c = 4    !变动一个输入变量的值
        cnt  = cnt + 1    !记录函数的被调用次数
    end function func2


    pure function func3(a,b,c)  !一个没有副作用的纯函数
        implicit none
        integer, intent(in) :: a,b,c
        integer :: func3
        func3 = a*b*c
    end function func3


    subroutine routine(d,e,f)
        implicit none
        real, intent(inout) :: f
        real, intent(in) :: d,e
        f = 2*d + 3*e + f
    end subroutine routine


end program example   ! 函数定义完毕 -----------------------

! 函数与子程序的外部声明对于生成程序清单来说,需要一个接口声明(即使它们在同一个源文件内)(见下)
! 使用 'contains' 可以很容易地在模块或程序内定义它们

elemental real function func4(a) result(res)
! 一个元函数(elemental function) 为一个纯函数使用一个标量输入值
! 但同时也可以用在一个数组并对其中的元素分别处理,之后返回一个新的数组
    real, intent(in) :: a
    res = a**2 + 1.0
end function func4


! 模块
! =======

! 模块十分适合于存放与复用相关联的一组声明、函数与子程序

module fruit
    real :: apple
    real :: pear
    real :: orange
end module fruit


module fruity

    ! 声明必须按照顺序: 模块、接口、变量
    ! (同样可在程序内声明模块和接口)

    use fruit, only: apple, pear    ! 使用来自于 fruit 模块的 apple 和 pear
    implicit none    !在模块导入后声明

    private    !使得模块内容为私有(private)(默认为公共 public)
    ! 显式声明一些变量/函数为公共
    public :: apple,mycar,create_mycar
    ! 声明一些变量/函数为私有(在当前情况下没必要)(译注: 因为前面声明了模块全局 private)
    private :: func4

    ! 接口
    ! ==========
    ! 在模块内显式声明一个外部函数/程序
    ! 一般最好将函数/程序放进 'contains' 部分内
    interface
        elemental real function func4(a) result(res)
            real, intent(in) :: a
        end function func4
    end interface

    ! 重载函数可以通过已命名的接口来定义
    interface myabs
        ! 可以通过使用 'module procedure' 关键词来包含一个已在模块内定义的函数
        module procedure real_abs, complex_abs
    end interface

    ! 派生数据类型
    ! ==================
    ! 可创建自定义数据结构
    type car
        character (len=100) :: model
        real :: weight    !(公斤 kg)
        real :: dimensions(3)    !例: 长宽高(米)
        character :: colour
    end type car

    type(car) :: mycar    !声明一个自定义类型的变量
    ! 用法具体查看 create_mycar()

    ! 注: 模块内没有可执行的语句

contains

    subroutine create_mycar(mycar)
        ! 展示派生数据类型的使用
        implicit none
        type(car),intent(out) :: mycar

        ! 通过 '%' 操作符来访问(派生数据)类型的元素
        mycar%model = "Ford Prefect"
        mycar%colour = 'r'
        mycar%weight = 1400
        mycar%dimensions(1) = 5.0    !索引默认起始值为 1 !
        mycar%dimensions(2) = 3.0
        mycar%dimensions(3) = 1.5

    end subroutine

    real function real_abs(x)
        real :: x
        if (x<0) then
            real_abs = -x
        else
            real_abs = x
        end if
    end function real_abs

    real function complex_abs(z)
        complex :: z
        ! 过长的一行代码可通过延续符 '&' 来换行
        complex_abs = sqrt(real(z)**2 + &
                                         aimag(z)**2)
    end function complex_abs


end module fruity

```

### 更多资源

了解更多的 Fortran 信息:

+ [wikipedia](https://en.wikipedia.org/wiki/Fortran)
+ [Fortran_95_language_features](https://en.wikipedia.org/wiki/Fortran_95_language_features)
+ [fortranwiki.org](http://fortranwiki.org)
+ [www.fortran90.org/](http://www.fortran90.org)
+ [list of Fortran 95 tutorials](http://www.dmoz.org/Computers/Programming/Languages/Fortran/FAQs%2C_Help%2C_and_Tutorials/Fortran_90_and_95/)
+ [Fortran wikibook](https://en.wikibooks.org/wiki/Fortran)
+ [Fortran resources](http://www.fortranplus.co.uk/resources/fortran_resources.pdf)
+ [Mistakes in Fortran 90 Programs That Might Surprise You](http://www.cs.rpi.edu/~szymansk/OOF90/bugs.html)