From 41ebb2ee6c373a5e746422afb894b970083a0cb7 Mon Sep 17 00:00:00 2001 From: Steven Watanabe Date: Sat, 13 Jun 2009 18:30:31 +0000 Subject: [PATCH] Merge [51862], [53211], [53212], and [53218] from the trunk [SVN r53872] --- doc/design_decisions_rationale.html | 52 ++++---- doc/tuple_advanced_interface.html | 38 +++--- doc/tuple_users_guide.html | 140 +++++++++++---------- include/boost/tuple/detail/tuple_basic.hpp | 11 +- include/boost/tuple/tuple_io.hpp | 32 +++++ test/another_tuple_test_bench.cpp | 1 - test/io_test.cpp | 43 ++++--- test/tuple_test_bench.cpp | 3 +- 8 files changed, 191 insertions(+), 129 deletions(-) diff --git a/doc/design_decisions_rationale.html b/doc/design_decisions_rationale.html index ec05494..21f7457 100644 --- a/doc/design_decisions_rationale.html +++ b/doc/design_decisions_rationale.html @@ -1,3 +1,4 @@ + Design decisions rationale for Boost Tuple Library @@ -20,8 +21,8 @@ Tuples were originally under a subnamespace. As a result of the discussion, tuple definitions were moved directly under the boost namespace. As a result of a continued discussion, the subnamespace was reintroduced. The final (I truly hope so) solution is now to have all definitions in namespace ::boost::tuples, and the most common names in the ::boost namespace as well. -This is accomplished with using declarations (suggested by Dave Abrahams): -
namespace boost {
+This is accomplished with using declarations (suggested by Dave Abrahams):

+
namespace boost {
   namespace tuples {
       ...      
     // All library code
@@ -32,8 +33,8 @@ This is accomplished with using declarations (suggested by Dave Abrahams):
   using tuples::tie;
   using tuples::get;
 }
-
-With this arrangement, tuple creation with direct constructor calls, make_tuple or tie functions do not need the namespace qualifier. +
+

With this arrangement, tuple creation with direct constructor calls, make_tuple or tie functions do not need the namespace qualifier. Further, all functions that manipulate tuples are found with Koenig-lookup. The only exceptions are the get<N> functions, which are always called with an explicitly qualified template argument, and thus Koenig-lookup does not apply. Therefore, get is lifted to ::boost namespace with a using declaration. @@ -54,9 +55,9 @@ The rationale for not using the most natural name 'tuple' is to avoid having an Namespace names are, however, not generally in plural form in boost libraries. First, no real trouble was reported for using the same name for a namespace and a class and we considered changing the name 'tuples' to 'tuple'. But we found some trouble after all. -Both gcc and edg compilers reject using declarations where the namespace and class names are identical: +Both gcc and edg compilers reject using declarations where the namespace and class names are identical:

-
namespace boost {
+
namespace boost {
   namespace tuple {
     ... tie(...);
     class tuple; 
@@ -66,13 +67,13 @@ Both gcc and edg compilers reject using declarations where the namespace and cla
   using tuple::tuple; // error
     ...
 }
-
+
-Note, however, that a corresponding using declaration in the global namespace seems to be ok: +

Note, however, that a corresponding using declaration in the global namespace seems to be ok:

-
+

 using boost::tuple::tuple; // ok;
-
+

The end mark of the cons list (nil, null_type, ...)

@@ -80,14 +81,15 @@ using boost::tuple::tuple; // ok;

Tuples are internally represented as cons lists: -

tuple<int, int>
-
-inherits from -
cons<int, cons<int, null_type> >
+
tuple<int, int>
+
+

inherits from

+
cons<int, cons<int, null_type> >
 
+

null_type is the end mark of the list. Original proposition was nil, but the name is used in MacOS, and might have caused problems, so null_type was chosen instead. -Other names considered were null_t and unit (the empty tuple type in SML). +Other names considered were null_t and unit (the empty tuple type in SML).

Note that null_type is the internal representation of an empty tuple: tuple<> inherits from null_type.

@@ -95,22 +97,22 @@ Note that null_type is the internal representation of an empty tupl

Element indexing

-Whether to use 0- or 1-based indexing was discussed more than thoroughly, and the following observations were made: +Whether to use 0- or 1-based indexing was discussed more than thoroughly, and the following observations were made:

