Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

When can I pass a function handle?

I have a function for cached evaluation. As one of the arguments, it takes a function handle. Under some circumstances, the function handle is unaccessible, and I don't quite understand why. The example below shows what got me stumped:

>> A.a = @plus; feval(@A.a, 1, 1)

ans =

     2

>> clear A
>> A.a.a = @plus; feval(@A.a.a, 1, 1)
Error using feval
Undefined function 'A.a.a' for input arguments of type 'double'.

So, if I have a function handle stored as a structure member, I can pass it along fine if it's one level deep, but not if it's two levels deep. In my real use case, I have a structure D that holds many (117) instances of various classes, so I actually have stct.obj.meth, where stct is a structure, obj is a class instance/object, and meth is a method. Passing @stct.obj.meth fails, but if I assign A = stct.obj, then passing @A.meth succeeds.

Under what conditions can I pass a function handle as an argument, so that it's still accessible down the stack?


Edit: Although in the use case above, I could simply remove the @ because @plus is already a function handle. However, consider the situation here:

>> type cltest.m

classdef cltest < handle
    methods
        function C = mymeth(self, a, b)
            C = a + b;
        end
    end
end

>> A.a = cltest();
>> feval(@A.a.mymeth, 1, 1)
Error using feval
Undefined function 'A.a.mymeth' for input arguments of type 'double'.
>> b = A.a;
>> feval(@b.mymeth, 1, 1)

ans =

     2

In this case, I need the @ before A.a.mymeth...

like image 330
gerrit Avatar asked Oct 16 '13 14:10

gerrit


People also ask

How do you pass a function handle in MATLAB?

To create a function handle, use the @ operator. For example, create a handle to an anonymous function that evaluates the expression x2 – y2: f = @(x,y) (x.

How does a function handle work?

Function handles are variables that you can pass to other functions. For example, calculate the integral of x2 on the range [0,1]. q = integral(f,0,1); Function handles store their absolute path, so when you have a valid handle, you can invoke the function from any location.

Can you pass a function into a function?

Functions can be passed into other functions Functions, like any other object, can be passed as an argument to another function.

Does Python have function handles?

Function handles can also be passed as arguments in Matlab. @MatlabSorter: In python a function (or a lambda) can be passed as an argument to another function.


2 Answers

Introducing classes was a big deal for MATLAB. So big, in fact, that they still do not work properly today. Your example shows that structure access and class method access conflict, because they had to overload the the meaning of dot '.' and didn't get it to work seamlessly. It all more or less works fine when you are calling class methods explicitly by their name on the MATLAB console, e.g. in your example >> A.a.mymeth(1,1). But when you have any type of indirection, it soon breaks.

You tried getting the function handle by >> @A.a.mymeth, which MATLAB cannot make sense of, probably because it gets confused by the mixed structure/class thing. Trying to work around using str2func doesn't work either. It works, again, only for explicit name access, as shown here. It breaks for your example, e.g. >> str2func('b.mymeth'). It does not even work inside the class. Try other indirections and watch them fail.

Additionally, MATLAB does not like giving you a class method's handles. There's no function for it. There's no way to get all function handles in one go, or even dynamically by a name string.

I see three options here. First, try changing your program, if possible. Do these functions need to sit in a classdef?

Second, follow your or nispio's workaround. They both create a temporary variable to hold a reference to the class instance in order to create a non-mixed access to its member methods. The problem is, they both require explicitly naming the function. You have to explicitly put this code for every function involved. No way to abstract that out.

Third, cheat by giving out your class' method handles from the inside. You can give them out in a structure.

classdef cltest < handle
    methods
        function C = mymeth(self, a, b)
            C = a + b;
        end
        function hs = funhandles(self)
            hs = struct('mymeth', @self.mymeth, ...
                        'mymeth2', @self.mymeth2);
        end
    end
end

You can then access the handles by name, even dynamically.

>> A.a = cltest;
>> feval(A.a.funhandles.mymeth, 1, 1);
>> feval(A.a.funhandles.('mymeth'), 1, 1)

ans =

     2

But be careful, by using this you can access Access=private methods from outside.

like image 109
Alex Avatar answered Sep 27 '22 16:09

Alex


Try this:

feval(@(varargin)A.a.mymeth(varargin{:}),1,1);

It is a little kludgy, but it should work.

EDIT:

The way it works is by creating an Anonymous Function that takes a variable number of arguments, and dumps those arguments into the method A.a.mymeth(). So you are not actually passing a pointer to the function A.a.mymeth, you are passing a pointer to a function that calls A.a.mymeth.

An alternative way of achieving the same thing without using varargin would be:

feval(@(x,y)A.a.mymeth(x,y),1,1);

This creates an anonymous function that accepts two arguments, and passes them along to A.a.mymeth.

<speculation> I think that it must be inherent in the way that the unary function handle operator @ works. The Matlab parser probably looks at @token and decides whether token is a valid function. In the case of a.mymeth it is smart enough to decide that mymeth is a member of a, and then return the appropriate handle. However, when it sees A.a.mymeth it may discover that A is not a class, nor does A have a member named a.mymeth and therefore no valid function is found. This seems to be supported by the fact that this works:

A.a.a = @plus; feval(A.a.a,1,1)

and this doesn't:

A.a.a = @plus; feval(@A.a.a,1,1)

</speculation>

like image 31
nispio Avatar answered Sep 27 '22 17:09

nispio