您的位置:首页 > 教程 > 编程开发 > 编程语言综合 > VC++调用Fortran生成的DLL实例

VC++调用Fortran生成的DLL实例

栏目: 编程语言综合 来源:   时间: 2012-04-14 17:54

  VC++++Fortran混合编程借助于Fortran生成的DLL进行

  (采用C默认的传址方式进行函数参数传递)

  1.Fortran 生成DLL

  新建Fortran DLL程序test1.f

  添加如下代码:

  ! test1.f90

  !

  ! FUNCTIONS/SUBROUTINES exported from test1.dll:

  ! test1 - subroutine

  !示例没有返回值的子例程

  subroutine test1(a,b)

  ! Expose subroutine test1 to users of this DLL

  !

  !DEC$ ATTRIBUTES C,DLLEXPORT::test1

  ! Variables

  ! Body of test1

  integer a,b

  integer sum

  sum=a+b

  return

  end subroutine test1

  !示例有返回值的整数四则运算

  !两数相加

  function add(a,b)

  implicit none

  !DEC$ ATTRIBUTES C,DLLEXPORT::add

  integer a,b,add

  add=a+b

  return

  end

  !两数相减

  function abstract(a,b)

  implicit none

  !DEC$ ATTRIBUTES C,DLLEXPORT::abstract

  integer a,b,abstract

  abstract=a-b

  return

  end

  !两数相乘

  function multiply(a,b)

  implicit none

  !DEC$ ATTRIBUTES C,DLLEXPORT::multiply

  integer a,b,multiply

  multiply=a*b

  return

  end

  !两数相除 (需要添加考虑被除数是否为0以及能否整除的判断)

  function divided(a,b)

  implicit none

  !DEC$ ATTRIBUTES C,DLLEXPORT::divided

  integer a,b,divided

  divided=a/b

  return

  end

  编译后生成test1.dll,test1.obj等文件。其中这两个文件是我们在VC中调用所需要的。

  查看test1.dll中生成的函数如下图所示。

   注意:使用伪注释语句!DEC$ ATTRIBUTES C,DLLEXPORT::functionName后,生成的函数名与Fortran中定义的函数名一致,并没有按照Fortran编译器的默认属性将 函数名都转变为大些,如下图所示。在后续的VC中调用的时候需要保持调用的函数名称一致,否则会出现找不到函数的错误提示。

VC++调用Fortran生成的DLL实例 三联

  2.VC控制台调用Fortra生成的DLL

  新建VC控制台应用程序,新建一个checktest1的工程。

   注意:需要在工程的project菜单下的add to project子菜单的file对话框中添加上一步生成的test1.dll,test1.obj两个文件,否则编译能通过,链接的时候失败。还需将以上 两个文件拷贝到checktest1工程的debug目录下,否则运行的时候出现找不到文件的错误提示。自己测试了一下,以上两步是必须的。

  添加如下代码:(注意红色的部分)

  #include “stdafx.h”

  #include “iostream.h”

  //extern “C”{_stdcall TEST1(int* x,int* y);}

  //extern “C”{_stdcall ADD(int* x,int* y);}

  //extern “C”{_stdcall ABSTRACT(int* x,int* y);}

  //extern “C”{_stdcall MULTIPLY(int* x,int* y);}

  //extern “C”{_stdcall DIVIDED(int* x,int* y);}

  //注意此处函数名称要与DLL生成时保持一致(如下中的蓝色部分),否则会出现找不到函数的错误提示。并且一定要记得去掉参数中的指针符号*。

  extern “C”{_cdecl test1(int x,int y);}

  extern “C”{_cdecl add(int x,int y);}

  //采用C的传值方式,则需要将_stdcall修改为_cdecl

  //相应的Fortran DLL处要添加C的调用方式,即将!DEC$ ATTRIBUTES DLLEXPORT::add修改为:!DEC$ ATTRIBUTES C,DLLEXPORT::add

  //适应伪注释!DEC$ ATTRIBUTES C,DLLEXPORT::add后生成的DLL函数中只存在函数名为add的函数,ADD 和_ADD@8 均不存在,参见上图中的DLL函数名称

  extern “C”{_cdecl abstract(int x,int y);}

  extern “C”{_cdecl multiply(int x,int y);}

  extern “C”{_cdecl divided(int x,int y);}

  int main(int argc, char* argv[])

  {

  int a=35,b=5;

  int sum=0;

  int abs=0;

  int mul=0;

  int div=0;

  //TEST1(&a,&b);

  //sum=ADD(&a,&b);

  //abs=ABSTRACT(&a,&b);

  //mul=MULTIPLY(&a,&b);

  //div=DIVIDED(&a,&b);

  test1(a,b);

  sum=add(a,b);

  abs=abstract(a,b);

  mul=multiply(a,b);

  div=divided(a,b);

  printf(“a+b= %dn”,sum);

  printf(“a-b= %dn”,abs);

  printf(“a*b= %dn”,mul);

  printf(“a/b= %dn”,div);

  printf(“Hello World!n”);

  return 0;

  }

  然后编译运行可以得出正确的结果: