Dr. Dobb's is part of the Informa Tech Division of Informa PLC

This site is operated by a business or businesses owned by Informa PLC and all copyright resides with them. Informa PLC's registered office is 5 Howick Place, London SW1P 1WG. Registered in England and Wales. Number 8860726.


Channels ▼
RSS

C++ Metaprogramming Applied


Listing 1: Class parameter_tuple

Listing 1: Class parameter_tuple

#include<boost/type_traits/transform_traits.hpp>
#include<boost/type_traits/cv_traits.hpp>

class nil {
public:
  typedef nil element_type;
  typedef nil nested_type;
  nil() {}
  nil(const nil&) {}
  nil(const nil&, const nil&) {}
  nil(const nil&, const nil&, const nil&) {}
  nil(const nil&, const nil&, const nil&, const nil&) {}
  /* EXTEND */
};
  
template<typename FP, int n>
class nth_parameter_type {
public:
  typedef nth_parameter_type<FP::nested_type, n-1>::ret 
    ret;
};

template<typename FP>
class nth_parameter_type<FP, 0> {
public:
  typedef typename FP::element_type ret;
};

// Tag classes for mandatory and default parameters.
class default_parameter_tag{};
class mandatory_parameter_tag{};

// Class parameter_tuple
template<typename Head, typename Tail>
class parameter_tuple {
  typedef typename Head element_type;
  typedef typename Tail nested_type;
  
  typedef boost::add_reference<element_type>::type 
     element_reference_type;

  typedef boost::add_reference<boost::add_const<
    element_type
    >::type>::type  element_const_reference_type;
  
  typedef boost::add_reference<boost::add_const<
    nested_type
    >::type>::type rest_const_reference_type;

  template<int n>
  class nth_parameter_tag{};

public:
  // Constructor from n elements
  parameter_tuple(
    element_const_reference_type elem_0,
    typename boost::add_reference<boost::add_const<
      nth_parameter_type<Tail, 0>::ret
      >::type>::type elem_1 = nil(),
    typename boost::add_reference<boost::add_const<
      nth_parameter_type<Tail, 1>::ret
      >::type>::type elem_2 = nil(),
    typename boost::add_reference<boost::add_const<
      nth_parameter_type<Tail, 2>::ret
      >::type>::type elem_3 = nil()
    /* EXTEND */ ) :
    m_element(elem_0),
    m_rest(elem_1, elem_2, elem_3 /* EXTEND */ ) {}

  // Copy constructor from two function parameter objects, 
  // default implementation: copy element from first arg.
  template<typename Rhs1Head, typename Rhs1Tail, 
           typename Rhs2Head, typename Rhs2Tail>
  parameter_tuple(
    const parameter_tuple<Rhs1Head, Rhs1Tail>& rhs1,
    const parameter_tuple<Rhs2Head, Rhs2Tail>& rhs2
    ) :
    m_element(rhs1.get_element()),
    m_rest(rhs1.get_rest(), rhs2.get_rest()) {}

  // Copy constructor from two function parameter objects,
  // overload when the element type of the first argument
  // is default_parameter: copy element from second arg.
  template<typename Rhs1Tail, typename Rhs2Head, 
           typename Rhs2Tail>
  parameter_tuple(
    const parameter_tuple<
      default_parameter_tag, Rhs1Tail>& rhs1,
    const parameter_tuple<Rhs2Head, Rhs2Tail>& rhs2) :
    m_element(rhs2.get_element()),
    m_rest(rhs1.get_rest(), rhs2.get_rest()) {}
  
  // Get function to retrieve the nth parameter.
  template<int m>
  typename boost::add_reference<
    typename nth_parameter_type<
      parameter_tuple<Head, Tail>, m>::ret>::type 
  get_parameter()
  {return get_parameter_internal(nth_parameter_tag<m>());}

  // Get functions to retrieve the element and the
  // tail. These must be public for the copy constructors.
  element_const_reference_type get_element() const
  { return m_element; }
  //
  rest_const_reference_type get_rest() const
  { return m_rest; }
  
private:
  element_type m_element;
  nested_type m_rest;

  // Get function to retrieve the nth parameter, default
  //  implementation: look in the nested rest object.
  template<int m>
  typename boost::add_reference<
    typename nth_parameter_type<
      parameter_tuple<Head, Tail>, m
      >::ret>::type 
  get_parameter_internal(nth_parameter_tag<m>)
  { return m_rest.get_parameter<m-1>(); };

  // Get function to retrieve the nth parameter, overload
  // for m=0: return the element.
  element_reference_type get_parameter_internal(
    nth_parameter_tag<0>)
  { return m_element; };
  
  // Prevent assignment and copy construction.
  parameter_tuple(const parameter_tuple&);
  parameter_tuple& operator=(const parameter_tuple&);

};

// parameter_tuple type generator
template<typename T1=nil, typename T2=nil, 
         typename T3=nil, typename T4=nil
         /* EXTEND */>
class make_parameter_tuple_type {
public:
  typedef typename parameter_tuple<
    T1, typename make_parameter_tuple_type<
      T2, T3, T4 /* EXTEND */>::ret> ret;
};

// Kicks in for zero arguments passed
template<>
class make_parameter_tuple_type<
  nil, nil, nil, nil /* EXTEND */ >
{
  public:
    typedef typename nil ret;
};


Related Reading


More Insights






Currently we allow the following HTML tags in comments:

Single tags

These tags can be used alone and don't need an ending tag.

<br> Defines a single line break

<hr> Defines a horizontal line

Matching tags

These require an ending tag - e.g. <i>italic text</i>

<a> Defines an anchor

<b> Defines bold text

<big> Defines big text

<blockquote> Defines a long quotation

<caption> Defines a table caption

<cite> Defines a citation

<code> Defines computer code text

<em> Defines emphasized text

<fieldset> Defines a border around elements in a form

<h1> This is heading 1

<h2> This is heading 2

<h3> This is heading 3

<h4> This is heading 4

<h5> This is heading 5

<h6> This is heading 6

<i> Defines italic text

<p> Defines a paragraph

<pre> Defines preformatted text

<q> Defines a short quotation

<samp> Defines sample computer code text

<small> Defines small text

<span> Defines a section in a document

<s> Defines strikethrough text

<strike> Defines strikethrough text

<strong> Defines strong text

<sub> Defines subscripted text

<sup> Defines superscripted text

<u> Defines underlined text

Dr. Dobb's encourages readers to engage in spirited, healthy debate, including taking us to task. However, Dr. Dobb's moderates all comments posted to our site, and reserves the right to modify or remove any content that it determines to be derogatory, offensive, inflammatory, vulgar, irrelevant/off-topic, racist or obvious marketing or spam. Dr. Dobb's further reserves the right to disable the profile of any commenter participating in said activities.

 
Disqus Tips To upload an avatar photo, first complete your Disqus profile. | View the list of supported HTML tags you can use to style comments. | Please read our commenting policy.