-
Notifications
You must be signed in to change notification settings - Fork 21
/
simpleNewtonRaphson.m
executable file
·247 lines (199 loc) · 6.65 KB
/
simpleNewtonRaphson.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
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
function [plt, pltel] = simpleNewtonRaphson(mod, control, plotdofs, plotelem )
% SIMPLENEWTONRAPHSON Nonlinear analysis of model with Newton-Raphson procedure
% [PLT, PLTEL]=SIMPLENEWTONRAPHSON(MOD,CONTROL,PLOTDOFS, PLOTELEM) Nonlinear analysis of
% a model using the Newton-Raphson strategy under load control.
% The loading is applied in equal increments. If convergence is not
% achieved, the solution terminates at the last converged load step.
% The function returns arrays of specified DOF and element responses
% for each load step.
%
% Input Parameters
% ----------------
% MOD = The model to be analyzed
% CONTROL = Array of elements for controling solution:
% CONTROL(1) = Number of load steps
% CONTROL(2) = load increment (delta-lambda)
% CONTROL(3) = Maximum number of equilibrium iterations
% per load step
% CONTROL(4) = Tolerance for equilibrium residual
% for convergence of load step
% PLOTDOFS = Array of DOF's for which response is stored (for plotting)
% PLOTELEM = Array of DOF's for which response is stored (for plotting)
%
% Return Variables
% ----------------
% PLT = Matrix of load factors (lambda) and values of DOF for
% each load step. PLT(:,1) contain the load factors (0-1)
% for the load steps. PLT(:,1+j) contains the values of
% dof=PLOTDOF(j) for the load steps.
% PLTEL = Cell array of element responses.
% G2 - Matrix Structural Analysis with Matlab
% Version 0.1
% University of California, Berkeley
% Copyright 1999, Gregory L. Fenves
% --------------------------------------
% Variables used in function
% ---------------------------------------------------------
% nsteps = number of load steps
% maxiter = max. equilibrium iterations per load step
% tol = tolerance for equilibrium iterations
% ndof = number of DOF
% ntot = total number ofall DOF
% free = range of free DOF (1:ndof)
% nelem = number of elements in elemlist
% elemlist = list of elements
% loadstep = current load step
% dlamb = load increment
% lambda = load factor for current load step
% iter = counter of equilibrium iterations
% totaliter = total number of iterations (equation solves)
% nplots1 = 1 + number of DOF to plot
% nplots2 = number of elements to plot
%
% U = matrix of displacements for all DOF
% U(:,1) = current total displacement vector
% U(:,2) = displ. vector since last committed state
% U(:,3) = increment displacement vector
%
% U1old = previous solution (used only for output if solution
% does not converge)
% Pref = nodal load vector in model for all DOF
% P = scaled nodal load vector (lambda*Pref)
% P0 = restoring force vector for all DOF
% Pr = residual force vector (P-P0) for free DOF
% nresidual = 2-norm of Pr (Euclidean norm)
% Kf = tangent stiffness matrix for all DOF
% K = tangent stiffness matrix for free DOF
% cnd = condition number of K
% ---------------------------------------------------------
% Define solution parameters
nsteps = control(1);
dlamb = control(2);
maxiter = control(3);
tol = control(4);
% Get data from model
[ndof ntot Pref elemlist] = getData(mod);
free = 1:ndof;
nelem = length(elemlist);
% Initialize restoring force vector, stiffness matrix,
% and displacement vectors
P0 = zeros(ntot,1);
Kf = zeros(ntot,ntot);
U = zeros(ntot,3);
U1old = U(:,1);
% Initialize plot data for displacements and element responses
if nargin > 2 & length(plotdofs) > 0
checkDOF( mod, plotdofs );
nplots1 = length(plotdofs) + 1;
plt = zeros( nsteps+1, nplots1 );
plotdofs = plotdofs';
plotdofflag = 1;
else
plt = [];
plotdofflag = 0;
end
if nargin > 3 & length(plotelem) > 0
checkElem( mod, plotelem )
nplots2 = length(plotelem);
pltel = cell(nsteps+1,nplots2 );
plotelemflag = 1;
else
pltel = [];
plotelemflag = 0;
end
% Initialize element state
for i=1:nelem
[id xyz ue] = localize( mod, i, U);
elemlist{i} = initialize( elemlist{i}, xyz, ue, 0 );
end
% Print header for summary of load steps
printHead(mod);
disp(sprintf(['SUMMARY OF LOAD STEPS\n' ...
'Step Lambda Iters Residual\n' ...
'------------------------------------' ]));
totaliter = 0;
lambda = 0;
% Loop over load steps
for loadstep = 1:nsteps
iter = 0;
% %Modified for homework # 4:
% if loadstep<=(nsteps/4)
% lambda = lambda + dlamb;
% elseif loadstep>(3*nsteps/4)
% lambda = lambda + dlamb;
% else
% lambda = lambda - dlamb;
% end
lambda = lambda + dlamb;
P = lambda * Pref;
% Perform equilibrium iterations for load step
for i=1:maxiter
% Loop over elements to assemble tangent stiffness
% and restoring force
P0(:) = 0;
Kf(:) = 0;
for i=1:nelem
[id xyz ue] = localize( mod, i, U );
[ke pe] = state( elemlist{i}, xyz, ue, lambda );
Kf(id,id) = Kf(id,id) + ke;
P0(id) = P0(id) + pe;
end
% Partition free DOF and compute residual vector
K = Kf(free,free);
Pr = P(free) - P0(free);
% Exit equilibrium loop if 2-norm of residual vector
% is less than tolerance
nresidual = norm(Pr);
if nresidual < tol
break
end
U(free,3) = K \ Pr;
% Update displacements
U(:,1) = U(:,1) + U(:,3);
U(:,2) = U(:,2) + U(:,3);
iter = iter + 1;
end % End equilibrium iteration loop
% Check that equilibrium iteration loop converged
% If not, break load step loop after reverting to
% previous load step that did converge.
if nresidual > tol
strc = ['No convergence with residual ' num2str(nresidual) ...
' > ' num2str(tol) ' in ' int2str(maxiter) ' iterations' ...
' for load step ' int2str(loadstep) ];
disp(strc);
lambda = lambda - dlamb;
U(:,1) = U1old;
break
end
% Commit and update element state using converged displacements
for i=1:nelem
[id xyz ue] = localize( mod, i, U);
elemlist{i} = commit( elemlist{i}, xyz, ue, lambda );
end
% Update model
mod = update( mod, elemlist, P, U(:,1) );
% Zero '2' displacement vector after commit
U(:,2) = 0;
% Store total displacement for recovery if solution does
% not converge in next iteration
U1old = U(:,1);
% Store DOF values and element responses for plotting
if plotdofflag
plt(loadstep+1,1) = lambda;
plt(loadstep+1,2:nplots1) = U(plotdofs,1)';
end
if plotelemflag
pltel(loadstep+1,:) = getElemResp( mod, U, lambda, plotelem );
end
% Print status of load step
str1 = sprintf(' %3.0f %10.3e %3.0f %10.3e', loadstep, lambda, iter, nresidual );
disp(str1);
totaliter = totaliter + iter;
end % End load step loop
% Print solution after last load step
strf = sprintf(['------------------------------------\n' ...
'Total %10.3e %3.0f\n'], lambda, totaliter);
disp(strf);
printDOF ( mod, U(:,1) );
printElemResp ( mod, U, lambda );