我有一个代码,它由一个包含多个函数的文件组成,其中一些函数使用persistent
变量。为了使其正常工作,持久变量必须为空。
有记录的方法可以清除多功能文件中的持久变量,例如:
clear functionName % least destructive
clear functions % more destructive
clear all % most destructive
不幸的是,我不能保证用户persistent
在调用函数之前记得清除变量,所以我正在探索在代码开头执行清除操作的方法。为了说明问题,请考虑以下示例:
function clearPersistent(methodId)
if ~nargin, methodId = 0; end
switch methodId
case 0
% do nothing
case 1
clear(mfilename);
case 2
eval(sprintf('clear %s', mfilename));
case 3
clear functions;
case 4
clear all;
end
subfunction();
subfunction();
end
function [] = subfunction()
persistent val
if isempty(val)
disp("val is empty");
val = 123;
else
disp("val is not empty");
end
end
首次运行时,我们得到:
>> clearPersistent
val is empty
val is not empty
我希望此时再次运行该函数,任何非 0 输入都会导致val
变量被清除,但可惜 - 情况并非如此。设置后val
,除非我们在外部使用顶部代码段中显示的替代方案之一,或修改.m
文件,否则它保持设置。
我的问题:是否可以从主函数的主体中清除子函数中的持久变量,如果可以,如何?
换句话说,我正在寻找一些可以在clearPersistent
调用子函数之前放入的代码,以便输出始终如一:
val is empty
val is not empty
附言
这是一个相关的过去问题(不涉及此特定用例):List/view/clear persistent variables in Matlab。
我知道重写代码以根本不使用
persistent
变量的可能性(例如,通过传递数据、使用appdata
、向所有子函数添加'clear'
标志等)。请注意,编辑函数的源代码并保存会隐式清除它(以及所有持久变量)。
我知道文档指出“该
clear
函数不会清除本地或嵌套函数中的持久变量。
问题的其他背景:
实际代码结构如下:
Main function (called once)
└ Global optimization solver (called once)
└ Objective function (called an unknown N≫1 times)
└ 1st function that uses persistents
└ 2nd function that uses persistents
正如评论中提到的,有几个原因使某些变量保持不变:
- 松散耦合/SoC:目标函数不需要知道子函数是如何工作的。
- 封装:它是一个实现细节。持久变量不需要存在于使用它们的函数范围之外(即没有其他人需要它们)。
- 性能:持久变量包含计算成本相当高的矩阵,但每次调用主函数时,此操作只需发生一次。
使用持久变量的一个(副作用?)效果是使整个代码有状态(有两种状态:在昂贵的计算之前和之后)。最初的问题源于这样一个事实,即在调用 main 函数之间未正确重置状态,导致运行依赖于使用先前(因此无效)配置创建的状态。
可以通过计算 main 函数中的一次性值(目前只解析用户提供的配置,调用求解器,最后存储/显示输出),然后将它们与用户配置一起传递到目标中来避免有状态函数,然后将它们传递给子函数。这种方法解决了状态清除问题,但损害了封装性并增加了耦合,进而可能损害可维护性。
不幸的是,目标函数没有表示'init'
etc.的标志,所以我们不知道它是第一次调用还是第 n次调用,而我们自己没有跟踪这一点(AKA state)。
理想的解决方案将具有几个属性:
- 计算一次昂贵的数量。
- 无国籍。
- 不传递不相关的数据(即“需要知道基础”;各个功能工作区只包含他们需要的数据)。