马上注册,学习空分知识,结交更多空分大神!
您需要 登录 才可以下载或查看,没有账号?加入空分之家
x
FORTRAN语言动态链接库的使用及混合编译 转载自网上搜索到的一篇介绍FORTRAN动态链接库的资料——《Fortran语言的动态链接库的开发及应用》,同上海汽轮机期刊中《水和水蒸汽热力性质动态链接库的开发及在汽轮机中的应用》一文一致。仅供参考。 1 引言
3 [! `5 C- t5 \. C由于FORTRAN在开发图形界面、可视计算、面对对象、事件驱动、数据库应用方面存在局限性,如果我们利用其它语言来重写,将费时费力,同时又损失了FORTRAN语言在科学计算上的优越性。经过努力,笔者寻求到一种折衷的方法:利用Microsoft Fortran PowerStation 4.0将FORTRAN源代码编译为动态链接库,然后用其它语言建立应用程序的项目文件,在适当的时候调用该动态链接库。这样就把DOS环境中的软件包括移植到Windows环境,使问题得以圆满解决。1 S& R, c/ F- L. }/ F
以下以最常用的子例程PT为例来说明动态链接库的建立和在各种流行程序语言中的调用方法。
4 B$ S; T! y/ N3 k7 |0 P: k4 a2 n" Q" @" a& ^. ?0 k! _# x
2 基于Fortran语言的动态链接库的建立
" C+ H6 X" q" d/ X3 `2.1 建立动态链接库的项目
Z$ x/ g6 n5 U0 D# }) _# N在Fortran PowerStation开发环境中,建立动态链接库的项目文件(IFCSW.MDP)。
5 c' w. s3 t, c( |% x2.2 编写动态链接库的源代码+ q3 D8 Y$ [9 ^' e- }
编写动态链接库的源代码文件(IFCSW.F90),并将源代码文件插入到动态链接库的项目文件中。
/ t' [4 T! F6 U7 }# N- t在动态链接库源代码(IFCSW.F90)中子例程PT的定义为
6 s: A$ j+ ]/ t% o' ]' H' dSubroutine PT(P,T,H,S,V,X,R)
1 ^! n7 R; I! X0 o3 O i2 e1 \+ Q!MS$Attributes Stdcall , Dllexport , Alias : ’PT’ :: PT/ m' O% O7 r, {9 O
!MS$Attributes Value :: P,T o( I# }7 Z& s: Q$ N
!MS$Attributes Reference :: H,S,V,X,R
+ \" z G/ D' V( hReal(4) P,T,H,S,V,X" t4 F6 _* ?) s7 T
Integer(2) R6 E' m6 f, u9 H" W7 {
…………
" q6 p- U8 _' X2 x* V3 l- mEnd Subroutine
& T4 j. w" T5 H+ t) e1 _8 n z$ ?6 T8 r+ O. a5 o
其中:P为压力,T为温度,H为比焓,S为比熵,V为比容,X为干度,R为所处的区间;如果调用前已知所处的区间,可直接给R赋区间号后再调用子例程,可避免区间判断而加快调用速度。当所处的区间未知或不确定时,由子例程内部进行区间判断后,针对不同区间再调用不同的其它例程,并将所处区间号传回。
! g+ T5 ~% x$ q3 D. e/ q, U由于编译后动态链接库中命名方式的差异,为避免在VC++中隐式加载该DLL时产生连接错误,则专为VC++定义如下的子例程(仅供VC++使用),即不要使用命名别名的Alias语句:- `' j$ g# V3 A* Y _: d, j1 G3 ~( I
Subroutine PTVC(P,T,H,S,V,X,R)" S( v7 A9 F1 ?6 P: k9 d y
!MS$Attributes Stdcall , Dllexport :: PTVC
8 K4 ` K& p: p) e1 j!MS$Attributes Value :: P,T; s' e V/ J; J1 S9 n, I
!MS$Attributes Reference :: H,S,V,X,R" E3 L1 c/ y5 ?* w' r" J& m% J
Real(4) P,T,H,S,V,X
6 H V1 k$ J" [8 M* e* }5 S$ ?Integer(2) R
3 g$ F7 d9 d' ?' f…………# |" @1 J, X6 h" d9 I$ R
End Subroutine
" y7 c# @# P) J1 f; v
* P" ~* x( M$ n2 D& E: e2.3 创建动态链接库) [7 V9 Q! g+ i- q" v
对动态链接库的项目文件进行编译连接,获得动态链接库(IFCSW.DLL)和接入库(IFCSW.LIB,该接入库通常用于隐性加载动态链接库,且只能用于Microsoft的开发语言,对于其它公司的产品,需利用专用程序生成对应的接入库)。
. n% U5 A7 r( ~/ X: g/ [- ^1 V! ~! R+ Q- S5 z
3 动态链接库的使用/ K& I: r$ S; B' {, _( }7 l
一旦生成了动态链接库,就可以在各种编程语言环境中调用该动态链接库中的各种子程序或子例程。
9 I- o! s6 @: N0 X0 X" k$ r动态链接库有隐式和显式两种装载方式,各有优缺点。采用隐式比显式友好,需要较少的编码,装载快速、可靠,但缺乏控制,影响动态使用,资源使用不经济;采用显式则正好相反。
% }0 C+ }8 y$ z( h采用隐式通常需要接入库和函数申明。
2 f5 D9 X7 i* Y4 Q# o! M7 @采用显式则应遵循以下步骤:; S- y M6 R1 D2 w
(1) 创建一个函数指针,其指针类型与要调用的DLL引出函数相吻合。
' x- e D( V/ q; _$ e, J(2) 通过Win32 API函数LoadLibrary()显式地调用DLL,此函数返回DLL的实例句柄。: A/ c1 U1 K, Y
(3) 通过Win32 API函数GetProAddress()获得要调用的DLL的函数地址,把结果赋予自定义函数指针类型的指针。
% y6 W2 Y! l: |2 Q7 U$ D(4) 使用函数指针来调用DLL中函数。* Z& K& S9 {7 W# ?7 A H9 T
(5) 调用完成后,通过Win32 API函数FreeLibrary()释放DLL。
; l7 o3 ?& q. L, K
8 I* w% P, }, s" D实际使用中可将隐式和显式两者结合使用,采用混合模式。; S2 r9 V7 [$ C3 k# C# d
, c. `+ |9 ^) _# s$ Y5 G- R3.1 在MS Fortran PowerStation 4.0中调用
$ Q @; K I' S6 W0 q, k9 A① 隐式调用. B5 K1 ^7 f2 K# c4 q9 ]" U
将接入库(IFCSW.LIB)包含到项目文件中,另在源文件的开始部分加入以下的声明语句
. B. ~! U' v# CInterface
7 r q. u. l/ gSubroutine PT(P,T,H,S,V,X,R)* Q6 x- }# f; _
!MS$Attributes Stdcall , Dllimport , Alias : ’PT’ :: PT! v) c- W; V2 n3 P! f
Real(4) P,T,H,S,V,X0 m1 g3 I" k' Z) Q
Integer(2) R
$ k" J2 t8 |% X3 [$ c, K7 ~+ b+ A!MS$Attributes Value :: P,T/ j1 S1 O8 ^$ v/ Q
!MS$Attributes Reference :: H,S,V,X,R1 `% b4 a; _3 k, f4 F+ b9 E
End Subroutine
& F* z1 L) o; h* A, v% S; P' FEnd Interface9 `' O) `0 w) L5 ^; A
此后,在需要的地方可用以下语句调用
5 j$ t3 ]/ q: z% F* a# ^( rCall PT(P,T,H,S,V,X,R)
/ I/ L; D" n5 d# @# g0 N
. F# n8 \, A4 d* l+ k0 E4 c$ a; q② 显式调用. E& B' E. b/ J3 Y' w5 s& w( x4 r
动态装载首先要用LoadLibrary函数将动态链接库装入;然后用GetProcAddress函数获取动态链接库中要调用的函数或过程的地址;由于Fortran 90没有提供函数指针,不能直接用地址进行函数的调用,通过编写一个用函数名或过程名作参数的子程序CallProc来解决;将函数名或过程名作为参数传递时,实际传递的是四字节的函数或过程的入口地址,CallProc子程序接收该地址后,根据所声明的函数接口,以正确的方式对函数或过程进行调用;动态链接库使用完后,要及时用FreeLibrary函数释放动态链接库。
0 W5 K' c" C! L, ^1 F/ kSubroutine CallProc(Pn,P,T,H,S,V,X,R)
- _' E' x+ x& e( o$ t9 u) {0 ^Interface0 P1 }8 F( V% C1 ]* u9 c. L
Subroutine Pn(P,T,H,S,V,X,R)8 Q, V% {/ I; w! u* V
!MS$Attributes Stdcall , Dllimport :: Pn
1 P9 _5 H5 J- a# r! {! GReal(4) P,T,H,S,V,X
' Z7 `! q" t) u# G2 L& [Integer(2) R
8 ?: y( z- H9 a8 a% |2 j!MS$Attributes Value :: P,T
, t# S$ K/ F; n7 K!MS$Attributes Reference :: H,S,V,X,R; E1 k; m% }( E, N: }: |! @
End Interface
q6 E% R6 B3 GReal(4) P,T,H,S,V,X+ \9 X8 J- {0 m1 V" I
Integer(2) R6 `, h) g9 t. W, [; Q7 U8 Y& T
Call Pn(P,T,H,S,V,X,R)$ i( H6 S9 d9 B; m7 {
End Subroutine
# r6 c) ]! |0 {4 r) F) D1 T- z. T; ^$ ?" x' j
CallProc(PT, P,T,H,S,V,X,R)
: \, r6 O- ~- Q7 O" y) n' g
: u. C% Z7 ?* w# g9 N3.2 在Visual Basic和Visual Basic Application中调用" }8 g3 h4 h# F: K
VB和VBA中由于显式加载过于繁琐,通常不使用,以下仅介绍隐式加载。8 N" i% Y9 v. i2 t" L
VB和VBA中隐式加载不需要接入库,只需在模块代码的声明部分增加Declare语句,声明需调用的函数在外部动态链接库中。
/ S/ P2 M' j6 |( @& u& nDecalre Sub PT Lib “IFCSW.DLL” (ByVal P as Double , ByVal T as Double , ByRef H as Double , ByRef S as Double , ByRef V as Double , ByRef X as Double , ByRef R as Integer). U$ Y8 Y2 h7 k0 |9 g. _3 N
而后,即可在任何需要的地方用以下语句调用子程序PT。
3 S8 p. q0 v4 Z4 M O) s6 TCALL PT(P,T,H,S,V,X,R): {3 b1 f0 }6 O( T' n- E
6 d; l: [, i9 x3.3 在Delphi(Visual Pascal)中调用, r& }* ]6 T; O% Z
|