6

在 Matlab 类中,同时声明依赖(计算而不存储)和 Observable的属性似乎在语法上是正确的。考虑代码

properties (Access = private)
    instanceOfAnotherClass
end
properties (SetAccess = private, Dependent, SetObservable)
    propertyTwo
end
methods
    function val = get.propertyTwo(this)
        val = this.instanceOfAnotherClass.propertyOne;   
    end
end

这是否按预期工作?也就是说,如果propertyOne存储的对象的instanceOfAnotherClass属性发生更改,是否会触发属性更改事件propertyTwo?请注意,这propertyOne不是Observable

编辑: 它不起作用(如我所料)。'PostSet' 事件未触发。那么我该如何处理这种情况呢?是否有更好的解决方案创建propertyTwo为非依赖项并将其设置为每次“propertyOne”更改时与“propertyOne”相同的值?

Edit2: 针对Amro 对其答案的编辑,我将解释更复杂的情况。考虑这两个类:

 classdef AClass < handle
     properties
         a
     end
 end
 classdef BClass < handle
     properties (Access = private)
         aClassInst
     end
     properties (Dependent, SetObservable, SetAccess = private)
         b
     end
     methods
         function this = BClass(aClass)
             this.aClassInst = aClass;
         end
         function val = get.b(this)
             val = this.aClassInst.a;
         end
     end
 end

使用所有这些代码的类不应访问AClass. 它只与实例交互BClass并想监听属性的变化b。但是,如果我使 observable 的属性a无法解决AClass我的问题,会吗?'PostSet' 事件不会传播到 property b,是吗?

4

1 回答 1

3

它可能在语法上是正确的,但侦听器回调永远不会执行。例子:

classdef MyClass < handle
    properties (Access = public)
        a
    end
    properties (SetAccess = private, Dependent, SetObservable)
        b
    end
    methods
        function val = get.b(this)
            val = this.a;
        end
    end
end

现在尝试:

c = MyClass();
lh = addlistener(c, 'b', 'PostSet',@(o,e)disp(e.EventName));
c.a = 1;
disp(c.b)

正如你所看到的,'PostSet' 回调永远不会被执行。


编辑

在我看来,SetObservable真的应该设置为anot b。因为它b是只读的,只有在更改时才能a更改。现在该PostSet事件将通知我们两个属性都已更改。

使用我上面使用的相同示例,只需SetObservable从移动ba。当然,现在您可以这样收听该事件:

lh = addlistener(c, 'a', 'PostSet',@(o,e)disp(e.EventName));

编辑#2

对不起,我没有注意到你有组合(BClass 有一个 AClass 的实例作为私有属性)。

考虑这个可能的解决方案:

A类.m

classdef AClass < handle
    properties (SetObservable)
        a                        %# observable property
    end
end

B类.m

classdef BClass < handle
    properties (Access = private)
        aClassInst               %# instance of AClass
        lh                       %# event listener on aClassInst.a
    end
    properties (Dependent, SetAccess = private)
        b                        %# dependent property, read-only
    end
    events (ListenAccess = public, NotifyAccess = private)
        bPostSet                 %# custom event raised on b PostSet
    end
    methods
        function this = BClass(aClass)
            %# store AClass instance handle
            this.aClassInst = aClass;
            %# listen on PostSet event for property a of AClass instance
            this.lh = addlistener(this.aClassInst, 'a',  ...
                'PostSet', @this.aPostSet_EventHandler);
        end
        function val = get.b(this)
            val = this.aClassInst.a;
        end
    end
    methods (Access = private)
        function aPostSet_EventHandler(this, src, evt)
            %# raise bPostSet event, notifying all registered listeners
            notify(this, 'bPostSet')
        end
    end
end

基本上我们将aAClass 的属性设置为可观察的。

接下来在 BClass 的构造函数中,我们为传递的 AClass 实例注册一个侦听器,以侦听属性a更改。在回调中,我们通知监听器这个对象b也发生了变化

由于我们无法真正PostSet手动引发,我创建了一个自定义事件bPostSet,我们在之前的回调函数中引发了该事件。您始终可以自定义传递的事件数据,请参阅文档以了解如何。

这是一个测试用例:

%# create the objects
a = AClass();
b = BClass(a);

%# change property a. We will not recieve any notification
disp('a.a = 1')
a.a = 1;

%# now lets listen for the 'bChanged' event on b
lh = addlistener(b, 'bPostSet',@(o,e) disp('-- changed'));

%# try to change the property a again. We shall see notification
disp('a.a = 2')
a.a = 2;

%# remove event handler
delete(lh)

%# no more notifications
disp('a.a = 3')
a.a = 3;

输出是:

a.a = 1
a.a = 2
-- changed
a.a = 3

请注意,当我们注册我们的侦听器时,我们只与 BClass 实例交互。当然,由于所有类都派生自handle类,因此实例a和私有属性aClassInst都引用同一个对象。因此,任何更改都会a.a立即反映在 上b.aClassInst.a,这会导致内部aPostSet_EventHandler执行,进而通知所有注册的侦听器到我们的自定义事件。

于 2012-07-25T15:23:18.460 回答