我有一个文件funcs.m,用于存储匿名函数。它们必须可由所在目录中的文件使用。当前,我使用匿名函数,以便在不同文件中执行文件funcs.m
,但是我认为这是错误的处理方式。其他函数(例如main.m及其嵌套函数nest.m)需要使用funcs.m中的匿名函数。我认为路径无法解决此问题,因为文件位于同一文件夹中。基本上,我可以通过将匿名函数复制粘贴到每个文件(但有代码气味)来解决此问题:
在Matlab中是否可以重用具有anon函数的funcs.m?
例
主目录
function main
funcs; % loads the anonymous functions
nest(par1,...,parN)
end
nest.m
function nest(par1,...,parN)
funcs; %ERRR: This fires err, why? Look: this was sourced already in main.m!
function neededOnlyHere(par100)
bq(q,A) %This needs the functions of the funcs
end
neededOnlyHere(somePar) %ERR to use the anon funcs from funcs
end
函数main.m和nest.m使用此函数funcs.m具有匿名函数
bq=@(q,A) q*A; %Bolded q
I=@(ii,jj,A) find(A(ii,:)==1 & A(jj,:)==0);
AiNotj=zeros(1,Ncut);
...
错误
Attempt to add "bq" to a static workspace.
See MATLAB Programming, Restrictions on
Assigning to Variables for details.
Error in funcs (line 10)
bq=@(q,A) q*A;
%Bolded q
最佳答案
为什么要打破
在nest.m
中调用它时会出现错误,因为具有嵌套函数会使其封闭函数的工作空间成为“静态工作空间”。也就是说,不能通过eval()
,assignin()
或其他“动态”技术添加变量名;仅允许在该函数的文本中明确分配的变量。评估脚本以定义局部变量(调用funcs.m
时正在执行的操作)是“动态的”,因此在带有嵌套函数的函数中禁止使用。它在main.m
中工作,因为main没有嵌套函数,因此是“动态”工作空间。
您可以通过几种方法将其更改为与静态工作区和嵌套函数一起使用。首先要问的是,您是否真的需要使它们具有匿名功能?
改用包函数
如果您不需要它们本身就是匿名函数,只需将它们分解并作为常规函数放在其自己的.m
文件中即可;例如bg.m
,I.m
,AiNotj.m
等。然后,它们对于该目录中的所有其他功能均可用。
如果结果变成一堆文件,或者您想对它们进行范围调整,并且可能仅将其提供给真正需要它们的选定函数(即,当前调用funcs()
的函数),则可以将其固定在其中一袋。创建一个名为+myfuncs
的子目录,并将所有小功能文件移到其中。例如+myfuncs/bq.m
,+myfuncs/I.m
,+myfuncs/AiNotj.m
。 (+
前缀告诉Matlab目录是一个包。)然后,您可以通过执行import myfuncs.*
来直接替换所有当前调用funcs()
的位置,将它们全部拉入函数范围。
function nest(par1,...,parN)
import myfuncs.*;
function neededOnlyHere(par100)
bq(q,A) % This will work, resolving to myfuncs.bq
end
您也可以从命令行执行
import myfuncs.*
,使其也可以交互使用。Matlab本身可能就是这样希望您组织这样的相关功能的集群,这将是我的第一种方法。这是最不“臭”的恕我直言。为了方便起见,如果您真的希望能够将它们全部编辑在一个类似
funcs.m
的文件中,则可以在Perl或任何解析funcs.m的文件中编写一点代码,然后将它们作为等效的单个函数输出,作为预处理步骤。 。 (我觉得您不能像这样在M文件中定义多个顶级函数有点不高兴,但是,哦。)如果您确实需要使用匿名函数,则有一些解决方法。
在结构中传递函数
您可以更改funcs()函数,以使用字段名而不是局部变量名实际返回所有这些匿名函数的结构。
function out = funcs
out.bq=@(q,A) q*A; %Bolded q
out.I=@(ii,jj,A) find(A(ii,:)==1 & A(jj,:)==0);
out.AiNotj=zeros(1,Ncut);
为此,您必须为所有函数引用加上要保留它们的结构名称的前缀。不知道这对您有多大的影响。
function nest(par1,...,parN)
fs = funcs;
function neededOnlyHere(par100)
fs.bq(q,A) %This needs the functions of the funcs
end
预分配变量
为了使funcs()照原样工作,您可以使用将要使用的所有函数名称来静态地预分配变量,因此Matlab解析器会将它们识别为静态分配的变量。然后,当您调用funcs()时,它将重新分配现有变量的值,这在动态工作空间中是允许的。
function nest(par1,...,parN)
[bq, I, AiNotj] = deal(); % Preallocate all names from funcs
funcs;
function neededOnlyHere(par100)
bq(q,A) %This needs the functions of the funcs
end
这会有些麻烦,因为每当添加新的函数名称时,您都必须重新编辑每个使用func的文件。您至少可以编写一个perl脚本,通过解析
funcs.m
并输出“ [bg,I,AiNotj,...] = deal();”来自动生成该行代码。以及它找到的所有功能,您可以将其复制到代码中。另一种方法是让func实际上返回其输出列表中的所有函数。只要您不删除或更改现有匿名函数的顺序,即使将新功能添加到
funcs.m
,这也将具有继续工作的好处。function [bg,I,AiNotj] = funcs()
bg = ...
I = ...
% And then in the calling functions:
[bg,I,AiNotj] = funcs(); % which you can copy and paste from funcs.m's header
关于matlab - 如何在Matlab中嵌套匿名函数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16974698/