Ошибка использования fzero в Matlab: неопределенная функция или метод 'det' для входных аргументов типа 'function_handle' - PullRequest
1 голос
/ 26 мая 2011

У меня такая же проблема, описанная в этом разделе: Использование fzero: неопределенная функция или метод 'isfinite' для входных аргументов типа 'sym'

Их ответы действительно помогли мне, но я все еще застрял.

Я также должен найти нули функции w, эта функция определяется в несколько шагов:

Таким образом, единственное неизвестное - это w, и я определилдругие объекты, такие как:

lambda= @(w) ((16*rho(i)*A(i)*w^2*Lprime(i)^2)/(E(j)*I(i)))^0.25;

beta=@(w) lambda*b(i)^0.5;

gamma=@(w) lambda*Lprime(i)^0.5;

Затем я определяю матрицу 4 * 4 M2:

M2=@(w) [besselj(4,beta) bessely(4,beta) besseli(4,beta) besselk(4,beta);
               besselj(3,beta) bessely(3,beta) besseli(3,beta) -besselk(3,beta);
               besselj(2,gamma) bessely(2,gamma) besseli(2,gamma) besselk(2,gamma);
               besselj(4,gamma) bessely(4,gamma) besseli(4,gamma) besselk(4,gamma)];

Тогда уравнение, которое должно быть решено: det (M2) = 0.Но w = 0 является одним из решений, и я хочу первое ненулевое решение, поэтому я написал:

delta = @(w) det(M2);

S(i,j)=fzero(delta,500);

Затем я запускаю программу, и Matlab говорит:

??? Error using ==> fzero at 235
FZERO cannot continue because user supplied function_handle ==> @(w)det(M2)
failed with the error below.

Undefined function or method 'det' for input arguments of type 'function_handle'.

Error in ==> frequencies at 57
    S(i,j)=fzero(delta,500);

Я также попробовал с подпрограммами и методами eval, и они тоже не работают, сообщения об ошибках в этих случаях:

??? Undefined function or method 'isfinite' for input arguments of type 'sym'.

Error in ==> fzero at 323
    elseif ~isfinite(fx) || ~isreal(fx)

Error in ==> frequencies at 58
    S(i,j)=fzero(@(w) subs(delta,'w',w),500);

Что является той же ошибкой, что и edio, я думаю.И:

??? Error using ==> fzero at 307
FZERO cannot continue because user supplied function_handle ==> @(w)eval(delta)
failed with the error below.

Undefined function or method 'eval' for input arguments of type 'function_handle'.

Error in ==> frequencies at 59
    S(i,j)=fzero(@(w)eval(delta),500);

Можете ли вы мне помочь, пожалуйста?

Ответы [ 2 ]

3 голосов
/ 26 мая 2011

Ваша проблема в том, что вы никогда не оцениваете ваши анонимные функции, когда вы помещаете их в другие анонимные функции. Например, вы определяете функцию lambda так:

lambda = @(w) ((16*rho(i)*A(i)*w^2*Lprime(i)^2)/(E(j)*I(i)))^0.25;

Но когда вы используете его в beta, вам нужно оценить его, используя входное значение для w, например:

beta = @(w) lambda(w)*b(i)^0.5;
                %# ^--------------Pass w to lambda to evaluate the function

Поэтому я считаю, что остальные ваши анонимные функции должны быть определены следующим образом:

gamma = @(w) lambda(w)*Lprime(i)^0.5;

M2 = @(w) [besselj(4,beta(w)) bessely(4,beta(w)) besseli(4,beta(w)) ...
           besselk(4,beta(w)); ...
           besselj(3,beta(w)) bessely(3,beta(w)) besseli(3,beta(w)) ...
           -besselk(3,beta(w)); ...
           besselj(2,gamma(w)) bessely(2,gamma(w)) besseli(2,gamma(w)) ...
           besselk(2,gamma(w)); ...
           besselj(4,gamma(w)) bessely(4,gamma(w)) besseli(4,gamma(w)) ...
           besselk(4,gamma(w))];

delta = @(w) det(M2(w));


Замечание об эффективности ...

Существует проблема эффективности GLARING , которую я здесь замечаю. Используя анонимные функции вместо функций любого другого типа ( основные функции , вложенные функции или подфункции ), вы собираетесь завершить оценка одной и той же функции с одним и тем же вводом несколько раз.

Например, каждый раз, когда вы оцениваете M2 для создания своей матрицы, вы будете оценивать как beta, так и gamma 8 раз с одним и тем же вводом! Обратите внимание на улучшение, которое вы могли бы сделать, поместив M2 в функцию и передав в качестве ввода w и два дескриптора функции beta и gamma:

function newMatrix = M2(w,betaFcn,gammaFcn)

  bw = betaFcn(w);   %# Evaluate the beta function once
  gw = gammaFcn(w);  %# Evaluate the gamma function once
  newMatrix = [besselj(4,bw) bessely(4,bw) besseli(4,bw) besselk(4,bw); ...
               besselj(3,bw) bessely(3,bw) besseli(3,bw) -besselk(3,bw); ...
               besselj(2,gw) bessely(2,gw) besseli(2,gw) besselk(2,gw); ...
               besselj(4,gw) bessely(4,gw) besseli(4,gw) besselk(4,gw)];

end

И ваша новая функция delta будет выглядеть так:

delta = @(w) det(M2(w,beta,gamma));
0 голосов
/ 27 мая 2011

Привет большое спасибо за вашу помощь. Это работает, но последняя строка, очевидно, должна измениться (мне понадобилось 10 минут, чтобы понять это):

lambda= @(w) ((16*rho(i)*A(i)*w^2*Lprime(i)^2)/(E(j)*I(i)))^0.25;
beta=@(w) lambda(w)*b(i)^0.5;
gamma=@(w) lambda(w)*Lprime(i)^0.5;


M2=@(w) [besselj(4,beta(w)) bessely(4,beta(w)) besseli(4,beta(w)) besselk(4,beta(w));
    besselj(3,beta(w)) bessely(3,beta(w)) besseli(3,beta(w)) -besselk(3,beta(w));
    besselj(2,gamma(w)) bessely(2,gamma(w)) besseli(2,gamma(w)) besselk(2,gamma(w));
    besselj(4,gamma(w)) bessely(4,gamma(w)) besseli(4,gamma(w)) besselk(4,gamma(w))]; 

delta = @(w) det(M2(w));
    S(i,j)=fzero(@(w) delta(w),500);

И теперь это действительно быстрее, чем раньше, в другом случае, когда функция решения может обработать разрешение, это заняло около 10 секунд для каждого цикла, теперь это похоже на 0,06 секунды

Я попробую ваше другое решение, чтобы увидеть улучшения.

Большое спасибо.

...