  • 0-based indexing is 'the C++ way' and used with arrays etc.
  • 1-based 'name like' indexing exists as well, eg. bind1st, bind2nd, pair::first, etc.
-Tuple access with the syntax get<N>(a), or a.get<N>() (where a is a tuple and N an index), was considered to be of the first category, hence, the index of the first element in a tuple is 0. +

Tuple access with the syntax get<N>(a), or a.get<N>() (where a is a tuple and N an index), was considered to be of the first category, hence, the index of the first element in a tuple is 0.

A suggestion to provide 1-based 'name like' indexing with constants like _1st, _2nd, _3rd, ... was made. By suitably chosen constant types, this would allow alternative syntaxes: -

a.get<0>() == a.get(_1st) == a[_1st] == a(_1st);
-
+
a.get<0>() == a.get(_1st) == a[_1st] == a(_1st);
+
-We chose not to provide more than one indexing method for the following reasons: +

We chose not to provide more than one indexing method for the following reasons:

  • 0-based indexing might not please everyone, but once its fixed, it is less confusing than having two different methods (would anyone want such constants for arrays?).
  • Adding the other indexing scheme doesn't really provide anything new (like a new feature) to the user of the library.
  • @@ -125,18 +127,18 @@ Such constants are easy to add.

    Tuple comparison

    -The comparison operator implements lexicographical order. -Other orderings were considered, mainly dominance (a < b iff for each i a(i) < b(i)). -Our belief is, that lexicographical ordering, though not mathematically the most natural one, is the most frequently needed ordering in everyday programming. +

    The comparison operator implements lexicographical order. +Other orderings were considered, mainly dominance (a < b iff for each i a(i) < b(i)). +Our belief is, that lexicographical ordering, though not mathematically the most natural one, is the most frequently needed ordering in everyday programming.

    Streaming

    The characters specified with tuple stream manipulators are stored within the space allocated by ios_base::xalloc, which allocates storage for long type objects. static_cast is used in casting between long and the stream's character type. -Streams that have character types not convertible back and forth to long thus fail to compile. +Streams that have character types not convertible back and forth to long thus fail to compile.

    -This may be revisited at some point. The two possible solutions are: +

    This may be revisited at some point. The two possible solutions are:

    • Allow only plain char types as the tuple delimiters and use widen and narrow to convert between the real character type of the stream. This would always compile, but some calls to set manipulators might result in a different diff --git a/doc/tuple_advanced_interface.html b/doc/tuple_advanced_interface.html index cde11c5..c783f02 100644 --- a/doc/tuple_advanced_interface.html +++ b/doc/tuple_advanced_interface.html @@ -15,39 +15,39 @@ The advanced features described in this document are all under namespace :

      Metafunctions for tuple types

      -Suppose T is a tuple type, and N is a constant integral expression. +Suppose T is a tuple type, and N is a constant integral expression.

      -
      element<N, T>::type
      +
      element<N, T>::type
      -gives the type of the Nth element in the tuple type T. If T is const, the resulting type is const qualified as well. +

      gives the type of the Nth element in the tuple type T. If T is const, the resulting type is const qualified as well. Note that the constness of T does not affect reference type elements.

      -
      length<T>::value
      +
      length<T>::value
      -gives the length of the tuple type T. +

      gives the length of the tuple type T.

      Cons lists

      Tuples are internally represented as cons lists. -For example, the tuple +For example, the tuple

      -
      tuple<A, B, C, D>
      +
      tuple<A, B, C, D>
      - inherits from the type -
      cons<A, cons<B, cons<C, cons<D, null_type> > > >
      -
      +

      inherits from the type

      +
      cons<A, cons<B, cons<C, cons<D, null_type> > > >
      +
      -The tuple template provides the typedef inherited to access the cons list representation. E.g.: +

      The tuple template provides the typedef inherited to access the cons list representation. E.g.: tuple<A>::inherited is the type cons<A, null_type>.

      Empty tuple

      -The internal representation of the empty tuple tuple<> is null_type. +The internal representation of the empty tuple tuple<> is null_type.

      Head and tail

