Fortran Coder

查看: 26833|回复: 22
打印 上一主题 下一主题

[文件读写] 关于二进制文件的读入问题?

[复制链接]

28

帖子

4

主题

0

精华

熟手

F 币
134 元
贡献
72 点
跳转到指定楼层
楼主
发表于 2020-5-15 22:14:11 | 只看该作者 |只看大图 回帖奖励 |正序浏览 |阅读模式
      纯新人一枚,由于科研需要,刚学77不久,在UBUNTU下解析一个由F77写的远古代码,我用CODEBLOCK移植到了WIN平台下,在跟踪代码过程中,一个子程序中读取了一个二进制文件,关于这里READ的用法,我没太搞清楚,希望得到指点。      读入二进制文件是这么写的,如图1所示。关键部分:
   OPEN(UNIT=IOPDS,FILE=FILNAM(1:LNFILE),ERR=100, STATUS='UNKNOWN',
     &     ACCESS='SEQUENTIAL', FORM='UNFORMATTED',IOSTAT=IOS)
                  ENDIF
       READ(UNIT=IOPDS,ERR=300,IOSTAT=IOS)  LNOW,(WORK(I),I=1,LNOW)    这个read后面的LNOW是什么类型,因为前面也没有显示声明。。?因为WORK在前面已经申明是WORK(1),那么这里是从二进制文件FILE(INQUIRE存在)中读入全部数据后放到LNOW中,在赋值到WORK(1)中吗? 最重要的是我设断点跟踪的过程中,发现执行完READ后,LNOW等于0(执行前为2000多),WORK为0,参见图2,这是说明二进制文件中的所有数据没有读入进去吗??还是我理解有问题,如果读进去了,为啥我用IDE的WATCH栏看到的是0。
      看了论坛上的帖子http://fcode.cn/guide-4-1.html帮助很大,里面是分行读入,我碰到的情况似乎和帖子所讲有出入,这个程序是用一个LNOW把二进制的所有数据都读入了吗?
       这个地方的理解,对我后面阅读学习这个程序非常关键,今天整了一天,还是没整明白,明明文件存在,也OPEN了,读进去为啥LNOW是0,不知道我哪里没理解对这个程序的意思. 另外这些代码在UBUNTU下编译跑例子都正常,也就是说这块代码是没问题的。只是放在WIN下我想用CODEBLOCK能够快速阅读,方便快速理解,应该也不会是平台的差异导致的。谢谢大家的帮助和指点。


       另外我单独把这个读取部分拿出来用77模仿了下,其中FILENAME具体化,执行前后LNOW和WORK皆为0.
       PROGRAM READTXT
       LOGICAL EX
       DIMENSION WORK(1)
      INQUIRE(FILE='FASTLIB',EXIST=EX)
      IF(.NOT.EX) THEN
                  WRITE (*,*) "Hello World!"
                  RETURN
                  ELSE   
     OPEN(UNIT=66,FILE='FASTLIB',
     &     STATUS='UNKNOWN',
     &     ACCESS='SEQUENTIAL', FORM='UNFORMATTED', IOSTAT=IOS)
                  ENDIF   
     READ (66,IOSTAT=IOS)   LNOW,(WORK(I),I=1,LNOW)


FILE文件的二进制和十进制文件我用这个远古程序附带的小工具在LINUX下转换出来了,也一起上传上来。




原程序.png (57.15 KB, 下载次数: 415)

原程序.png

2.png (104.42 KB, 下载次数: 396)

2.png

十进制文件.txt

2.07 KB, 下载次数: 5

二进制文件.txt

632 Bytes, 下载次数: 7

读取的名为FASTLIB文件

分享到:  微信微信
收藏收藏 点赞点赞 点踩点踩

28

帖子

4

主题

0

精华

熟手

F 币
134 元
贡献
72 点
23#
 楼主| 发表于 2020-5-19 10:21:11 | 只看该作者
