【问题标题】:How can I overload find function in Matlab如何在 Matlab 中重载查找函数
【发布时间】:2013-08-13 07:48:39
【问题描述】:

我可以在 Matlab 中重载 find 函数吗?

如您所知:ind = find(X) 定位数组 X 的所有非零元素,并返回向量 ind 中这些元素的线性索引。

但是当 X 是对象数组时,如何使用 find 函数根据这些属性查找元素?

【问题讨论】:

  • 是否要使用相同的函数名“FInd”?
  • 我想使用 matlab 的 find 函数来查找具有指定属性的对象。这是使用 sort 函数对对象数组进行排序的示例。 mathworks.com/matlabcentral/newsreader/view_thread/292163
  • 避免混淆并使用新名称,如 findproperties 或 myobjsearch

标签: matlab overloading


【解决方案1】:

正如 Lucius Domitius Ahenobarbus 在他提供的链接中指出的那样,对于何时可以在 matlab 中重载函数有严格的规定。

参加以下课程:

classdef myclass
    methods
        function find(self)
            fprintf('find(myclass) has been called.\n');
        end
    end
end

然后执行

X = [myclass myclass myclass]
find(X)

这给出了以下输出:

X = 

  1x3 myclass with no properties.
  Methods

find(myclass) has been called.

您在重载的 find 函数中做什么取决于您自己。只需编写一个函数,打印出与您所谓的“这些属性”相匹配的元素的索引,不管是什么。

【讨论】:

  • 问题-> 如果您在课堂上使用另一个查找功能会发生什么?如果可以在没有错误的情况下继续写入,那么将调用哪一个并且它仍然会重载还是只是替换旧定义?
  • @LuciusDomitiusAhenobarbus 在myclass 中只能使用一个find 方法。如果我没记错的话,在调用find(foo) 时,matlab 会检查foo 的类型。如果它是一个具有find 方法的类,那么它会被调用。否则会调用免费的find 函数。如果搜索路径中有多个 find 函数,它会选择找到的第一个。因此,对于不是myclass 的所有内容,您都会得到旧的行为。仅当第一个参数的类型为 myclass 时,才会调用自定义 find。我个人不会超载find,而是eqgt 等并使用标准find
  • 请看看我编辑的答案。恕我直言,您的回答与超载无关!您只需将函数 find 替换为仅被调用的类函数,因为它是函数优先顺序找到的第一个函数。从我的角度来看,Mathworks 文档是错误的或至少具有误导性。如果重载的定义如我的回答中所述,您不能真正重载 Matlab 中的函数...
  • 请继续输入a=myclass; b=1:10; find(a); find(b)。你发现了什么?它会打印一次或两次find(myclass) has been called吗?我是否通过完全覆盖内置功能使自定义查找功能成为主导功能?您是否希望find(a) 使用更通用的内置find 而不是方法?我挑战你对超载的定义。我将使用“相同的名称,但根据参数调用不同的函数”。根据我的经验,维基百科所说的内容提供了一个大致的概念,但既不是最终权威,也不是最后一个字母总是正确的。
  • 技术 nitpick:Matlab 方法可以在第一个参数的类型或任何参数的类型上调度,具体取决于使用的调用语法。对于变量x、函数名f和其他参数abc,如果你做x.f(a, b, c),那么只考虑x的类型。但是,如果您使用替代语法f(x, a, b, c),则考虑所有输入的类型,参数顺序和superiorto/precedence 关系决定了它在哪个对象上分派。 (mathworks.com/help/matlab/matlab_oop/class-precedence.html)
【解决方案2】:

您可以在这里做几件事:重载find,或弹出您要搜索的条件并在这些条件上调用常规find。在属性访问表达式上使用正则查找可能更有意义。

属性访问表达式

要将find 应用于对象或其他结构,您可以使用属性访问语法创建逻辑表达式,以识别满足您正在查找的条件的对象,并将它们传递给find。假设您的班级有一个 qty 属性,这就是您要搜索的内容。

ind = find( [X.qty] ~= 0 );

您可以组合这些逻辑表达式来进行更复杂的搜索。

ind = find( [X.ghz] > 3 && [X.cacheMB] > 2 && [X.price] < 600 )

重载 find()

如果您的对象方法将具有相似的语义,您可能应该只重载众所周知的 Matlab 函数,如 findfind 函数接受一个逻辑数组并返回数字索引。因此,如果该类的元素本身在某种意义上可以被视为零或非零值,那么它可能仅对您的类有意义。

要重载一个函数来处理你的类,只需在你的类中定义一个与函数同名的方法。为了与其他 Matlab 代码一起工作,除了允许对象的实例之外,它可能应该接受与常规函数相同的典型参数。

假设您的班级将二维空间中的点表示为 (X,Y) 坐标,并且您希望将原点 (0,0) 处的点视为零,而所有其他点均不为零。您将提供一个 find 方法来测试这两个点。为了使行为与 Matlab 的 find 一致,您可以在代码中实现非零测试,并将其他所有内容传递给常规的 find 函数。

class point
    properties
        X;
        Y;
    end
    methods
        function out = find(obj)
            % Test for zero/nonzero points
            x = reshape([obj.X], size(x));
            y = reshape([obj.Y], size(y));
            isNonzero = x + y; % Quantity is not meaningful, but covers zero/nonzero/NaN
            out = find(isNonzero);
        end
    end
end

为了与find 完全一致,它有点复杂,因为find 支持额外的输入和输出参数,重载方法也应该支持。

class point
    properties
        X;
        Y;
    end
    methods
        function varargout = find(obj, varargin)
            varargout = cell(1, max(nargout, 1));
            % TODO: In production code, verify that varargin does not 
            % contain @point objects, to avoid infinite recursion

            % Test for zero/nonzero points
            x = reshape([obj.X], size(obj));
            y = reshape([obj.Y], size(obj));
            isNonzero = x + y; % Quantity is not meaningful, but covers zero/nonzero/NaN

            [varargout{:}] = find(isNonzero, varargin{:});
        end
    end
end

所有这一切都是一种痛苦,因此如果您需要对象的多态行为,您可能只想重载find:也就是说,如果您想将它们传递给编写为调用@987654338 的其他代码@ 在其输入上。如果您只需要在代码中本地输出find(),则进行属性访问可能更容易。或者您可以只提供一个isnonzero() 方法来快速转换为find() 可以很好地使用的输入。

【讨论】:

    猜你喜欢
    • 2013-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-12-26
    • 1970-01-01
    • 1970-01-01
    • 2020-08-15
    相关资源
    最近更新 更多