cell array of classes

Hi,
Consider a simple class Aclass:
classdef Aclass
properties
w
end
methods
function a = Aclass(b)
a.w = b;
end
end
end
The only data member is a vector. Now say I have a very large matrix that I'd like to turn into a cell array of Aclass objects:
n = 60000; k = 20;
l = zeros(n,k);
% Convert to a cell array of Aclass objects,
sz = size(l,1);
l2 = mat2cell(l,ones(1,sz),k);
l2 = reshape(cellfun(@Aclass,l2,'UniformOutput',0),[sz 1]);
Here, the matrix l with n rows is converted into a cell array with n members, each of which is an Aclass.
My question is: is this the best way to do this? The problem is that it's quite slow when n becomes large. Is there a way to make a "custom" constructor that would return a cell array of Aclass rapidly, perhaps using a MEX file?
Many thanks,
JLT

 Accepted Answer

per isakson
per isakson on 13 Jan 2012

1 vote

R2010b doc says:
Building Arrays in the Constructor
A constructor function can return an object array by making the necessary assignments when initializing the output argument. ...

More Answers (1)

Is there any reason to use a cell array rather than an array of Aclass objects? The latter should be more efficient in both time and storage.
You can make such an array in various ways, but probably the neatest is to provide the functionality in the class constructor. Per isakson's link gives the framework. Here's what it might look like for your case:
classdef Aclass
properties
w
end
methods
function a = Aclass(b)
if nargin > 0
a(size(b,1)) = Aclass; % preallocate array
for k = 1:size(b,1)
a(k).w = b(k,:);
end
end
end
end
end
and to create the array you simply do
arr = Aclass(l);
I think this goes faster than your cell-based code, but I haven't checked very carefully.

9 Comments

Jean-Luc
Jean-Luc on 15 Jan 2012
Thanks, David and Per! This is indeed *much* faster: for some reason I thought that arrays of classes were not allowed.
But then the member functions for Aclass must all be written to potentially deal with a array? For instance, would it be reasonable to write
function disp(obj)
for i = 1:length(obj)
disp(['( ' num2str(obj(i).w) ' )']);
end
end
as a member function in Aclass, to display each row as a vector bracketed by parentheses?
Best,
Jean-Luc
Jean-Luc
Jean-Luc on 15 Jan 2012
And another quick question: what is the best way to convert *back* from an array of Aclass to a vector w. For instance, if arr is an Aclass array:
w = cell2mat({arr.w}')
works but is slow. The problem is that arr.w returns a bunch of w, on each line, rather than a 2D array. Is the solution to provide get.w explicitly?
jL
Yes, I'd go for explicit get.w, essentially reversing the constructor. Make sure to preallocate the output matrix before the loop to fill its rows.
"cell2mat({arr.w}')" looks strange to me. Why not use [arr.w]'
Jean-Luc
Jean-Luc on 16 Jan 2012
Per: you're right, [arr.w] works better. I guess it's not clear to me what the internal "status" of the object arr.w is within Matlab, and how [arr.w] makes that into a proper array.
David: could you provide me with some tip as to how to write the code for get.w? I tried but I don't see how to avoid recursion.
Best,
jL
What a.w is is a comma-separated list. It's equivalent to a(1).w, a(2).w, a(3).w (in the case where a has length 3). So [a.w] does a horizontal concatenation of the w's. However, to reconstruct your original matrix, you need vertical concatenation.
On reflection, I'd probably go for a normal method rather than get.w, as you might want to retain the possibility of getting the comma-separated list. Such a method just looks like:
function b = getw(a)
b = vertcat(a.w);
end
Jean-Luc
Jean-Luc on 17 Jan 2012
David: brilliant, thanks! I'm still grappling with some Matlab concepts, but a "comma-separated list" explains a lot.
jL
Jean-Luc
Jean-Luc on 17 Jan 2012
Update: cell2mat({arr.w}') is about the same speed (and maybe even slightly faster) that vertcat(arr.w). Weird.
Also, I don't see how one could use get.w instead of getw:
function b = get.w(a)
b = vertcat(a.w);
end
leads to an infinite recursion, I think, since a.w calls get.w.
jL
In fact, it looks like get.w does not work, but for a different reason.
There is no recursion, because within get.w a reference to a.w goes directly to the property. (Oddly enough, the documentation doesn't seem to say this for get methods, though it does say it for set methods. However, it has to be the case that recursion is avoided, or get methods just wouldn't be useful.)
The reason get.w fails is that it is called once for each element of the array, rather than being passed the whole array. (I don't understand the rationale for this.) What it boils down to is that it looks like you have to use an ordinary method like getw.

Sign in to comment.

Products

Community Treasure Hunt

Find the treasures in MATLAB Central and discover how the community can help you!

Start Hunting!