zoukankan      html  css  js  c++  java
  • Fortran 文件操作

    1)--------------------------------------------------------------------------------------
    FOR的输出分 有格式form = 'formatted'、无格式form = 'unformatted'两种,前者是默认输出格式,即如果open语句里不声明form的话,那就是formatted。
    无格式又分 直接存取access = 'direct'、间接存取access = 'seuqential' 两种,后者是无格式文件的默认格式。
    几个open示例:
    有格式文件: 
        open( 1, file = '...', status = 'new', form = 'formatted' )
        或者
        open( 1, file = '...', status = 'old')
    无格式/顺序文件: 
        open( 1, file = '...', status = 'new', form = 'unformatted', access = 'sequential' )
        或者
        open( 1, file = '...', status = 'new', form = 'unformatted' )
    无格式/直接文件:
        open( 1, file = '...', status = 'new', form = 'unformatted', access = 'direct', recl = ... )


    2)--------------------------------------------------------------------------------------
    如果是向 有格式文件 输出,则write语句可以写成:表控格式,如write( 1,* ) var; 或者显式说明格式,如write( 1,'(5i2)' ) var
    如果是向 无格式/顺序文件 输出,则write语句应写成: write( 1 ) var
    如果是向 无格式/直接文件 输出,则write语句应写成: write( 1, rec = k ) var


    3)--------------------------------------------------------------------------------------
    对于 无格式/直接文件 , open中的recl指明了一个输出记录的长度。它具体等于多少要看输出write语句如何写,同时write语句中的记录号rec也要写正确。
    比如有一个m(20,30)的二维数组,可以有如下几种写法(注意recl, rec, m的变化):
    a) 把整个数组看作一个记录(1个记录,1个记录长度20*30):
    open ( 1, file = '...', status = 'new', form = 'unformatted', access = 'direct', recl = 20*30 )
    write( 1, rec = 1 ) m
    b) 把一行看作一个记录(20个记录,1个记录长度30):
    open ( 1, file = '...', status = 'new', form = 'unformatted', access = 'direct', recl = 30 )
    do k = 1, 20
      write( 1, rec = k ) ( m( k, p ), p = 1, 30 )
    end do
    c) 把一个数组元素看作一个记录(20*30个记录,1个记录长度1):
    open ( 1, file = '...', status = 'new', form = 'unformatted', access = 'direct', recl = 1 )
    k = 0
    do i = 1, 20
       do j = 1, 30
          k = k + 1
          write( 1, rec = k ) m( i, j )
    或者
          write( 1, rec = (i-1)*30+j ) m( i, j )
       end do
    end do

    顺说一句,二维数组是先存列再存行的,所以上述a)打印m元素的顺序与c)不同,而与下述d)相同
    d) 把一个数组元素看作一个记录:
    open ( 1, file = '...', status = 'new', form = 'unformatted', access = 'direct', recl = 1 )
    k = 0
    do j = 1, 30
       do i = 1, 20
          k = k + 1
          write( 1, rec = k ) m( i, j )
    或者
          write( 1, rec = (j-1)*20+i ) m( i, j )
       end do
    end do

    在一些旧的FOR编译器中(比如Fortran PowerStation 4.0),对recl解释成字长而非字节,因此还需要将recl乘以4才能正确读取。
    PS:你的33个数据可能并没有分放在3行半上。可能只是记事本查看的原因,当一屏显示不下时自动换行显示,但数据可能还是在一行上。所以要读取的话可试试:
    write( 1, * ) ( m( k ), k = 1, 33 )
    如果m一开始就声明成一个只有33个元素的数组,则也可写成
    write( 1, * ) m

    4)--------------------------------------------------------------------------------------
    二进制存放的数据还有一个反位的问题。一般在UNIX机上是二进制数是高位结束,在WIN和LINUX中是低位结束(没记错的话)。如果读数时出现很大的怪数,比如1.e+35之类,或者执行时显示读取文件到末尾,或者是记录不够长,并且你又确定程序没错,那么多半是反位问题的原因。
    这时如果你用的编译器有反位选项,可以加上,比如ifort编译器可以写:
    ifort -convert big_endian xxx.f90  或
    ifort -convert little_endian xxx.f90
    也可以直接在FOR程序的OPEN语句中说明,比如:
    open( 1, file='...', form='unformatted', access='direct', recl=..., convert='big_endian' )

    open( 1, file='...', form='unformatted', access='direct', recl=..., convert='little_endian' )
    big_endian 适用于在本地PC读取UNIX服务器上生成的二进制数据。
    little_endian适用于在UNIX服务器读取本地PC生成的二进制数据。

  • 相关阅读:
    计算机网络原理 课后题7
    计算机网络原理 课后题6
    计算机网络原理 课后题5
    计算机网络原理 课后题4
    计算机网络原理 课后题3
    计算机网络原理 课后题2
    计算机网络原理 课后题1
    C#学习——条件判断(三)
    C#学习——循环结构(二)
    python3.7版本安装Ride
  • 原文地址:https://www.cnblogs.com/pankejia/p/4014918.html
Copyright © 2011-2022 走看看