Я использую функцию fzero для решения нелинейного уравнения, зависящего от одного параметра, и меня не устраивает мой метод. У меня есть следующие проблемы:
1) Можно ли избежать цикла for для параметра?
2) Чтобы избежать сложных решений, я сначала должен предварительно вычислить допустимый интервал для fzero. Есть ли лучшее решение здесь?
Если я уменьшу размер шага параметра, время выполнения станет медленным. Если я не вычисляю интервал заранее, я получаю сообщение об ошибке «Значения функции в конечных точках интервала должны быть конечными и действительными». в Matlab и «fzero: неверный начальный брекетинг» в Octave.
Вот код
% solve y = 90-asind(n*(sind(90-asind(sind(a0)/n)-y)))
% set the equation paramaters
n=1.48; a0=0:0.1:60;
% loop over a0
for i = 1:size(a0,2)
% for each a0 find where the argument of outer asind()
% will not give complex solutions, i.e. argument is between 1 and -1
fun1 = @(y) n*(sind(90-asind(sind(a0(i))/n)-y))-0.999;
y1 = fzero(fun1,[0 90]);
fun2 = @(y) n*(sind(90-asind(sind(a0(i))/n)-y))+0.999;
y2 = fzero(fun2,[0 90]);
% use y1, y2 as limits in fzero interval
fun3 = @(y) 90-asind(n*(sind(90-asind(sind(a0(i))/n)-y)))-y;
y(i) = fzero(fun3, [y1 y2]);
end
% plot the result
figure; plot(y); grid minor;
xlabel('Incident ray angle [Deg]');
ylabel('Lens surface tangent angle');