9

编写一个子类dynamicprops允许我向对象动态添加属性:

addprop(obj, 'new_prop')

这很棒,但我也很想set / get为这些属性动态创建函数。或对这些动态属性起作用的分析函数。

到目前为止,我对 Matlab 的经验是,一旦我创建了一个类的实例,就不可能添加新方法。这非常麻烦,因为我的对象可能包含大量数据,每次我想添加新方法时都必须重新加载(因为我必须这样做clear classes)。

那么有没有办法即时添加方法?

4

3 回答 3

10

您不能像添加动态属性那样添加方法。但是,有两种方法可以在开发过程中实现新方法,不需要您每次都重新加载数据。

(1) 我将标准方法编写为单独的函数,并myMethod(obj)在开发过程中调用它们。一旦我确定它们是稳定的,我就将它们的签名添加到类定义文件中——这clear classes当然需要一个 .

(2) 使用 set/get 方法,事情就有点棘手了。如果您使用dynamicprops添加新属性,您也可以指定它们的 set/get 方法,但是(这些方法/函数很可能希望接收属性的名称,以便他们知道要引用什么):

addprop(obj,'new_prop');
prop = findprop(obj,'new_prop');
prop.SetMethod = @(obj,val)yourCustomSetMethod(obj,val,'new_prop')

编辑

(2.1) 下面是一个如何设置隐藏属性来存储和检索结果的示例(基于jmlopez 的回答)。显然,如果您对实际设计的内容有更好的了解,则可以大大改善

classdef myDynamicClass < dynamicprops
    properties (Hidden)
        name %# class name
        store %# structure that stores the values of the dynamic properties
    end
    methods
        function self = myDynamicClass(clsname, varargin)
            % self = myDynamicClass(clsname, propname, type)
            % here type is a handle to a basic datatype.
            self.name_ = clsname;
            for i=1:2:length(varargin)
                key = varargin{i};
                addprop(self, key);
                prop = findprop(self, key);
                prop.SetMethod = @(obj,val)myDynamicClass.setMethod(obj,val,key);
                prop.GetMethod = @(obj)myDynamicClass.getMethod(obj,key);
            end
        end
        function out = classname(self)
            out = self.name_;
        end
    end
    methods (Static, Hidden) %# you may want to put these in a separate fcn instead
        function setMethod(self,val,key)
           %# have a generic test, for example, force nonempty double
           validateattributes(val,{'double'},{'nonempty'}); %# will error if not double or if empty

           %# store
           self.store.(key) = val;

        end
        function val = getMethod(self,key)
           %# check whether the property exists already, return NaN otherwise
           %# could also use this to load from file if the data is not supposed to be loaded on construction 
           if isfield(self.store,key)
              val = self.store.(key);
           else
              val = NaN;
           end
        end
    end
end
于 2012-08-08T10:51:40.683 回答
3

我添加这个答案是因为我认为这不直观。至少此刻对我自己不是。找到这个问题后,我认为我拥有了为我的动态类定义 set/get 方法所需的东西。我想要实现的只是类似于 python 使用它的__setattr__方法所做的事情。无论如何,这是@jonas 不久前创建的类的延续,并进行了一些修改以添加我们的自定义 set 方法。

classdef myDynamicClass < dynamicprops
    properties (Hidden)
        name_ %# class name
    end
    methods
        function self = myDynamicClass(clsname, varargin)
            % self = myDynamicClass(clsname, propname, type)
            % here type is a handle to a basic datatype.
            self.name_ = clsname;
            for i=1:2:length(varargin)
                key = varargin{i};
                addprop(self, key);
                prop = findprop(self, key);
                prop.SetMethod = makefunc(key, varargin{i+1});
            end
        end
        function out = classname(self)
            out = self.name_;
        end
    end
end

function h = makefunc(key, argtype)
    h = @newfunc;
    function newfunc(obj, val)
       obj.(key) = argtype(val); 
    end
end

通过这个类,我定义了 set 方法,以便将传递给属性的参数复制到正确的类型。要了解我的意思,请考虑以下用法:

>> p_int = myDynamicClass('Point', 'x', @int8, 'y', @int32);
>> p_int.x = 1000 

p_int = 

  myDynamicClass with properties:

    y: []
    x: 127

>> class(p_int.x)

ans =

int8

有了这个,我们强制x属性为 8 位整数,它只能保存从 -128 到 127 的整数。还要注意每个属性的类如何为我们提供预期的类型。

于 2013-12-28T04:23:46.297 回答
2

到目前为止,我对 Matlab 的经验是,一旦我创建了一个类的实例,就不可能添加新方法。这非常麻烦,因为我的对象可能包含大量数据,每次我想添加新方法时都必须重新加载(因为我必须这样做clear classes)。

对于这个问题的当今读者来说,值得注意的是,这不再是正确的。从 MATLAB R2014b 开始, MATLAB 会在您保存类定义时更新它们,并且现有类实例的行为会相应地自动更新。在添加新方法的情况下,这并不复杂:新方法可以简单地用于调用类实例,即使它们是在方法添加到类之前创建的。

为动态属性选择 set/get 方法给出的解决方案仍然适用。

在某些情况下,您可能希望动态地向实例添加方法,并且该方法不构成属性 set/get 方法。我认为在这种情况下唯一的答案是将函数句柄作为值分配给动态属性。这不会创建真正的方法,但允许您以与方法调用相同的方式调用它:

addprop(obj, 'new_method');
obj.new_method = @(varargin) my_method(obj,varargin{:});

调用obj.new_method(args)因此传递给my_method; 但是这只适用于标量obj;实例数组将具有单独的new_method属性值,因此obj.new_method不再解析为可以调用的单个函数句柄,如果obj是一个数组。

于 2019-09-06T15:51:34.100 回答