How can you create local variables from class properties?
Show older comments
Is there a way to automatically create local variables corresponding to a class' properties?
Lets say I have a class instance foo, with properties A, B, C. I want to create local variables A, B, and C that are copies of foo.A, foo.B, and foo.C.
foo = fooclass;
A=foo.A;
B=foo.B;
C=foo.C;
Obviously, one could simply do the above -- however, for a case with a moderate number of parameters (and different parameters per situation), it would be nice if they could all be created in one go -- kinda like a smarter deal. For bonus points, the inverse operation would also be great. i.e.:
foo.A = A;
foo.B = B;
foo.C = C;
I can imagine writing a script that will do this in a given workspace -- but I don't see a way to make it a function that will create these variables in a calling workspace. Of course this uses eval (shudder) -- and you aren't supposed to do this -- and yes, I know my whole question is an elaborate version of creating variables with eval, which everyone should know better than. However, in this case, I think/hope it could actually result in less error prone code.
pnames = parameters(foo);
for i=1:length(pnames)
eval( [ pnames{i} ' = foo.' pnames{i} ';']);
end
I guess you could write a function that would build up a long eval string and return it, you could then just eval that...
eval( foo.abomination() );
6 Comments
Steven Lord
on 1 Oct 2019
Why do you want to do this? Why not just use the class properties directly?
Stephen23
on 1 Oct 2019
"However, in this case, I think/hope it could actually result in less error prone code."
Would you care to explain your reasoning? It really is not obvious why you think/hope that.
What it would definitely result in: slower, more obfuscated code.
Rob McDonald
on 1 Oct 2019
Steven Lord
on 1 Oct 2019
Without elegant support, forcing use of the class members all the time will likely be the solution (foo.A, foo.B, etc). That just gets ugly when you need to pass 6-8 such parameters to another function.
Why not pass the object into the "another function" and let that function retrieve only the properties from the object that it requires? That way, the calling function neither knows nor cares if the function it's calling suddenly requires more information from the object to do its work.
"...forcing use of the class members all the time will likely be the solution (foo.A, foo.B, etc)"
That is exactly what I would recommend, and it seems to be the one simple solution that you are trying to avoid... Once I started doing this my own code also became easier to work with: easier to search, to follow the data flow, and identify origins of data. I see explicitly naming the class/structure/whatever as a benefit (not as a disadvantage), much like obtaining and using explicit graphics handles adds 10% work but gives a 20% benefit.
Rob McDonald
on 4 Oct 2019
Answers (1)
A better alternative to eval is to use my structvars FEX submission
It works on objects too. Example:
classdef myclass
properties
A
B
C
end
end
>> structvars(myclass)
A = S.A;
B = S.B;
C = S.C;
2 Comments
Rob McDonald
on 4 Oct 2019
Matt J
on 4 Oct 2019
I guess you could write a function that would build up a long eval string and return it, you could then just eval that... eval( foo.abomination() );
Note that the foo.abomination idea that you mention in your original post is exactly what structvars does.
eval(structvars(foo).')
Of course, I strongly advise against using it that way...
Categories
Find more on Whos in Help Center and File Exchange
Community Treasure Hunt
Find the treasures in MATLAB Central and discover how the community can help you!
Start Hunting!