-
Notifications
You must be signed in to change notification settings - Fork 11
/
dehoog.m
155 lines (136 loc) · 5.49 KB
/
dehoog.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
function f = dehoog(F, x, t, alpha, tol);
% DEHOOG - numerical inverse Laplace transform
%
% Syntax: f = dehoog(F, x, t, alpha, tol )
%
% F = laplace-space function (string refering to an m-file),must
% have form F(s, P1,..,P9), where s is the Laplace parameter
% and return column vector as result
% t = column vector of times for which real-space function values
% are sought
% alpha = largest pole of F (default zero)
% tol = numerical tolerance of approaching pole (default 1e-9)
% P1-P9 = optional parameters to be passed on to F
% f = vector of real-space values f(t)
%
% Description:
% algorithm: de Hoog et al's quotient difference method with accelerated
% convergence for the continued fraction expansion
%
% [de Hoog, F. R., Knight, J. H., and Stokes, A. N. (1982). An improved
% method for numerical inversion of Laplace transforms. S.I.A.M. J. Sci.
% and Stat. Comput., 3, 357-366.]
%
% Modification: The time vector is split in segments of equal magnitude
% which are inverted individually. This gives a better overall accuracy.
%
% Details: de Hoog et al's algorithm f4 with modifications (T->2*T and
% introduction of tol). Corrected error in formulation of z.
%
% Copyright: Karl Hollenbeck
% Department of Hydrodynamics and Water Resources
% Technical University of Denmark, DK-2800 Lyngby
% email: karl@isv16.isva.dtu.dk
%
% 22 Nov 1996, MATLAB 5 version 27 Jun 1997 updated 1 Oct 1998
% IF YOU PUBLISH WORK BENEFITING FROM THIS M-FILE, PLEASE CITE IT AS:
% Hollenbeck, K. J. (1998) INVLAP.M: A matlab function for numerical
% inversion of Laplace transforms by the de Hoog algorithm,
% http://www.isva.dtu.dk/staff/karl/invlap.htm
%
% Renamed by P. Renard to be integrated in the hytool toolbox.
%
% example:
% identity function in Laplace space:
% function F = identity(s); % save these two lines
% F = 1./(s.^2); % ... as "identity.m"
% invlap('identity', [1;2;3]) % gives [1;2;3]
%
% See also: stefhest
%
if( size(t,2)>1 ) % This algorithm works only for column vector
t=t'; % If the input vector is a line vector, let's transpose it.
flag=1;
else
flag=0;
end
if nargin <= 3,
alpha = 0;
elseif isempty(alpha),
alpha = 0;
end
if nargin <= 4,
tol = 1e-9;
elseif isempty(tol),
tol = 1e-9;
end
f = [];
% split up t vector in pieces of same order of magnitude, invert one piece
% at a time. simultaneous inversion for times covering several orders of
% magnitudes gives inaccurate results for the small times.
allt = t; % save full times vector
logallt = log10(allt);
iminlogallt = floor(min(logallt));
imaxlogallt = ceil(max(logallt));
for ilogt = iminlogallt:imaxlogallt, % loop through all pieces
t = allt(find((logallt>=ilogt) & (logallt<(ilogt+1))));
if ~isempty(t), % maybe no elements in that magnitude
T = max(t)*2;
gamma = alpha-log(tol)/(2*T);
% NOTE: The correction alpha -> alpha-log(tol)/(2*T) is not in de Hoog's
% paper, but in Mathematica's Mathsource (NLapInv.m) implementation of
% inverse transforms
nt = length(t);
M = 20; % Commentaire P. Renard: 20 est la valeur par defaut.
% Pour amŽliorer le rŽsultat on peut augmenter M
run = [0:1:2*M]'; % so there are 2M+1 terms in Fourier series expansion
% find F argument, call F with it, get 'a' coefficients in power series
s = gamma + i*pi*run/T;
a=feval(F,x,s);
a(1) = a(1)/2; % zero term is halved
% build up e and q tables. superscript is now row index, subscript column
% CAREFUL: paper uses null index, so all indeces are shifted by 1 here
e = zeros(2*M+1, M+1);
q = zeros(2*M , M+1); % column 0 (here: 1) does not exist
e(:,1) = zeros(2*M+1,1);
q(:,2) = a(2:2*M+1,1)./a(1:2*M,1);
for r = 2:M+1, % step through columns (called r...)
e(1:2*(M-r+1)+1,r) = ...
q(2:2*(M-r+1)+2,r) - q(1:2*(M-r+1)+1,r) + e(2:2*(M-r+1)+2,r-1);
if r<M+1, % one column fewer for q
rq = r+1;
q(1:2*(M-rq+1)+2,rq) = ...
q(2:2*(M-rq+1)+3,rq-1).*e(2:2*(M-rq+1)+3,rq-1)./e(1:2*(M-rq+1)+2,rq-1);
end
end
% build up d vector (index shift: 1)
d = zeros(2*M+1,1);
d(1,1) = a(1,1);
d(2:2:2*M,1) = -q(1,2:M+1).'; % these 2 lines changed after niclas
d(3:2:2*M+1,1) = -e(1,2:M+1).'; % ...
% build up A and B vectors (index shift: 2)
% - now make into matrices, one row for each time
A = zeros(2*M+2,nt);
B = zeros(2*M+2,nt);
A(2,:) = d(1,1)*ones(1,nt);
B(1:2,:) = ones(2,nt);
z = exp(i*pi*t'/T); % row vector
% after niclas back to the paper (not: z = exp(-i*pi*t/T)) !!!
for n = 3:2*M+2,
A(n,:) = A(n-1,:) + d(n-1,1)*ones(1,nt).*z.*A(n-2,:); % different index
B(n,:) = B(n-1,:) + d(n-1,1)*ones(1,nt).*z.*B(n-2,:); % shift for d!
end
% double acceleration
h2M = .5 * ( ones(1,nt) + ( d(2*M,1)-d(2*M+1,1) )*ones(1,nt).*z );
R2Mz = -h2M.*(ones(1,nt) - ...
(ones(1,nt)+d(2*M+1,1)*ones(1,nt).*z/(h2M).^2).^.5);
A(2*M+2,:) = A(2*M+1,:) + R2Mz .* A(2*M,:);
B(2*M+2,:) = B(2*M+1,:) + R2Mz .* B(2*M,:);
% inversion, vectorized for times, make result a column vector
fpiece = ( 1/T * exp(gamma*t') .* real(A(2*M+2,:)./B(2*M+2,:)) )';
f = [f; fpiece]; % put pieces together
end % if not empty time piece
end % loop through time vector pieces
if( flag==1 )
f=f';
end