7

我有一个关于 Fortran 77 的问题,但我找不到解决方案。

我正在尝试存储定义如下的字符串数组:

character matname(255)*255

这是一个由 255 个长度为 255 的字符串组成的数组。

后来我从一个文件中读取了名称列表,并像这样设置了数组的内容:

matname(matcount) = mname

编辑:实际上mnamevalue 是硬编码mname = 'AIR'的 type ,它是执行上一行character*255的函数的参数。matadd()但这仅用于测试,将来它将从文件中读取。

稍后我想用以下方式打印它:

write(*,*) matname(matidx)

但它似乎打印了所有 255 个字符,它打印了我分配的字符串和很多垃圾。

  • 所以这是我的问题,我怎么知道存储的字符串的长度?
  • 我应该有另一个所有长度的数组吗?
  • 我怎么知道读取的字符串的长度?

谢谢。

4

4 回答 4

5

您可以使用此函数获取长度(没有空白尾部)

integer function strlen(st)
      integer       i
      character     st*(*)
      i = len(st)
      do while (st(i:i) .eq. ' ')
        i = i - 1
      enddo
      strlen = i
      return
      end

从这里得到:http: //www.ibiblio.org/pub/languages/fortran/ch2-13.html

PS:当你说: matname(matidx) 它得到整个字符串(256)个字符......所以这是你的字符串加上空格或垃圾

于 2009-04-07T12:37:30.097 回答
4

只要您感兴趣的字符串部分仅包含空格, Timotei 发布的函数将为您提供字符串的长度,如果您在程序中分配值应该为真,因为 FORTRAN 应该初始化变量为空,对于表示空格的字符。

但是,如果您正在从文件中读取,您可能会在行尾拾取其他控制字符(特别是回车和/或换行符,\r 和/或 \n,具体取决于您的操作系统)。您还应该将它们扔到函数中以获得正确的字符串长度。否则你会得到一些有趣的打印语句,因为这些字符也会被打印出来。

这是我的函数版本,它在末尾检查除空格之外的备用空白字符。

  function strlen(st)
  integer           i,strlen
  character         st*(*)
  i = len(st)
  do while ((st(i:i).eq.' ').or.(st(i:i).eq.'\r').or.
 +  (st(i:i).eq.'\n').or.(st(i:i).eq.'\t'))
    i = i - 1
  enddo
  strlen = i
  return
  end

如果“垃圾”部分中有其他字符,这仍然无法完全工作。

但是,假设它确实适用于您的数据,那么您可以将您的 write 语句更改为如下所示:

write(*,*) matname(matidx)(1:strlen(matname(matidx)))

它只会打印出实际的字符串。

至于是否应该使用另一个数组来保存字符串的长度,这取决于您。strlen() 函数是 O(n),而在表中查找长度是 O(1)。如果您发现自己经常计算这些静态字符串的长度,那么在读入它们时计算一次长度,将它们存储在数组中并在需要时查找它们可能会提高性能。但是,如果您没有注意到减速,我不会担心。

于 2009-04-07T16:35:08.620 回答
1

根据您使用的编译器,您可以使用trim()内部函数从字符串中删除任何前导/尾随空格,然后像往常一样处理它,即

character(len=25) :: my_string
my_string = 'AIR'
write (*,*) ':', trim(my_string), ':'

应该打印:AIR:

编辑:更好的是,看起来有一个len_trim()函数可以在修剪后返回字符串的长度。

于 2009-04-07T19:27:41.030 回答
0

intel 和 Compaq Visual Fortran 具有内在函数 LEN_TRIM(STRING),它返回不带尾随空格或空格的长度。

如果要抑制前导空格或空格,请使用“向左调整”,即 ADJUSTF(STRING)

在这些 FORTRAN 中,我还注意到一个有用的特性:如果将字符串作为参数传递给函数或子例程,并且在子例程中将其声明为 CHARACTER*(*),然后在子例程中使用 LEN(STRING) 函数返回传入的实际字符串长度,而不是调用程序中声明的字符串长度。

示例:CHARACTER*1000 STRING

        .
        .

  CALL SUBNAM(STRING(1:72)

  SUBROUTINE SYBNAM(STRING)
  CHARACTER*(*) STRING
  LEN(STRING) will be 72, not 1000
于 2013-06-18T03:15:53.410 回答