      @@ -83,11 +83,11 @@ inline void set_to_zero(cons<H, T>& x) { x.get_head() = 0; set_to_zero A cons list can be default constructed provided that all its elements can be default constructed.

      -A cons list can be constructed from its head and tail. The prototype of the constructor is: +A cons list can be constructed from its head and tail. The prototype of the constructor is:

      cons(typename access_traits<head_type>::parameter_type h,
            const tail_type& t)
       
      -The traits template for the head parameter selects correct parameter types for different kinds of element types (for reference elements the parameter type equals the element type, for non-reference types the parameter type is a reference to const non-volatile element type). +

      The traits template for the head parameter selects correct parameter types for different kinds of element types (for reference elements the parameter type equals the element type, for non-reference types the parameter type is a reference to const non-volatile element type).

      For a one-element cons list the tail argument (null_type) can be omitted. @@ -98,16 +98,16 @@ For a one-element cons list the tail argument (null_type) can be om

      access_traits

      -The template access_traits defines three type functions. Let T be a type of an element in a tuple: +The template access_traits defines three type functions. Let T be a type of an element in a tuple:

        -
      1. access_traits<T>::non_const_type maps T to the return type of the non-const access functions (nonmeber and member get functions, and the get_head function).
      2. +
      3. access_traits<T>::non_const_type maps T to the return type of the non-const access functions (nonmember and member get functions, and the get_head function).
      4. access_traits<T>::const_type maps T to the return type of the const access functions.
      5. access_traits<T>::parameter_type maps T to the parameter type of the tuple constructor.

      make_tuple_traits

      -The element types of the tuples that are created with the make_tuple functions are computed with the type function make_tuple_traits. -The type function call make_tuple_traits<T>::type implements the following type mapping: +

      The element types of the tuples that are created with the make_tuple functions are computed with the type function make_tuple_traits. +The type function call make_tuple_traits<T>::type implements the following type mapping:

      • any reference type -> compile time error
      • @@ -119,7 +119,7 @@ The type function call make_tuple_traits<T>::type implements
      -Objects of type reference_wrapper are created with the ref and cref functions (see The make_tuple function.) +

      Objects of type reference_wrapper are created with the ref and cref functions (see The make_tuple function.)

      Reference wrappers were originally part of the tuple library, but they are now a general utility of boost. diff --git a/doc/tuple_users_guide.html b/doc/tuple_users_guide.html index c448b1a..c68edfc 100644 --- a/doc/tuple_users_guide.html +++ b/doc/tuple_users_guide.html @@ -1,3 +1,4 @@ + The Boost Tuple Library @@ -53,20 +54,22 @@ To compensate for this "deficiency", the Boost Tuple Library implement

      Using the library

      -

      To use the library, just include: +

      To use the library, just include:

      #include "boost/tuple/tuple.hpp"
      -

      Comparison operators can be included with: +

      Comparison operators can be included with:

      #include "boost/tuple/tuple_comparison.hpp"
      -

      To use tuple input and output operators, +

      To use tuple input and output operators,

      #include "boost/tuple/tuple_io.hpp"
      -Both tuple_io.hpp and tuple_comparison.hpp include tuple.hpp. +

      Both tuple_io.hpp and tuple_comparison.hpp include tuple.hpp.

      -

      All definitions are in namespace ::boost::tuples, but the most common names are lifted to namespace ::boost with using declarations. These names are: tuple, make_tuple, tie and get. Further, ref and cref are defined directly under the ::boost namespace. +

      All definitions are in namespace ::boost::tuples, but the most common names are lifted to namespace +::boost with using declarations. These names are: tuple, make_tuple, tie and get. +Further, ref and cref are defined directly under the ::boost namespace.

      Tuple types

      @@ -80,11 +83,11 @@ C++ types, but objects of such types cannot exist. Hence, if a tuple type contains such types as elements, the tuple type can exist, but not an object of that type. There are natural limitations for element types that cannot -be be copied, or that are not default constructible (see 'Constructing tuples' - below). +be copied, or that are not default constructible (see 'Constructing tuples' + below).

      -For example, the following definitions are valid tuple instantiations (A, B and C are some user defined classes): +For example, the following definitions are valid tuple instantiations (A, B and C are some user defined classes):

      tuple<int>
       tuple<double&, const double&, const double, double*, const double*>
      @@ -98,7 +101,7 @@ tuple<A*, tuple<const A*, const B&, C>, bool, void*>
       

      The tuple constructor takes the tuple elements as arguments. For an n-element tuple, the constructor can be invoked with k arguments, where 0 <= k <= n. -For example: +For example:

      tuple<int, double>() 
       tuple<int, double>(1) 
       tuple<int, double>(1, 3.14)
      @@ -106,7 +109,7 @@ tuple<int, double>(1, 3.14)
       
       

      If no initial value for an element is provided, it is default initialized (and hence must be default initializable). -For example. +For example.

      class X {
         X(); 
      @@ -118,7 +121,7 @@ tuple<X,X,X>()                                              // error: no d
       tuple<X,X,X>(string("Jaba"), string("Daba"), string("Duu")) // ok
       
      -In particular, reference types do not have a default initialization: +

      In particular, reference types do not have a default initialization:

      tuple<double&>()                // error: reference must be 
                                       // initialized explicitly
      @@ -134,7 +137,7 @@ tuple<const double&>(d+3.14)    // ok, but dangerous:
       

      Using an initial value for an element that cannot be copied, is a compile -time error: +time error:

      class Y { 
         Y(const Y&); 
      @@ -148,15 +151,15 @@ tuple<char[10], Y>(a, Y()); // error, neither arrays nor Y can be copied
       tuple<char[10], Y>();       // ok
       
      -Note particularly that the following is perfectly ok: -
      Y y;
      +

      Note particularly that the following is perfectly ok:

      +
      Y y;
       tuple<char(&)[10], Y&>(a, y); 
       
      -It is possible to come up with a tuple type that cannot be constructed. +

      It is possible to come up with a tuple type that cannot be constructed. This occurs if an element that cannot be initialized has a lower index than an element that requires initialization. -For example: tuple<char[10], int&>. +For example: tuple<char[10], int&>.

      In sum, the tuple construction is semantically just a group of individual elementary constructions.

      @@ -165,19 +168,19 @@ For example: tuple<char[10], int&>.

      Tuples can also be constructed using the make_tuple (cf. std::make_pair) helper functions. -This makes the construction more convenient, saving the programmer from explicitly specifying the element types: +This makes the construction more convenient, saving the programmer from explicitly specifying the element types:

      tuple<int, int, double> add_multiply_divide(int a, int b) {
         return make_tuple(a+b, a*b, double(a)/double(b));
       }
       

      -By default, the element types are deduced to the plain non-reference types. E.g: +By default, the element types are deduced to the plain non-reference types. E.g.:

      void foo(const A& a, B& b) { 
         ...
         make_tuple(a, b);
       
      -The make_tuple invocation results in a tuple of type tuple<A, B>. +

      The make_tuple invocation results in a tuple of type tuple<A, B>.

      Sometimes the plain non-reference type is not desired, e.g. if the element type cannot be copied. @@ -185,8 +188,9 @@ Therefore, the programmer can control the type deduction and state that a refere non-const type should be used as the element type instead. This is accomplished with two helper template functions: ref and cref. Any argument can be wrapped with these functions to get the desired type. -The mechanism does not compromise const correctness since a const object wrapped with ref results in a tuple element with const reference type (see the fifth code line below). -For example: +The mechanism does not compromise const correctness since a const object wrapped with ref results +in a tuple element with const reference type (see the fifth example below). +For example:

      A a; B b; const A ca = a;
       make_tuple(cref(a), b);      // creates tuple<const A&, B>
      @@ -198,19 +202,19 @@ make_tuple(ref(ca));         // creates tuple<const A&>
       
       
       

      -Array arguments to make_tuple functions are deduced to reference to const types by default; there is no need to wrap them with cref. For example: +Array arguments to make_tuple functions are deduced to reference to const types by default; there is no need to wrap them with cref. For example:

      make_tuple("Donald", "Daisy");
       
      -This creates an object of type tuple<const char (&)[7], const char (&)[6]> +

      This creates an object of type tuple<const char (&)[7], const char (&)[6]> (note that the type of a string literal is an array of const characters, not const char*). However, to get make_tuple to create a tuple with an element of a -non-const array type one must use the ref wrapper. +non-const array type one must use the ref wrapper.

      Function pointers are deduced to the plain non-reference type, that is, to plain function pointer. A tuple can also hold a reference to a function, -but such a tuple cannot be constructed with make_tuple (a const qualified function type would result, which is illegal): +but such a tuple cannot be constructed with make_tuple (a const qualified function type would result, which is illegal):

      void f(int i);
         ...
       make_tuple(&f); // tuple<void (*)(int)>
      @@ -222,19 +226,19 @@ make_tuple(f);                    // not ok
       

      Accessing tuple elements

      -Tuple elements are accessed with the expression: +Tuple elements are accessed with the expression:

      t.get<N>()
       
      -or +

      or

      get<N>(t)
       
      -where t is a tuple object and N is a constant integral expression specifying the index of the element to be accessed. +

      where t is a tuple object and N is a constant integral expression specifying the index of the element to be accessed. Depending on whether t is const or not, get returns the Nth element as a reference to const or non-const type. The index of the first element is 0 and thus N must be between 0 and k-1, where k is the number of elements in the tuple. -Violations of these constrains are detected at compile time. Examples: +Violations of these constraints are detected at compile time. Examples:

      double d = 2.7; A a;
       tuple<int, double&, const A&> t(1, d, a);
      @@ -253,16 +257,18 @@ A aa = get<3>(t);     // error: index out of bounds
       ++get<0>(t);  // ok, can be used as any variable
       
      +

      Note! The member get functions are not supported with MS Visual C++ compiler. Further, the compiler has trouble with finding the non-member get functions without an explicit namespace qualifier. -Hence, all get calls should be qualified as: tuples::get<N>(a_tuple) when writing code that shoud compile with MSVC++ 6.0. +Hence, all get calls should be qualified as: tuples::get<N>(a_tuple) when writing code that should compile with MSVC++ 6.0. +

      Copy construction and tuple assignment

      A tuple can be copy constructed from another tuple, provided that the element types are element-wise copy constructible. Analogously, a tuple can be assigned to another tuple, provided that the element types are element-wise assignable. -For example: +For example:

      class A {};
       class B : public A {};
      @@ -274,32 +280,32 @@ tuple<int, A*, C, C> a(t); // ok
       a = t;                     // ok 
       
      -In both cases, the conversions performed are: char -> int, B* -> A* (derived class pointer to base class pointer), B -> C (a user defined conversion) and D -> C (a user defined conversion). +

      In both cases, the conversions performed are: char -> int, B* -> A* (derived class pointer to base class pointer), B -> C (a user defined conversion) and D -> C (a user defined conversion).

      -Note that assignment is also defined from std::pair types: +Note that assignment is also defined from std::pair types:

      tuple<float, int> a = std::make_pair(1, 'a');
       

      Relational operators

      -Tuples reduce the operators ==, !=, <, >, <= and >= to the corresponding elementary operators. +Tuples reduce the operators ==, !=, <, >, <= and >= to the corresponding elementary operators. This means, that if any of these operators is defined between all elements of two tuples, then the same operator is defined between the tuples as well. -The equality operators for two tuples a and b are defined as: +The equality operators for two tuples a and b are defined as:

      • a == b iff for each i: ai == bi
      • a != b iff exists i: ai != bi
      -The operators <, >, <= and >= implement a lexicographical ordering. +

      The operators <, >, <= and >= implement a lexicographical ordering.

      -Note that an attempt to compare two tuples of different lengths results in a compile time error.

      -Also, the comparison operators are "short-circuited": elementary comparisons start from the first elements and are performed only until the result is clear. +Note that an attempt to compare two tuples of different lengths results in a compile time error. +Also, the comparison operators are "short-circuited": elementary comparisons start from the first elements and are performed only until the result is clear.

      -

      Examples: +

      Examples:

      tuple<std::string, int, A> t1(std::string("same?"), 2, A());
       tuple<std::string, long, A> t2(std::string("same?"), 2, A());
      @@ -316,7 +322,7 @@ t1 == t3;               // false, does not print "All the..."
       
       

      Tiers are tuples, where all elements are of non-const reference types. -They are constructed with a call to the tie function template (cf. make_tuple): +They are constructed with a call to the tie function template (cf. make_tuple):

      int i; char c; double d; 
         ...
      @@ -329,26 +335,26 @@ The same result could be achieved with the call make_tuple(ref(i), ref(c),
       

      -A tuple that contains non-const references as elements can be used to 'unpack' another tuple into variables. E.g.: +A tuple that contains non-const references as elements can be used to 'unpack' another tuple into variables. E.g.:

      int i; char c; double d; 
       tie(i, c, d) = make_tuple(1,'a', 5.5);
       std::cout << i << " " <<  c << " " << d;
       
      -This code prints 1 a 5.5 to the standard output stream. +

      This code prints 1 a 5.5 to the standard output stream. A tuple unpacking operation like this is found for example in ML and Python. -It is convenient when calling functions which return tuples. +It is convenient when calling functions which return tuples.

      -The tying mechanism works with std::pair templates as well: +The tying mechanism works with std::pair templates as well:

      int i; char c;
       tie(i, c) = std::make_pair(1, 'a');
       

      Ignore

      -There is also an object called ignore which allows you to ignore an element assigned by a tuple. -The idea is that a function may return a tuple, only part of which you are interested in. For example (note, that ignore is under the tuples subnamespace): +

      There is also an object called ignore which allows you to ignore an element assigned by a tuple. +The idea is that a function may return a tuple, only part of which you are interested in. For example (note, that ignore is under the tuples subnamespace):

      char c;
       tie(tuples::ignore, c) = std::make_pair(1, 'a');
      @@ -374,10 +380,10 @@ For Example:
       
       cout << a; 
       
      -outputs the tuple as: (1.0 2 Howdy folks!) +

      outputs the tuple as: (1.0 2 Howdy folks!)

      -The library defines three manipulators for changing the default behavior: +The library defines three manipulators for changing the default behavior:

      • set_open(char) defines the character that is output before the first element.
      • @@ -387,27 +393,27 @@ last element. elements.
      -Note, that these manipulators are defined in the tuples subnamespace. -For example: -
      cout << tuples::set_open('[') << tuples::set_close(']') << tuples::set_delimiter(',') << a; 
      +

      Note, that these manipulators are defined in the tuples subnamespace. +For example:

      +
      cout << tuples::set_open('[') << tuples::set_close(']') << tuples::set_delimiter(',') << a; 
       
      -outputs the same tuple a as: [1.0,2,Howdy folks!] +

      outputs the same tuple a as: [1.0,2,Howdy folks!]

      The same manipulators work with operator>> and istream as well. Suppose the cin stream contains the following data:

      (1 2 3) [4:5]
      -The code: +

      The code:

      -
      tuple<int, int, int> i;
      +
      tuple<int, int, int> i;
       tuple<int, int> j;
       
       cin >> i;
      -cin >> tuples::set_open('[') >> tuples::set_close(']') >> tules::set_delimiter(':');
      +cin >> tuples::set_open('[') >> tuples::set_close(']') >> tuples::set_delimiter(':');
       cin >> j;
       
      -reads the data into the tuples i and j. +

      reads the data into the tuples i and j.

      Note that extracting tuples with std::string or C-style string @@ -417,9 +423,9 @@ parseable.

      Performance

      -All tuple access and construction functions are small inlined one-liners. -Therefore, a decent compiler can eliminate any extra cost of using tuples compared to using hand written tuple like classes. -Particularly, with a decent compiler there is no performance difference between this code: +

      All tuple access and construction functions are small inlined one-liners. +Therefore, a decent compiler can eliminate any extra cost of using tuples compared to using hand-written tuple like classes. +Particularly, with a decent compiler there is no performance difference between this code:

      class hand_made_tuple { 
         A a; B b; C c;
      @@ -435,7 +441,7 @@ hand_made_tuple hmt(A(), B(), C());
       hmt.getA(); hmt.getB(); hmt.getC();
       
      -and this code: +

      and this code:

      tuple<A, B, C> t(A(), B(), C());
       t.get<0>(); t.get<1>(); t.get<2>(); 
      @@ -446,23 +452,23 @@ t.get<0>(); t.get<1>(); t.get<2>();
       

      Depending on the optimizing ability of the compiler, the tier mechanism may have a small performance penalty compared to using non-const reference parameters as a mechanism for returning multiple values from a function. -For example, suppose that the following functions f1 and f2 have equivalent functionalities: +For example, suppose that the following functions f1 and f2 have equivalent functionalities:

      void f1(int&, double&);
       tuple<int, double> f2();
       
      -Then, the call #1 may be slightly faster than #2 in the code below: +

      Then, the call #1 may be slightly faster than #2 in the code below:

      int i; double d;
         ...
       f1(i,d);         // #1
       tie(i,d) = f2(); // #2
       
      -See +

      See [1, 2] - for more in-depth discussions about efficiency. + for more in-depth discussions about efficiency.

      Effect on Compile Time

      @@ -470,7 +476,7 @@ See Compiling tuples can be slow due to the excessive amount of template instantiations. Depending on the compiler and the tuple length, it may be more than 10 times slower to compile a tuple construct, compared to compiling an equivalent explicitly written class, such as the hand_made_tuple class above. However, as a realistic program is likely to contain a lot of code in addition to tuple definitions, the difference is probably unnoticeable. -Compile time increases between 5 to 10 percentages were measured for programs which used tuples very frequently. +Compile time increases between 5 and 10 percent were measured for programs which used tuples very frequently. With the same test programs, memory consumption of compiling increased between 22% to 27%. See [1, 2] @@ -492,10 +498,10 @@ Below is a list of compilers and known problems with each compiler:

      Acknowledgements

      -Gary Powell has been an indispensable helping hand. In particular, stream manipulators for tuples were his idea. Doug Gregor came up with a working version for MSVC, David Abrahams found a way to get rid of most of the restrictions for compilers not supporting partial specialization. Thanks to Jeremy Siek, William Kempf and Jens Maurer for their help and suggestions. +

      Gary Powell has been an indispensable helping hand. In particular, stream manipulators for tuples were his idea. Doug Gregor came up with a working version for MSVC, David Abrahams found a way to get rid of most of the restrictions for compilers not supporting partial specialization. Thanks to Jeremy Siek, William Kempf and Jens Maurer for their help and suggestions. The comments by Vesa Karvonen, John Max Skaller, Ed Brey, Beman Dawes, David Abrahams and Hartmut Kaiser helped to improve the library. -The idea for the tie mechanism came from an old usenet article by Ian McCulloch, where he proposed something similar for std::pairs. +The idea for the tie mechanism came from an old usenet article by Ian McCulloch, where he proposed something similar for std::pairs.

      References

      diff --git a/include/boost/tuple/detail/tuple_basic.hpp b/include/boost/tuple/detail/tuple_basic.hpp index ba59441..348fd80 100644 --- a/include/boost/tuple/detail/tuple_basic.hpp +++ b/include/boost/tuple/detail/tuple_basic.hpp @@ -213,7 +213,7 @@ struct element_impl<0, T, true /* IsConst */> template -struct element: +struct element: public detail::element_impl::value> { }; @@ -488,11 +488,20 @@ struct length > { BOOST_STATIC_CONSTANT(int, value = 0); }; +template<> +struct length const> { + BOOST_STATIC_CONSTANT(int, value = 0); +}; + template<> struct length { BOOST_STATIC_CONSTANT(int, value = 0); }; +template<> +struct length { + BOOST_STATIC_CONSTANT(int, value = 0); +}; namespace detail { diff --git a/include/boost/tuple/tuple_io.hpp b/include/boost/tuple/tuple_io.hpp index b355e56..10cdb1c 100644 --- a/include/boost/tuple/tuple_io.hpp +++ b/include/boost/tuple/tuple_io.hpp @@ -285,6 +285,21 @@ print(std::basic_ostream& o, const cons& t) { } // namespace detail #if defined (BOOST_NO_TEMPLATED_STREAMS) + +inline std::ostream& operator<<(std::ostream& o, const null_type& t) { + if (!o.good() ) return o; + + const char l = + detail::format_info::get_manipulator(o, detail::format_info::open); + const char r = + detail::format_info::get_manipulator(o, detail::format_info::close); + + o << l; + o << r; + + return o; +} + template inline std::ostream& operator<<(std::ostream& o, const cons& t) { if (!o.good() ) return o; @@ -305,6 +320,23 @@ inline std::ostream& operator<<(std::ostream& o, const cons& t) { #else +template +inline std::basic_ostream& +operator<<(std::basic_ostream& o, + const null_type& t) { + if (!o.good() ) return o; + + const CharType l = + detail::format_info::get_manipulator(o, detail::format_info::open); + const CharType r = + detail::format_info::get_manipulator(o, detail::format_info::close); + + o << l; + o << r; + + return o; +} + template inline std::basic_ostream& operator<<(std::basic_ostream& o, diff --git a/test/another_tuple_test_bench.cpp b/test/another_tuple_test_bench.cpp index 1d1e535..15074da 100644 --- a/test/another_tuple_test_bench.cpp +++ b/test/another_tuple_test_bench.cpp @@ -23,7 +23,6 @@ #include #include -using namespace std; using namespace boost; using namespace boost::tuples; diff --git a/test/io_test.cpp b/test/io_test.cpp index f65ad42..cf6dfb0 100644 --- a/test/io_test.cpp +++ b/test/io_test.cpp @@ -27,15 +27,14 @@ #include #endif -using namespace std; using namespace boost; #if defined BOOST_NO_STRINGSTREAM -typedef ostrstream useThisOStringStream; -typedef istrstream useThisIStringStream; +typedef std::ostrstream useThisOStringStream; +typedef std::istrstream useThisIStringStream; #else -typedef ostringstream useThisOStringStream; -typedef istringstream useThisIStringStream; +typedef std::ostringstream useThisOStringStream; +typedef std::istringstream useThisIStringStream; #endif int test_main(int argc, char * argv[] ) { @@ -70,19 +69,28 @@ int test_main(int argc, char * argv[] ) { os1 << make_tuple(1, 2, 3); BOOST_CHECK (os1.str() == std::string("[1,2,3][1,2,3]") ); - ofstream tmp("temp.tmp"); + // check empty tuple. + useThisOStringStream os3; + os3 << make_tuple(); + BOOST_CHECK (os3.str() == std::string("()") ); + os3 << set_open('['); + os3 << set_close(']'); + os3 << make_tuple(); + BOOST_CHECK (os3.str() == std::string("()[]") ); + + std::ofstream tmp("temp.tmp"); #if !defined (BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION) tmp << make_tuple("One", "Two", 3); #endif tmp << set_delimiter(':'); - tmp << make_tuple(1000, 2000, 3000) << endl; + tmp << make_tuple(1000, 2000, 3000) << std::endl; tmp.close(); // When teading tuples from a stream, manipulators must be set correctly: - ifstream tmp3("temp.tmp"); - tuple j; + std::ifstream tmp3("temp.tmp"); + tuple j; #if !defined (BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION) tmp3 >> j; @@ -98,12 +106,19 @@ int test_main(int argc, char * argv[] ) { // reading tuple in format (a b c); - useThisIStringStream is("(100 200 300)"); - - tuple ti; - BOOST_CHECK(bool(is >> ti)); - BOOST_CHECK(ti == make_tuple(100, 200, 300)); + useThisIStringStream is1("(100 200 300)"); + tuple ti1; + BOOST_CHECK(bool(is1 >> ti1)); + BOOST_CHECK(ti1 == make_tuple(100, 200, 300)); + + useThisIStringStream is2("()"); + tuple<> ti2; + BOOST_CHECK(bool(is2 >> ti2)); + useThisIStringStream is3("[]"); + is3 >> set_open('['); + is3 >> set_close(']'); + BOOST_CHECK(bool(is3 >> ti2)); // Note that strings are problematic: // writing a tuple on a stream and reading it back doesn't work in diff --git a/test/tuple_test_bench.cpp b/test/tuple_test_bench.cpp index 134815f..eec9e1c 100644 --- a/test/tuple_test_bench.cpp +++ b/test/tuple_test_bench.cpp @@ -21,7 +21,6 @@ #include #include -using namespace std; using namespace boost; // ---------------------------------------------------------------------------- @@ -274,7 +273,7 @@ make_tuple_test() BOOST_CHECK(get<1>(t1) == 'a'); tuple t2; - t2 = make_tuple((short int)2, std::string("Hi")); + t2 = boost::make_tuple((short int)2, std::string("Hi")); BOOST_CHECK(get<0>(t2) == 2); BOOST_CHECK(get<1>(t2) == "Hi");