li913 发表于 2020-5-17 20:15
1、equivalence(b,n), 二者的二进制表示形式一样(同一块内存),但值不一样;
2、n=b, 值一样(或近似,存在 ...

谢谢李老师,彻底明白了!感谢您的指点!继续前行,加油!

798

帖子

2

主题

0

精华

大宗师

F 币
3793 元
贡献
2268 点
22#
发表于 2020-5-17 20:15:34 | 只看该作者
1、equivalence(b,n), 二者的二进制表示形式一样(同一块内存),但值不一样;
2、n=b, 值一样(或近似,存在取舍), 二进制不一样。
3、文件中的某个地址上的值,到底是整型还是实型,需要通过判断才能确定,但不管是啥类型,二进制形式是相同的(文件不会改变)。
4、参考 http://fcode.cn/guide-120-1.html

28

帖子

4

主题

0

精华

熟手

F 币
134 元
贡献
72 点
21#
 楼主| 发表于 2020-5-17 20:10:17 | 只看该作者
我思考了下,是因为共用一块地址,地址里的值是二进制的,如010110,那么分别用不同类型的变量取值这段二进制的值时,就会翻译为相应的数据类型值。而n=b这种是数据上的强制转换

28

帖子

4

主题

0

精华

熟手

F 币
134 元
贡献
72 点
20#
 楼主| 发表于 2020-5-17 19:57:25 | 只看该作者
本帖最后由 evanustc 于 2020-5-17 20:04 编辑
li913 发表于 2020-5-17 16:13
研究了两天,发现这个代码是比较奇葩的,典型的国外开源代码风格,故意让人看不懂(也可能是代码太老了)。 ...

李老师,请收下我的膝盖orz....太厉害了。在结合工具代码的基础上,阅读您的代码,我明白了意思。在您的代码里,有一项我不太明白原理,就是a(1)读进来的时候还是REAL型的小数,但是在最后的循环中,使用了 b=a(1),然后整形n就成了74了。。。  equivalence(b,n)这一句,实型b和整型n共用同一块内存地址,是由于整型和实型的原因导致取同一块地址里的值时导致不一样的吗?为啥不用n=b转换呢

798

帖子

2

主题

0

精华

大宗师

F 币
3793 元
贡献
2268 点
19#
发表于 2020-5-17 16:13:17 | 只看该作者
本帖最后由 li913 于 2020-5-17 17:51 编辑

研究了两天,发现这个代码是比较奇葩的,典型的国外开源代码风格,故意让人看不懂(也可能是代码太老了)。核心也就那么十来行。将所有数据读入数组,通过一定规则(numchk函数)判断每个数据的真实类型(real 或 integer),然后每6个数据输出一行。

[Fortran] 纯文本查看 复制代码
!判断数据是整数还是实数
subroutine numchk (ia,icode)
character aho*7
write(aho,'(i7)') ia
if(aho.eq.'*******') then
  icode = 0
else
  icode = 1
endif
end
      
program test
implicit none
integer(4) m, n, i, j, k
real,allocatable:: a(:)
real b
equivalence(b,n)
character cl*3

!读取
open(10,file='FASTLIB', ACCESS='SEQUENTIAL',FORM='UNFORMATTED')
read(10) m
allocate(a(m)) !分配空间
rewind(10)
read(10) m, a(1:m)
close(10)

!输出
open(10,file='out.txt')
k=0
do i = 1, m
  b = a(i)
  k=k+1
  if(mod(k,6)==0) then !每6个数据输出一行
    cl = 'yes'
  else
    cl = 'no'
  end if
  call numchk(n,j) !判断数据是整数还是实数
  if(j==0) then
    write(10,'(1pe12.5)', advance=cl) b
  else
    write(10,'(4x,i7,1x)',advance=cl) n
  end if
end do
end

QQ截图20200517160850.png (106.58 KB, 下载次数: 299)

QQ截图20200517160850.png

1.png (81.34 KB, 下载次数: 278)

1.png

28

帖子

4

主题

0

精华

熟手

F 币
134 元
贡献
72 点
18#
 楼主| 发表于 2020-5-17 13:42:50 | 只看该作者
necrohan 发表于 2020-5-17 12:49
整数和浮点数在二进制文件里表示方法不一样,不能通用,否则读出来数据是错的。
你附件那个二进制文件按6 ...

是的,二进制读入没有问题。我明白您的意思,只是有点疑惑的就是小工具中关于二进制代码读取的部分:
      DIMENSION WORK(1)
      OPEN(UNIT=IOPDS, FILE=FILNAM, ERR=100, STATUS='UNKNOWN',
     &     ACCESS='SEQUENTIAL', FORM='UNFORMATTED',IOSTAT=IOS)
      READ(UNIT=IOPDS,ERR=300,IOSTAT=IOS)  LENG,(WORK(I),I=1,LENG)
      CLOSE(UNIT=IOPDS, ERR=200, STATUS='KEEP',IOSTAT=IOS)
      RETURN74
WORK声明的是数组,也没有指定类型,在UBUNTU下编译后,转出的10进制文件是正确的。我推断WORK中即读入了整型也有实型,不然打印成十进制不可能对的上。这部分我在win下实现后就出现了我刚才问的问题,屏幕输出的前4个本应该是74 10 22 45却变为了神秘小数。。

250

帖子

2

主题

0

精华

宗师

F 币
1730 元
贡献
872 点

规矩勋章

17#
发表于 2020-5-17 12:49:20 | 只看该作者
evanustc 发表于 2020-5-17 11:34
我把 DIMENSION WORK(300)改为INTEGER  WORK(300) 能够读入74 10 22 45但是后面小数就不行了,读的全是整数 ...

整数和浮点数在二进制文件里表示方法不一样,不能通用,否则读出来数据是错的。
你附件那个二进制文件按6楼的方式读没问题。

28

帖子

4

主题

0

精华

熟手

F 币
134 元
贡献
72 点
16#
 楼主| 发表于 2020-5-17 11:34:44 | 只看该作者
我把 DIMENSION WORK(300)改为INTEGER  WORK(300) 能够读入74 10 22 45但是后面小数就不行了,读的全是整数。如果WORK为REAL型,为啥不能把74当做74.0读入呢?类似C++里面的读入处理

28

帖子

4

主题

0

精华

熟手

F 币
134 元
贡献
72 点
15#
 楼主| 发表于 2020-5-17 11:25:44 | 只看该作者
有点激动,整了快2天的时间,对Fortran理解不少,太有意思了
您需要登录后才可以回帖 登录 | 极速注册

本版积分规则

捐赠本站|Archiver|关于我们 About Us|小黑屋|Fcode ( 京ICP备18005632-2号 )

GMT+8, 2024-5-5 09:28

Powered by Tencent X3.4

© 2013-2024 Tencent

快速回复 返回顶部 返回列表