-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathNewtonRhapsonFcn.m
82 lines (68 loc) · 2.96 KB
/
NewtonRhapsonFcn.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
function [XR,EA,M] = NewtonRhapsonFcn(f,xi,Niter,es)
%Autor: Rolando Valdez Guzmán
%Alias: Tutoingeniero
%Canal de Youtube: https://www.youtube.com/channel/UCU1pdvVscOdtLpRQBp-TbWg
%Versión: 3.0
%Actualizado: 3/ene/2021
% ESTA FUNCION PIDE LOS SIGUIENTES DATOS DE ENTRADA:
% f = función como un vector de caracteres ej. 'cos(x) + x'
% xi = Valor inicial de x. Este dato es un escalar.
% Niter = Número de iteraciones.
% es = Error relativo porcentual máximo.
% VARIABLES DE SALIDA:
% XR = Ultima iteración de la raíz de la función.
% ER = Ultima iteracion del error relativo.
% M = Tabla de resultados {'xi','Error relativo (%)'}
%METODOS DE SOLUCION
%Método 1: Si Niter está vacío (Niter = []) entonces se debe especificar un
%error relativo máximo para converger.
%Método 2: Si Tol está vacío (Tol = []) entonces se debe especificar un
%número máximo de iteraciones para el código. Es posible que un número muy
%grande de iteraciones cree un error y un mensaje aparecerá sugiriendo
%reducir el número de iteraciones.
%Protección contra errores en las entradas.
if nargin ~= 4
error('Se necesita definir una función, un valor inicial, un número máximo de iteraciones y un error aproximado máximo');
%Si se ingresan todos los datos de entrada, elegir un método de solución
else
if isempty(Niter) == 1
metodo = 1; Niter = 1000; disp(newline);
disp('Solución por error aproximado máximo para converger');
elseif isempty(es) == 1
metodo = 2; disp(newline);
disp('Solución por número máximo de iteraciones para converger');
elseif isempty(Niter) == 0 && isempty(es) == 0
error('"Niter" y "es" no pueden tener un dato de entrada al mismo tiempo, uno de los dos debe estar vacío (ejemplo: Niter = [])');
end
end
fs = str2sym(f);
vs = symvar(fs);
dfs = diff(fs,vs,1);
if subs(dfs,vs,xi) == 0
error('Derivada de la función igual a cero en este punto de x. Utilice otro valor inicial para x');
end
osc = 0;
for i = 1:Niter + 1
xi_2(i) = double(xi(i) - subs(fs,vs,xi(i))/subs(dfs,vs,xi(i)));
xi(i+1) = xi_2(i); fxi(i) = double(subs(fs,vs,xi(i)));
ea(i) = abs((xi_2(i) - xi(i)) / xi_2(i)) * 100; %Calcula el error relativo actual
if i >= 2
if xi(i+1) == xi(i-1) && ea(i) > es
osc = osc + 1;
if osc == 3
error('Divergencia oscilatoria detectada. Use otro valor inicial de x');
end
end
end
if i >= 30 && ea(i) > es
error('Convergencia lenta o divergencia detectada. Use otro valor inicial de x');
end
if metodo == 1
if ea(i) < es %Si el error relativo es menor a la tolerancia exigida, se acaba el ciclo.
break;
end
end
end
M1 = {'xi', 'f(xi)','Error relativo (%)'};
M2 = num2cell([xi(1:end-1)',fxi',ea']);
M = [M1; M2]; XR = xi(end); EA = ea(end);