matlab面向对象编程入门笔记

文章目录

  • 1. 类和结构
  • 2. 定义类
  • 3. 属性
    • 3.1 private/protected/public属性
    • 3.2 constant属性
    • 3.3 hidden属性
  • 4. 方法
    • 4.1 private/protected/public方法
    • 4.2 static方法
    • 4.3 外部方法
  • 5. 动态调用
  • 6. 继承-超类
    • 6.1 handle超类
    • 6.2 dynamicprops 和 hgsetget子类
  • 7. 封闭(sealed)类、方法和属性
  • 8. 抽象(abstract)方法和属性
  • 9. Operator 重载
  • 10. 类数组
  • 11. 事件events
  • 12. mataclass 元类

matlab可以支持面向对象的编程(OOP),最近使用到了,记录一下基本的语法:

1. 类和结构

matlab可以支持类和结构体,结构体的关键字是struct,它们都可以具有属性,不过和结构体不同,我们必须预定义整个类具有哪些属性,这点和C++差不多。

另外类还多了和结构体不同的方法(对数据执行的函数)。而matlab的结构体不能定义方法。

2. 定义类

首先,创建一个与要创建的类同名的新 m 文件,并使用 classdef 关键字后跟类名为开头行。属性properties和方法methods在此行下方定义。

在方法methods我们必须包含构造函数 ,它负责构造新对象,并且它必须与类同名。构造函数可以接受任意数量的参数来指定属性的初始值,并且必须返回一个构造对象的参数。属性可以设置默认值。如果未指定默认值并且构造函数也没有对属性赋值,这些属性被赋值为空矩阵 []

   classdef date% write a description of the class here.properties% define the properties of the class here, (like fields of a struct)minute = 0;hour;day;month;year;endmethods% methods, including the constructor are defined in this blockfunction obj = date(minute,hour,day,month,year)% class constructorif(nargin > 0)obj.minute = minute;obj.hour   = hour;obj.day    = day;obj.month  = month;obj.year   = year;endendfunction obj = rollDay(obj,numdays)obj.day = obj.day + numdays;endendend

我们可以按照构造函数的格式创建对象:

d1 = date(0,3,27,2,1998);

在这里插入图片描述
如果我们没有指定任何参数地创建对象,没有默认值的属性都被赋值为了[]。

d2 = date();

在这里插入图片描述

3. 属性

3.1 private/protected/public属性

上面定义的属性是public的,也就是说,可以从类外部访问:

day = d1.day;                                 % access the day property
d1.year = 2008;

在这里插入图片描述

但有时候我们不希望用户任意修改这些属性,所以我们需要对这些属性设置访问级别,matlab力提供了private、protected和public的级别。

访问级别访问权限
public任何地方访问
private只能从类的方法和公共属性访问
protected私有的,也可以从子类访问(参考后文的子类和继承)

在属性里我们还可以进行细分GetAccessSetAccess

% 外部只读属性,不可写
properties(GetAccess = 'public', SetAccess = 'private')% public read access, but private write access.
end% 外部不可读也不可写
properties(GetAccess = 'private', SetAccess = 'private')% private read and write access
end

3.2 constant属性

我们还可以分配Constant属性,让该属性块的属性是常量(默认是不是常量),并且不能在任何地方进行修改:

properties(Constant = true)DAYS_PER_YEAR =  365;MONTHS_PER_YEAR = 12;WEEKS_PER_YEAR  = 52;
end

3.3 hidden属性

另外还有Hidden,可以隐藏属性(默认是不隐藏)

properties(Hidden= true)DAYS_PER_YEAR =  365;MONTHS_PER_YEAR = 12;WEEKS_PER_YEAR  = 52;
end

完整的类的定义我们修改为了:

classdef date% write a description of the class here.properties (Hidden = false)% define the properties of the class here, (like fields of a struct)minute = 0;hour;day;month;year;endproperties(Constant = true)DAYS_PER_YEAR =  365;MONTHS_PER_YEAR = 12;WEEKS_PER_YEAR  = 52;endmethods% methods, including the constructor are defined in this blockfunction obj = date(minute,hour,day,month,year)% class constructorif(nargin > 0)obj.minute = minute;obj.hour   = hour;obj.day    = day;obj.month  = month;obj.year   = year;endendfunction obj = rollDay(obj,numdays)obj.day = obj.day + numdays;endend
end

