class - Fortran 函数一次性获取指针的关联状态和大小
问题描述
我正在编写一个 Fortran (2003) 整数函数来检查指向任意元素数组的指针的关联状态和大小,使用class(*)
. 如果没有关联,则返回 -1,否则返回数组元素的数量。假设指针被定义为指向null()
。
目标是避免两个后续if
语句。(我知道在 C 中可以将 2 个检查组合成一个语句。)
该代码在 gfortran (6.1.0) 和 Intel ifort (17.0.4) 下编译和运行,但只有 gfortran 提供了所需的结果。
包含测试例程和名为“size_field”的函数的完整代码如下所示:
program tpa
implicit none
real,pointer :: tp(:)=>null()
write (6,*) size_field(tp)
allocate (tp(53)); write (6,*) size_field(tp)
deallocate (tp); write (6,*) size_field(tp)
tp=>null(); write (6,*) size_field(tp)
allocate (tp(0)); write (6,*) size_field(tp)
deallocate (tp); write (6,*) size_field(tp)
contains
function size_field(ff)
integer :: size_field
class(*),target :: ff(:)
class(*),pointer :: field(:)
field => ff
size_field=-1 ! shall indicate not associated
if (associated(field)) size_field=size(field)
end function
end program
使用 gfortran(默认编译器选项)编译,输出符合预期:
-1
53
-1
-1
0
-1
使用 ifort(默认编译器选项)编译,输出为:
0
53
53
53
0
0
为什么 ifort 没有给出我想要的结果?
如果我替换class(*)
为real
,则 ifort 编译的程序也会显示正确的结果。
解决方案
您的程序不是有效的 Fortran。当被要求编译这个程序时,允许编译器给出它喜欢的任何结果。
哑论ff
class(*), target :: ff(:)
是一个非指针、非可选的虚拟参数,因此(Fortran 2018 15.5.2.3 p1)引用函数时的任何指针实际参数size_field
,在这种情况下tp
,必须是与目标关联的指针。
在第一个引用tp
上,没有指针关联。
至于如何满足您的要求,“非可选”部分是一个很大的提示。在 Fortran 2008+ 下,可以将虚拟参数设为可选,然后在引用函数时实际参数不需要与指针关联。如果实际参数与指针无关,则虚拟参数不存在:
function size_field(ff)
integer :: size_field
class(*), target, optional :: ff(:)
if (PRESENT(ff)) then
size_field = SIZE(ff)
else
size_field = -1
end if
end function
但是,在问题的情况下,解决限制的“非可选”部分是唯一有用的方法。试图将虚拟参数改为指针是没有帮助的:对于多态虚拟指针参数,实际参数也必须是多态的(并且对于无限多态虚拟,实际参数也必须是无限多态的)。
推荐阅读
- python - 使用 tkinter 的网格布局管理器,按钮出现在错误的列中
- node.js - 将文件从谷歌云存储直接下载到nodejs中的客户端
- javascript - 更改数组中所有输入的值
- mqtt - 获取保留消息列表
- linux - 如何挂载 nginx docker 的 /etc/nginx/ 目录来托管虚拟机?
- install4j - 安装程序图像的 Install4j Web url
- asp.net-core - IDX20803:无法从 - 多权限获取配置
- php - 如何在列 SQL 中获取 5 个最新值和计数
- google-cloud-dataflow - 触发后在条件下将元素发送到下一个窗口
- python - Python overlapping sliding dataframe