Return 从 Fortran 到 C++ 的字符串
Return string from Fortran to C++
我在 C++ 中有以下函数调用:
int strLength = 20;
char* name;
getName(name, strLength);
printf("name: %s\n", name);
在 Fortran 中:
subroutine getName(name) bind (c, name='GETNAME')
use,intrinsic :: iso_c_binding
implicit none
character, intent(out) :: name
name = 'Martin'
end subroutine getName
当我执行 C++ 例程时,输出是:name: M
。现在,我想这是因为 character, intent(out) :: name
声明了大小为 1 的 name
变量,但是如果我将声明更改为:character(len=6), intent(out) :: name
我会收到此错误消息:Error: Character argument 'name' at (1) must be length 1 because procedure 'getname' is BIND(C)
.我也试过这个:character(len=6,kind=c_char), intent(out) :: name
,有同样的错误信息。最后,我尝试了这个声明:character(c_char),dimension(6), intent(out) :: name
,编译但产生了这个结果:name: MMMMMM
.
我的问题归结为:如何 return 从 Fortran 到 C++ 的字符串?
问题在于将等级 n
的字符数组分配给长度 n
的字符标量,这在 this 问题中得到解决。如下所示更改 Fortran 例程解决了问题。
解决方案
因为我在遗留系统中工作,所以我不得不寻求下面指出的解决方案。显然不完全一样,但还是显示了大概的结构。如果其他人也应该采用此解决方案,您可能应该遵循 Matt P 评论中的建议。
虽然这是我寻求的解决方案,但我觉得 Matt P 的回答是对问题标题中所述一般问题的更好解决方案,这就是我接受它作为答案的原因。
C++
int strLength = 20;
char* name = new char[strLength];
getName(name, strLength);
printf("name: %s\n", name);
Fortran
subroutine getName(name) bind (c, name='GETNAME')
use,intrinsic :: iso_c_binding
implicit none
character(c_char),dimension(20), intent(out) :: name
character(20) :: fName
fName = 'Martin'//c_null_char
do j=1,len(fName )
name(j) = fName (j:j)
enddo
end subroutine getName
这种基于 Fortran 函数的方法简洁明了,非常适用于尚未在 C 例程中设置(或为其分配内存)的字符串。请注意,您不需要传入字符串长度参数,或使用假定大小的数组来 build/return 字符串值。1
使用Fortran 可分配字符串常量,因此该函数可重复用于任何字符串。
一个整数参数也被传递到 Fortran 函数以演示,例如,您可以如何指示所需的响应应该是什么。
注意在本例中 "intent(out)" 用于指示整数参数在传入之前不需要定义,但可以在 returned 之前对其进行更新。因此,您可以修改其值并将其 return 修改为调用程序,以便它可以用作 "return code".
Fortran 函数
! f_string.f90
! Called from C routine as: `myString = get_string(rc)`
function get_string(c_rc) bind(c, name='get_string')
use, intrinsic :: iso_c_binding
implicit none
integer(c_int), intent(out) :: c_rc ! <- Pass by reference; acts as return code in this example.
type(c_ptr) :: get_string ! <- C_PTR to pass back to C
character(len=:), allocatable, target, save :: fortstring ! <- Allocatable/any length is fine
fortstring = "Append C_NULL_CHAR to any Fortran string constant."//C_NULL_CHAR
get_string = c_loc(fortstring) ! <- C_LOC intrinsic gets loc of our string.
c_rc = 1 ! <- Set the return code value.
end function get_string
C 程序
// c_string.c
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
char *get_string(int *rc); // <- The Fortran function signature.
int main(){
int rc; // <- No value set.
char *mynameptr; // <- Just a plain ol' char *, no memory allocation needed.
mynameptr = get_string(&rc);
printf("mynameptr=%s\n", mynameptr);
printf("len =%d\n", strlen(mynameptr));
printf("rc =%d\n", rc);
return rc;
}
编译、调用、输出:
ifort /c f_string.f90
icl c_string.c /link f_string.obj
c_string.exe
# example output:
# mynameptr=Append C_NULL_CHAR to any Fortran string constant.
# len =48
# rc =1
1 这也可以在没有警告的情况下干净地编译(在没有修改的 中确实会发生我在评论中提出了建议)。
我在 C++ 中有以下函数调用:
int strLength = 20;
char* name;
getName(name, strLength);
printf("name: %s\n", name);
在 Fortran 中:
subroutine getName(name) bind (c, name='GETNAME')
use,intrinsic :: iso_c_binding
implicit none
character, intent(out) :: name
name = 'Martin'
end subroutine getName
当我执行 C++ 例程时,输出是:name: M
。现在,我想这是因为 character, intent(out) :: name
声明了大小为 1 的 name
变量,但是如果我将声明更改为:character(len=6), intent(out) :: name
我会收到此错误消息:Error: Character argument 'name' at (1) must be length 1 because procedure 'getname' is BIND(C)
.我也试过这个:character(len=6,kind=c_char), intent(out) :: name
,有同样的错误信息。最后,我尝试了这个声明:character(c_char),dimension(6), intent(out) :: name
,编译但产生了这个结果:name: MMMMMM
.
我的问题归结为:如何 return 从 Fortran 到 C++ 的字符串?
问题在于将等级 n
的字符数组分配给长度 n
的字符标量,这在 this 问题中得到解决。如下所示更改 Fortran 例程解决了问题。
解决方案
因为我在遗留系统中工作,所以我不得不寻求下面指出的解决方案。显然不完全一样,但还是显示了大概的结构。如果其他人也应该采用此解决方案,您可能应该遵循 Matt P 评论中的建议。
虽然这是我寻求的解决方案,但我觉得 Matt P 的回答是对问题标题中所述一般问题的更好解决方案,这就是我接受它作为答案的原因。
C++
int strLength = 20;
char* name = new char[strLength];
getName(name, strLength);
printf("name: %s\n", name);
Fortran
subroutine getName(name) bind (c, name='GETNAME')
use,intrinsic :: iso_c_binding
implicit none
character(c_char),dimension(20), intent(out) :: name
character(20) :: fName
fName = 'Martin'//c_null_char
do j=1,len(fName )
name(j) = fName (j:j)
enddo
end subroutine getName
这种基于 Fortran 函数的方法简洁明了,非常适用于尚未在 C 例程中设置(或为其分配内存)的字符串。请注意,您不需要传入字符串长度参数,或使用假定大小的数组来 build/return 字符串值。1
使用Fortran 可分配字符串常量,因此该函数可重复用于任何字符串。
一个整数参数也被传递到 Fortran 函数以演示,例如,您可以如何指示所需的响应应该是什么。
注意在本例中 "intent(out)" 用于指示整数参数在传入之前不需要定义,但可以在 returned 之前对其进行更新。因此,您可以修改其值并将其 return 修改为调用程序,以便它可以用作 "return code".
Fortran 函数
! f_string.f90
! Called from C routine as: `myString = get_string(rc)`
function get_string(c_rc) bind(c, name='get_string')
use, intrinsic :: iso_c_binding
implicit none
integer(c_int), intent(out) :: c_rc ! <- Pass by reference; acts as return code in this example.
type(c_ptr) :: get_string ! <- C_PTR to pass back to C
character(len=:), allocatable, target, save :: fortstring ! <- Allocatable/any length is fine
fortstring = "Append C_NULL_CHAR to any Fortran string constant."//C_NULL_CHAR
get_string = c_loc(fortstring) ! <- C_LOC intrinsic gets loc of our string.
c_rc = 1 ! <- Set the return code value.
end function get_string
C 程序
// c_string.c
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
char *get_string(int *rc); // <- The Fortran function signature.
int main(){
int rc; // <- No value set.
char *mynameptr; // <- Just a plain ol' char *, no memory allocation needed.
mynameptr = get_string(&rc);
printf("mynameptr=%s\n", mynameptr);
printf("len =%d\n", strlen(mynameptr));
printf("rc =%d\n", rc);
return rc;
}
编译、调用、输出:
ifort /c f_string.f90
icl c_string.c /link f_string.obj
c_string.exe
# example output:
# mynameptr=Append C_NULL_CHAR to any Fortran string constant.
# len =48
# rc =1
1 这也可以在没有警告的情况下干净地编译(在没有修改的