当我们使用properties()时只能查看非隐藏的属性:

properties(date)

(隐藏了minute等属性,Hidden = true
在这里插入图片描述

(未隐藏minute等属性,Hidden = false
在这里插入图片描述

4. 方法

我们前面已经在类里定义了一个简单的方法rollDay(),前面我们已经初始化了d1:

d1 = date(0,3,27,2,1998);

我们可以使用下面两种方式进行方法的调用(大家可以自己输入命令尝试一下看属性里的day是不是发生了改变):


d1 = rollDay(d1,3); %
d1 = d1.rollDay(3);

第二种写法和C++的差不多,两种方式都是可以的,注意当我们修改任何属性的时候我们必须注意返回对象,默认情况下,Matlab 中的对象是按值传递的,而不是通过引用传递的, 这意味着对象的完整副本被传递给方法,它是已修改的此副本。如果我们不传回此副本, 有效地覆盖了原始内容,我们不会观察到任何更改。当然我们也可以选择编写对象被传递的类引用(这在后面的Handle Superclass将会介绍到)

方法也像属性一样可以有三个访问级别:private、protected或public,private方法不能用于类外部使用,把一个public的方法拆分成一系列的private的函数的调用是很有效的。

4.1 private/protected/public方法

我们通过将private、protected或public分配给Access属性定义对方法的访问,我们可以自由地创建任意数量的方法块,每个方法块都具有不同的访问属性。例如private的方法:

methods(Access = private)function sec = calcSecs(obj)sec = obj.minute*60 + obj.hour*60*60 + obj.day*24*60*60;endfunction TF = isValid(obj)TF = obj.minute >= 0 && obj.minute <= 60;end
end

4.2 static方法

static的方法是和类关联的方法,而不是与该类的实例相关联的方法:

 methods(Static = true)function printCurrentDate()display(datestr(now));endend

完整代码:

classdef date% write a description of the class here.properties (Hidden = false)% define the properties of the class here, (like fields of a struct)minute = 0;hour;day;month;year;endproperties(Constant = true)DAYS_PER_YEAR =  365;MONTHS_PER_YEAR = 12;WEEKS_PER_YEAR  = 52;endmethods% methods, including the constructor are defined in this blockfunction obj = date(minute,hour,day,month,year)% class constructorif(nargin > 0)obj.minute = minute;obj.hour   = hour;obj.day    = day;obj.month  = month;obj.year   = year;endendfunction obj = rollDay(obj,numdays)obj.day = obj.day + numdays;endendmethods(Static = true)function printCurrentDate()display(datestr(now));endendend

当方法和类相关的时候,但是不需要使用类实例的特定的信息的时候,我们可以设置属性为static。如果要调用static方法,我们需要指定类名,后面跟.和静态方法的名称(把前面的程序加入类里),调用的时候使用下面的语句都是可以的:

date.printCurrentDate()
d1.printCurrentDate()

方法也可以用 methods(Hidden = true) 隐藏,从显示类方法的函数(例如 methods()methodsview() )中隐藏它们。

4.3 外部方法

类的方法可以自由地调用任何在matlab路径的外部函数来实现计算,此外如果我们在与类同名的文件夹里保存classdef m文件,但文件夹的名字前面加了前缀@(如@date),我们就可以在这个文件夹把类的方法和声明类似C++的h文件和cpp文件分离开来。具体来说:

我们在当前文件夹创建@date文件夹,在里面放入文件date.m(类文件),然后我们可以把rollDay()函数放在外面编写,新建一个rollDay.m文件:
在这里插入图片描述
date.m类文件里我们改写关于rollDay()的声明:

methodsobj = rollDay(obj,numdays)
end

然后在rollDay.m编写rollDay()的定义

function obj = rollDay(obj,numdays)obj.day = obj.day + numdays;
end

我们可以使用同样的调用的方式得到和内部定义方法一样的效果:

d1 = date(0,3,27,2,1998);
d1 = d1.rollDay(3)

但是在类文件外定义的这些方法将无法分配前面讨论的任何访问级别(static、hidden、protected),它们是自动为public的。

如果要编写外部的私有方法,需要把它保存在名为private的子目录下,例如\@date\private\rollDay.m

在这里插入图片描述

date.m我们改写rollDay()函数(改成private属性),并且加上一个rollDayCall()函数为public属性方便我们外部调用,然后我们把前面的rollDay函数移动到private文件夹下面

methodsfunction obj = rollDayCall(obj, numdays)obj = rollDay(obj,numdays)end
end
methods(Access = private)obj = rollDay(obj,numdays)
end

我们同样可以测试一下,是可以正常运行的:

d1 = date(0,3,27,2,1998);
d1 = d1.rollDay(3)

关于这一部分的内容maltab也有相关的参考的内容:
Methods in Separate Files

5. 动态调用

OOP的一个优点是我们可以创建两个不同的类,每个类可以有相同的方法名称,matlab会自动根据传递的对象的类型调用正确的方法。例如,像 obj = increment(obj) 这样的调用将从 obj 碰巧的类调用 increment 方法,即使多个类具有 increment 方法也是如此。

当多个对象传递给一个方法时,Matlab 会根据上下关系确定要调用哪个类的方法。调用最高级类的方法,如果所有类具有相同的优越性,则最左边的对象 优先。

我们在创建类时指定这些关系,就在 classdef 语句中,如下所示:

classdef (InferiorClasses = {?ClassName1,?ClassName2}) MyClass...
end

这里的? 用来构造元类metaclass实例,但是具体的细节并不是很重要(可以看matlab文档:metaclass和Class Precedence)。class1 和 class2 是 下层类inferior 的实例。我们将在后面的部分中讨论元类meta class。

类对象数组也可以传递给方法,但类对象数组的类加和对象类存储在数组里面一样(数组的所有元素必须是同一种类类型),并且此类用于确定优先级。有关更多信息,请参阅对象数组部分。

6. 继承-超类

我们经常会发现有些类是特例,比如地球是行星类的一个特定的实例,而行星是天体类的一个子类subclass。

我们可以编写子类来继承其超类superclass的所有属性和方法,这样我们就不需要重新编写所有的父类的功能,而子类只需要扩展或专门化这个功能。我们可以将子类的方法视为它自己的所有方法和其父超类的结合,属性也是这样的。

子类可以重新定义父类的方法(在子类里编写同名的方法即可,方法参数的数量和名字也不需要和父类相同):子类版本被用于子类的对象,而父类的版本被用于父类的对象(这是另一个被用于动态调用的例子)。

当前发现在为两个或更多的类编写方法时,可以考虑使用继承,让子类拥有父类的所有共同的代码。我们可以在类的classdef语句中使用下面的语法来为我们正在编写的类指定一个超类的定义:

classdef classname < superclass

classname是当前类的名字,而superclass是超类的名字,matlab也可以支持多重继承,即具有多个超类,注意不要发生命名的冲突,要使用多个类继承,我们使用&来分割超类:

classdef classname < superclass1 & superclass2 & superclass3

当方法已经在子类重新定义,有时候我们需要从子类调用超类的版本,例如希望子类版本做超类版本可以做的事,我们不需要从超类复制代码,而可以直接调用超类的版本,我们可以使用@运算符来对超类的方法和属性进行访问:

methodname@superclassname(input1, input2)

如果没有重新定义这些超类的方法,也没有必要使用@运算符来得到这些继承的方法(尽管这也不会发生错误),可以像子类的方法和属性一样直接访问这些超类的方法和属性而不用使用@运算符。

可以使用空的classdef声明创建相同类的多个别名(继承了但是没有对超类进行任何的扩展):

classdef newclassname < oldclassname
end

6.1 handle超类

前面我们已经提到matlab的对象默认情况下是值传递,即在方法中来回传递完整的副本调用,然而matlab的图形对象是通过引用传递的(通过Handle句柄),如果我们子类化内置的句柄类,如:

classdef myclass < handle

这样我们类的对象也将通过引用传递,而不是通过值传递。这样做有很多好处。当我们构造一个句柄对象时,如 h = myclass() 中,h 存储的是指向该对象的指针或句柄,而不是对象本身。如果我们随后执行 h2 = h,我们只需创建另一个指向同一基础对象的指针。例如,我们可以调用 h.prop = 3 ,然后 p = h2.prop ,我们可以得到 p = 3

在句柄方法调用中,不需要返回对象,因为分配就地发生(返回对象的句柄不会造成任何危害)。

如果我们的对象需要使用非常大的存储空间,使用handle对象可以更加节省空间,因为我们不再需要复制整个对象到每个方法调用中。

而且只有handle类可以支持evenets(事件),在后面我们会进行讨论。

使用handle类最主要的优点在于它可以更简单地写入数据结构,尤其是那些需要递归的结构(比如二叉树),这里给出一个非常简单的二叉树实现并说明如何只通过句柄轻松地递归所有节点。

在这里插入图片描述

bnode.m

classdef bnode < handle               % subclass handlepropertiesleft;        % left  childright;       % right childdata;        % data stored at the nodeendmethodsfunction obj = bnode(data)obj;if(nargin > 0)obj.data = data;endendend
end

labelNodes.m(实现的是前序遍历)

function labelNodes(node)
% recursively label the depth of the nodesif(isempty(node))returnenddisp(node.data);labelNodes(node.left);labelNodes(node.right);
end

我们可以编写测试代码随便生成一个二叉树:

n1 = bnode(1);
n2 = bnode(2);
n1.left = n2;
n3 = bnode(3);
n2.left = n3;
n4 = bnode(4);
n2.right = n4;

然后测试一下前序遍历二叉树:

labelNodes(n1)

在这里插入图片描述

创建句柄对象的相同副本则要复杂得多,因为我们不能简单地转到 h2 = h1。但是,我们可以使用以下代码来创建任何对象的浅拷贝。它需要完全访问所有属性等所以需要被添加为一个类的方法。另一种方法是使用 struct() 函数将对象转换为结构体,然后编写构造函数以选择性地采用结构体,构建一个新对象。

function copy = copyobj(obj)
% Create a shallow copy of the calling object.copy = eval(class(obj));meta = eval(['?',class(obj)]);for p = 1: size(meta.Properties,1)pname = meta.Properties{p}.Name;tryeval(['copy.',pname,' = obj.',pname,';']);catchfprintf(['\nCould not copy ',pname,'.\n']);endend
end

拷贝的使用直接调用函数就可以了,例如:

copy = copyobj(n1)% n1是前面定义的二叉树的结点

当堆栈上没有剩余对象的句柄时,该对象将被声明为无效,并且 Matlab 垃圾回收器当有机会时会释放内存。我们可以使用 isvalid(h) 方法测试对象的句柄是否有效并使用 delete(h)删除该对象,使其所有句柄变得无效。

6.2 dynamicprops 和 hgsetget子类

handle有两个子类分别为dynamicprops 和 hgsetget,可以对它们继续进行子类化,而产生其他功能。

通过子类化dynamicprops,可以无需修改类的定义而将附加动态属性到对象上,我们只需要调用继承的addprop()函数,如p=obj.addprop('newProperty'),然后可以进行类似obj.newProperty = 3val = obj.newProperty的调用,addprop()的返回值p可以用于设置属性的属性(如‘hidden’),或通过delete(p)来删除属性。具体可以参考matlab的文档:

  • Dynamic Properties — Adding Properties to an Instance

  • dynamicprops class

hgsetget 类也是 handle 的子类,它是用于通过 set 和 get 方法派生句柄类的抽象类,在matlab图像里我们经常使用的set和get函数就是通过它来实现的,比如 set(h,'property',value)。然而matlab称在以后的版本可能会删除hgsetget。推荐改用 matlab.mixin.SetGet。另外我们还可以对内置的类型(如double)进行子类化。

7. 封闭(sealed)类、方法和属性

封闭类不能被子类化,封闭的方法或属性不能在子类里重新定义,这类似于C++的final关键字,下面是定义封闭的例子(封闭类、封闭方法和封闭属性)

classdef(Sealed = true) myclass
methods(Sealed = true)
properties(Sealed = true)

8. 抽象(abstract)方法和属性

抽象方法简单来说是具有函数头的方法,但不是函数体(还没有实现),它们可以定义当前和未来子类的通用接口(类似与C++的虚函数),抽象方法必须由子类来实现,因此子类集成超类是一种必然的条件。当然如果所有子类中的实现都是一样的,最好在超类写一个通用的方法以被子类继承。也就是说,当我们知道子类需要具有特定的方法,这些方法的实现会有所不同的时候,我们就可以使用抽象了。

比如我们有一个shape超类,它具有许多子类,如球体、立方体、椭圆体、金字塔等。我们想要每个子类(以及未来可能的子类)有一个 calculateVolume() 方法来计算这些形状的体积。每一种子类的函数的计算的方法都有所不同,但是通过shape超类的的方法的抽象,我们可以在每个子类里来具体实现这些方法。

属性也可以是抽象的,但是用处不大,它们必须在子类里定义。

一个具有一个或多个抽象的方法或属性的类被认为是抽象类,我们无法从中创建实例,任何没有实现抽象方法和属性的子类也是抽象的,实质上是将部分或全部抽象方法和属性的实现委托给了更多的子类。

我们可以使用Abstract将方法块定义为抽象,在编写函数的时候,我们可以不包含主体:

methods(Abstract = true)function vol = calculateVolume(obj,units);function area = calculateSurfaceArea(obj,units);function obj = doubleSize(obj);
end

9. Operator 重载

每次使用 Matlab 运算符,例如

+ - .* * ./ .\ / \ .^ ^ < > <= >=
== ~ ~= & | && || : ' .' [] [;] () .

实际上是调用了命名函数的简写或语法,如plus()minus()等,我们可以通过使用同名的类方法为这些操作符定义自定义行为。由于类方法是动态分派的,因此当我们对对象使用相应的运算符时,我们自己版本的这些函数就会执行。比如我们可以在date类里编写自己的plus方法,将日期相加,然后使用d1+d2调用这个函数,类调用会自动转换为plus(d1, d2),然后调用我们对这些函数的实现。具体实现是我们添加plus方法:

classdef date% write a description of the class here.properties (Hidden = false)% define the properties of the class here, (like fields of a struct)minute = 0;hour;day;month;year;endproperties(Constant = true)DAYS_PER_YEAR =  365;MONTHS_PER_YEAR = 12;WEEKS_PER_YEAR  = 52;endmethods% methods, including the constructor are defined in this blockfunction obj = date(minute,hour,day,month,year)% class constructorif(nargin > 0)obj.minute = minute;obj.hour   = hour;obj.day    = day;obj.month  = month;obj.year   = year;endendfunction obj = plus(obj1, obj2)obj = obj1;obj.day = obj1.day + obj2.day;endendmethodsfunction obj = rollDayCall(obj, numdays)obj = rollDay(obj,numdays)endendmethods(Access = private)obj = rollDay(obj,numdays)endmethods(Static = true)function printCurrentDate()display(datestr(now));endendend

我们进行一下测试,可以看到:

d1 = date(0,3,27,2,1998);
d2 = date(0,3,2,2,1998);
d = d1+d2

d最后的day是d1和d2的day的和。
在这里插入图片描述

更具体的参考我们可以看:运算符重载

运算符保留其自然优先级,如* 在运算顺序上优先于 +,即使一个或两个运算符被重载。

另一个有用的重载方法是 display() - 它是当我们不用分号禁止显示输出时自动显示对象的函数。编写我们自己的显示函数允许我们以任何我们喜欢的方式显示对象。

有时,重复使用属于内置函数(如 plot())的简明信息名称非常有用。虽然 plot 不会被自动调用,也不与操作符相对应,但它在 Matlab 中的使用频率很高,因此在我们自己的对象中重复使用这个名称可以很好地自我记录其行为(假设我们的 plot 函数做了一些合理的事情)。

如果重载了运算符或函数,但出于某种原因想要使用原始实现,请使用 builtin() 函数,该函数将函数的字符串名称作为第一个输入,然后是该函数的输入。

具体的可以参考matlab的文档:builtin

有两个非常重要的函数经常被重载,值得特别一提:subsrefsubsasgn。在索引操作中使用点操作符.、括号()或大括号 {} 时,会调用 subsref;在赋值操作中使用它们时,会调用 subsasgn。通过重载这些函数,我们可以为我们的类创建自定义行为。

例如,假设我们编写了自己的数据结构类,并希望调用 obj(3) 来获取结构中的第三个元素,我们就可以通过重载 subsref 来实现这一目标。下面是这两个函数的定义:

function obj = subsasgn(obj, S, value)
function value = subsref(obj, S)

obj 是调用对象,如 obj.prop 或 obj(3)

value 是新的或返回的值,如 obj.prop = value 或 value = obj.prop

S 是具有两个字段的结构:type 和 subs: type是"()""{}"".",具体取决于调用。subs 是cell数组或实际使用的下标字符串。

在涉及多个运算符(如 obj(5,9).prop(1:19)=value)的复杂调用中,对 subsasgn 进行一次调用,并将调用中的所有信息传递给该函数。在本例中,S 是 具有以下值的结构数组。

S(1)S(2)S(3)
S(1).type=‘()’S(2).type=‘.’S(3).type=‘()’
S(1).subs={5,9}S(2).subs=‘prop’S(3).subs={1:10}

这里也把有关的参考给列出来:

  • subsref
  • subsasgn
  • Code Patterns for subsref and subsasgn Methods
  • 对象数组索引

10. 类数组

同类对象可以一起存储在对象数组(元胞数组和结构体也是可以的),其操作方式与数字数组类似。我们可以将对象连接在一起,并以通常的方式在这些数组中建立索引。

d1 = date(0,3,27,2,1998);   % 创建第一个日期对象
d2 = date(1,4,22,3,2008);   % 创建第二个日期对象
dates = [d1 d2];            % 可以把这两个对象连接成一个数组
[nrows ncols] = size(dates) % 也可以使用数组的函数
d1 = dates(1,1);            % 检索数组第一个条目
dates(1,1) = d1;            % 赋值数组第一个条目

在这里插入图片描述
单个日期对象的类型实际上是由日期对象组成的对象数组,尽管大小为 1x1。在 Matlab 中,几乎所有东西都可以组成数组,对象也不例外。其后果之一是,涉及对象数组的方法调用会调度与只涉及一个对象相同的方法;这包括调用 subsref()subsasgn()

11. 事件events

Matlab 现在对基于事件的编程有很好的支持,在这种编程中,对象会根据状态的变化触发事件,并通知一个或多个已注册为监听器的其他对象。当适当的控制流取决于程序外部的事物(如用户与图形界面或环境传感器的交互)时,这种编程方式就特别有用。不过,它本身也是一种有用的范例,尤其是对于模拟而言。

首先,所有涉及的类都必须继承自handle类(或其子类之一)。触发类必须在其类定义中声明一个事件块。事件块的属性与方法和属性块一样,都定义了事件访问控制access control。

ListenAccess 属性决定了可以在哪里创建事件监听器,而 NotifyAccess 则决定了可以在哪里触发事件。在下面的示例中,我们将 ListenAccess 设置为 public,这样就可以在任何地方将对象注册为监听器,而将 NotifyAccess 设置为 protected,这样就只有 date 类(或 date 的任何子类)的方法可以触发事件。

这一部分不是本文的重点,具体可以参考matlab的文档:

  • 事件和侦听程序语法
  • 侦听对属性值的更改
  • 事件和侦听程序概念
  • 事件和侦听程序概述
  • 事件

12. mataclass 元类

Matlab的 有一个相当新颖的功能mataclass ,它允许动态地检查特定类的属性。每个使用 classdef 语法的类具有相应的元类,可以调用?得到元类。生成的对象存储了类的方法、属性、事件、超类等信息及其属性。元类可用于编写高度通用的代码。查看类树(viewClassTree())方法(可显示项目的完整类层次结构)广泛使用了元类,代码如下所示:

function  viewClassTree(directory)
% View a class inheritence hierarchy. All classes residing in the directory
% or any subdirectory are discovered. Parents of these classes are also
% discovered as long as they are in the matlab search path. 
% There are a few restrictions:
% (1) classes must be written using the new 2008a classdef syntax
% (2) classes must reside in their own @ directories.
% (3) requires the bioinformatics biograph class to display the tree.
% (4) works only on systems that support 'dir', i.e. windows. 
%  
% directory  is an optional parameter specifying the directory one level
%            above all of the @ class directories. The current working
%            directory is used if this is not specified.
%Written by Matthew Dunham 
if nargin == 0directory = '.';
end
info = dirinfo(directory);
baseClasses = vertcat(info.classes);
if(isempty(baseClasses))fprintf('\nNo classes found in this directory.\n');return;
end
allClasses = baseClasses;
for c=1:numel(baseClasses)allClasses = union(allClasses,ancestors(baseClasses{c}));
end
matrix = zeros(numel(allClasses));
map = struct;
for i=1:numel(allClasses)map.(allClasses{i}) = i; 
end
for i=1:numel(allClasses)trymeta = eval(['?',allClasses{i}]);parents = meta.SuperClasses;catch MEwarning('CLASSTREE:discoveryWarning',['Could not discover information about class ',allClasses{i}]);continue;endfor j=1:numel(parents)matrix(map.(allClasses{i}),map.(parents{j}.Name)) = 1;end
end
for i=1:numel(allClasses)allClasses{i} = ['@',allClasses{i}]; 
end
view(biograph(matrix,allClasses));
function info = dirinfo(directory)
%Recursively generate an array of structures holding information about each
%directory/subdirectory beginning, (and including) the initially specified
%parent directory. info = what(directory);flist = dir(directory);dlist =  {flist([flist.isdir]).name};for i=1:numel(dlist)dirname = dlist{i};if(~strcmp(dirname,'.') && ~strcmp(dirname,'..'))info = [info, dirinfo([directory,'\',dirname])]; endend
end
function list = ancestors(class)
%Recursively generate a list of all of the superclasses, (and superclasses
%of superclasses, etc) of the specified class. list = {};trymeta = eval(['?',class]);parents = meta.SuperClasses;catchreturn;endfor p=1:numel(parents)if(p > numel(parents)),continue,end %bug fix for version 7.5.0 (2007b)list = [parents{p}.Name,ancestors(parents{p}.Name)];end
end
end

在这里插入图片描述
我们在当前路径下查看一下类的层次:

viewClassTree()

在这里插入图片描述

以下示例函数查找特定类的所有超类,包括其超类的超类。metaclass() 函数的运行方式就像 ?运算符,但可以与字符串名称一起使用,而 ?需要对象的实例。

function list = ancestors(class)
% input is the string name of the base class
% output is a cell array of ancestor class nameslist = {};meta = metaclass(class);parents = meta.SuperClasses;for p=1:numel(parents)list = [parents{p}.Name,ancestors(parents{p}.Name)];end
end

在这里插入图片描述
以下是可用数据类型。

metadata = ?date

在这里插入图片描述

整理自:

  1. Object Oriented Programming in Matlab: basics
  2. Object Oriented Programming in Matlab: More advanced topics

看了这篇博客应该对基本的一些matlab面向对象编程的知识有了一定的了解,如果想继续往下深入推荐matlab面向对象编程的官方文档一起学习:

MATLAB® 2023b Object-Oriented Programming

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/284511.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

计算机网络(2):物理层

物理层的基本概念 物理层考虑的是怎样才能在连接各种计算机的传输媒体上传输数据比特流&#xff0c;而不是指具体的传输媒体。 物理层的作用是要尽可能地屏蔽掉差异&#xff08;硬件设备&#xff0c;传输媒体&#xff0c;通信手段&#xff09;&#xff0c;使物理层上面的数据链…

在centos7上安装docker

1.CentOS安装Docker Docker CE 支持 64 位版本 CentOS 7&#xff0c;并且要求内核版本不低于 3.10&#xff0c; CentOS 7 满足最低内核的要求&#xff0c;所以我们在CentOS 7安装Docker。 1.1.卸载&#xff08;可选&#xff09; 如果之前安装过旧版本的Docker&#xff0c;可…

AngularJS

理解实现代码的逻辑为主要&#xff0c;代码怎么写为次要。 参考资料&#xff1a; 《AngularJS入门与进阶》&#xff0c;江荣波著 前端开发常用框架 React&#xff1a;由Facebook开发&#xff0c;用于构建用户界面的JavaScript库&#xff0c;以组件化和虚拟DOM著称。 Angular&…

【精选】计算机网络教程(第5章运输层)

目录 前言 第5章运输层 1、用户数据报协议&#xff08;UDP&#xff09; 2、传输控制协议&#xff08;TCP&#xff09; 3、TCP/IP网络中的运输层协议 4、UDP的特点和应用 5、TCP的连接过程 6、慢启动和拥塞避免 7、快速恢复 前言 总结计算机网络教程课程期末必记知识…

Windows11编译x265源码生成Visual Studio工程详细步骤

概述 x265是一款开源符合HEVC标准的编码器&#xff0c;也属于VLC项目之一。 由于x265是开源的&#xff0c;因此它得到了广泛的应用和开发。许多开源项目和商业产品都使用x265进行视频压缩处理。同时&#xff0c;x265也支持多种编程语言和平台&#xff0c;使得开发者可以方便地…

【C语言】鹏哥C语言刷题训练营——第5节内容笔记(含代码全面分析和改进,讲解)

系列文章目录 身躯已然冰封&#xff0c;灵魂仍旧火热 本文由睡觉待开机原创&#xff0c;未经允许不得转载。 本内容在csdn网站首发 欢迎各位点赞—评论—收藏 如果存在不足之处请评论留言&#xff0c;共同进步&#xff01; 文章目录 系列文章目录前言题目链接&#xff08;有需要…

Linux shell编程学习笔记36:read命令

目录 0 前言1 read命令的功能、格式、返回值和注意 1.1 命令功能1.2 命令格式1.3 返回值1.4 注意事项2 命令应用实例 2.1 一次读入多个变量值2.2 不指定变量名2.3 测试read命令的返回值2.3 指定输入时限并进行相应处理2.4 -t 指定结束符2.5 -n 指定输入字符个数2.6 -N 指定输入…

被我们忽略的HttpSession线程安全问题

1. 背景 最近在读《Java concurrency in practice》(Java并发实战)&#xff0c;其中1.4节提到了Java web的线程安全问题时有如下一段话&#xff1a; Servlets and JPSs, as well as servlet filters and objects stored in scoped containers like ServletContext and HttpSe…

Java+Swing: 删除数据 整理15

1. 添加点击事件 2. 在MainViewHandler处理类中&#xff0c;实现相应的处理操作 if ("删除".equals(text)){int[] selectedRowIds mainView.getSelectedRowIds();if (selectedRowIds.length 0){JOptionPane.showMessageDialog(mainView, "请选择要删除的数据…

JWT令牌的作用和生成

JWT令牌&#xff08;JSON Web Token&#xff09;是一种用于身份验证和授权的安全令牌。它由三部分组成&#xff1a;头部、载荷和签名。 JWT令牌的作用如下&#xff1a; 身份验证&#xff1a;JWT令牌可以验证用户身份。当用户登录后&#xff0c;服务器会生成一个JWT令牌并返回…

Linux - 非root用户使用systemctl管理服务

文章目录 方式一 &#xff08;推荐&#xff09;1. 编辑sudoers文件&#xff1a;2. 设置服务文件权限&#xff1a;3. 启动和停止服务&#xff1a; 方式二1. 查看可用服务&#xff1a;2. 选择要配置的服务&#xff1a;3. 创建自定义服务文件&#xff1a;4. 重新加载systemd管理的…

【网络安全】-Linux操作系统—操作系统发展历史与Linux

文章目录 操作系统发展历史初期的操作系统分时操作系统个人计算机操作系统 Linux的诞生UNIX与GNU项目Linux内核的创建 Linux的特点开放源代码多样性社区支持 Linux的应用服务器和超级计算机嵌入式系统桌面系统 总结 操作系统发展历史 操作系统&#xff08;Operating System&am…