.. _methods: Functions and Methods ===================== In the :ref:`tutorial` we touched upon how to wrap C++ functions, static member functions and member functions. Now let's go into more depth. .. _default_arguments: Default Arguments ----------------- Going back to our initial C++ class example, lets add additional arguments to the ``hello()`` method, one of which has a default value: .. code-block:: cpp class Test { public: Test(); std::string hello(std::string first, std::string second = "world"); }; Since default parameter values are not available through templates, it is necessary to tell Rice about it using ``Rice::Arg``: .. code-block:: cpp #include using namespace Rice; extern "C" void Init_test() { Data_Type rb_cTest = define_class("Test"). define_constructor(Constructor()). define_method("hello", &Test::hello, Arg("first"), Arg("second") = "world" ); } The syntax is ``Arg(nameOfParameter)[ = defaultValue]``. The name of the parameter is not important here (it is for readability), but the value set via ``operator=`` must match the type of the parameter. As such it may be necessary to explicitly cast the default value. .. code-block:: cpp .define_method("hello", &Test::hello, Arg("first"), Arg("second") = (std::string)"world" ); These ``Rice::Arg`` objects must be in the correct positional order. Thus if the second argument has a default value, then there must be two Arg objects. Now, Ruby will now know about the default arguments, and this wrapper can be used as expected: .. code-block:: ruby t = Test.new t.hello("hello") t.hello("goodnight", "moon") This also works with Constructors: .. code-block:: cpp .define_constructor(Constructor(), Arg("arg1") = 1, Arg("otherArg") = 12); .. _return: Return ------- Similarly to the ``Arg`` class, Rice also supports a ``Return`` class that let's you tell Rice how to handle returned values from C++. This is particularly important in correctly managing memory (see :ref:`cpp_to_ruby`). It is also helpful in dealing with Ruby's VALUE type which represent Ruby objects. Most of the time Rice will automatically handle VALUE instances, but if a native method takes a VALUE argument or returns a VALUE instance then you have tell Rice about it. This is because VALUE is a typedef for ``unsigned long long`` - under the hood it is really a pointer to a Ruby object. However, to Rice it is just an integer that needs to be converted to a Ruby numeric value. As a result, if a method takes a VALUE parameter then Rice will convert it to a C++ unsigned long long value instead of passing it through. Similarly, if a method returns a VALUE then Rice will also convert it to a numeric Ruby object as opposed to simply returning it. To avoid this incorrect conversion, use the ``setValue()`` method on the ``Arg`` and ``Return`` classes. For example: .. code-block:: cpp VALUE some_function(VALUE ary) { VALUE new_ary = rb_ary_dup(ary); rb_ary_push(new_ary, Qtrue); return new_ary; } define_global_function("some_function", &some_function, Arg("ary").setValue(), Return.setValue()); Note that you can mix ``Arg`` and ``Return`` objects in any order. For example this also works: .. code-block:: cpp define_global_function("some_function", &some_function, Return.setValue(), Arg("ary").setValue()); .. _keyword_arguments: Keyword Arguments ----------------- Starting with version 4.5, Rice supports using Ruby keyword parameters to call C++ functions. The names of the keyword arguments must match the names specified in the ``Arg`` parameters used to define the method. The actual underlying names of the C++ parameters are irrelevant because C++ templates have no access to them. For example, reusing the example above: .. code-block:: cpp .define_method("hello", &Test::hello, Arg("hello"), Arg("second") = (std::string)"world" ); The ``hello`` function can be called from Ruby like this: .. code-block:: ruby test = Test.new test.hello(first: "Hello", second: "World") test.hello(first: "Hello") # This is ok because the second parameter has a default value And it can also be called in the traditional manner like this: .. code-block:: ruby test = Test.new test.hello("Hello", "World") test.hello("Hello") The ability to call the function in two different ways (position and keyword) could cause problems in the future. Imagine that you decide to move the code to Ruby - you will need to chose one of the two forms. That could result in breaking someone else's code. This risk seems low though, so for the moment Rice only supports defining arguments using the ``Arg`` class. In the future Rice may introduce a ``KeyArg`` class to avoid this issue. .. _return_self: Return Self ----------- In the case of methods that return ``self`` - meaning they return back the same C++ object that was the receiver of the function call - Rice ensures that the same Ruby object is returned. Returning self is a common pattern in Ruby. For example: .. code-block:: ruby a = Array.new a << 1 << 2 << 3 << 4 The above code works because the ``<<`` method returns the Array ``a``. You can mimic this behavior by the use of lambdas when wrapping C++ classes. For example, Rice wraps ``std::vector`` like this: .. code-block:: cpp define_vector>(). define_method("<<", [](std::vector& self, int32_t value) -> std::vector& // <----- DON'T MISS THIS { self.push_back(value); return self; // <------ Allows chaining on calls }); Pay careful attention to the lambda return type of ``std::vector&``. If the return type is *not* specified, then by default the lambda will return by value. That will invoke ``std::vector``'s copy constructor, resulting in *two* ``std::vector`` instance and two Ruby objects. Not at all what you want. Lambda Functions ---------------- Ruby classes are expected to define a ``to_s`` method that provides a string representation of an object. A good way of adding addition methods to a class is to use lambda functions: .. code-block:: cpp Data_Type rb_cTest = define_class("Test"). define_method("to_s", [](Test& self) { return ""; }); We define the ``to_s`` method to take a single parameter, self, which is an C++ instance of ``Test``. Note that ``self`` is passed by reference - we do not want to create a copy of the Test object! The lambda function can take any number of additional parameters. It can be either a stateless or stateful lambda.