| 
									
										
										
										
											2001-02-11 05:25:19 +00:00
										 |  |  | // (C) Copyright Jeremy Siek 2000. Permission to copy, use, modify, sell and
 | 
					
						
							|  |  |  | // distribute this software is granted provided this copyright notice appears
 | 
					
						
							|  |  |  | // in all copies. This software is provided "as is" without express or implied
 | 
					
						
							|  |  |  | // warranty, and with no claim as to its suitability for any purpose.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <functional>
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <iostream>
 | 
					
						
							|  |  |  | #include <boost/iterator_adaptors.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-03-29 16:18:41 +00:00
										 |  |  | // What a bummer. We can't use std::binder1st with transform iterator
 | 
					
						
							|  |  |  | // because it does not have a default constructor. Here's a version
 | 
					
						
							|  |  |  | // that does.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace boost { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   template <class Operation>  | 
					
						
							|  |  |  |   class binder1st | 
					
						
							|  |  |  |     : public std::unary_function<typename Operation::second_argument_type, | 
					
						
							| 
									
										
										
										
											2002-02-04 20:29:35 +00:00
										 |  |  |                                  typename Operation::result_type> { | 
					
						
							| 
									
										
										
										
											2001-03-29 16:18:41 +00:00
										 |  |  |   protected: | 
					
						
							|  |  |  |     Operation op; | 
					
						
							|  |  |  |     typename Operation::first_argument_type value; | 
					
						
							|  |  |  |   public: | 
					
						
							|  |  |  |     binder1st() { } // this had to be added!
 | 
					
						
							|  |  |  |     binder1st(const Operation& x, | 
					
						
							| 
									
										
										
										
											2002-02-04 20:29:35 +00:00
										 |  |  |               const typename Operation::first_argument_type& y) | 
					
						
							|  |  |  |         : op(x), value(y) {} | 
					
						
							| 
									
										
										
										
											2001-03-29 16:18:41 +00:00
										 |  |  |     typename Operation::result_type | 
					
						
							|  |  |  |     operator()(const typename Operation::second_argument_type& x) const { | 
					
						
							|  |  |  |       return op(value, x);  | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   template <class Operation, class T> | 
					
						
							|  |  |  |   inline binder1st<Operation> bind1st(const Operation& op, const T& x) { | 
					
						
							|  |  |  |     typedef typename Operation::first_argument_type arg1_type; | 
					
						
							|  |  |  |     return binder1st<Operation>(op, arg1_type(x)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace boost
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-02-11 05:25:19 +00:00
										 |  |  | int | 
					
						
							|  |  |  | main(int, char*[]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   // This is a simple example of using the transform_iterators class to
 | 
					
						
							|  |  |  |   // generate iterators that multiply the value returned by dereferencing
 | 
					
						
							|  |  |  |   // the iterator. In this case we are multiplying by 2.
 | 
					
						
							|  |  |  |   // Would be cooler to use lambda library in this example.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int x[] = { 1, 2, 3, 4, 5, 6, 7, 8 }; | 
					
						
							|  |  |  |   const int N = sizeof(x)/sizeof(int); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-03-29 16:18:41 +00:00
										 |  |  |   typedef boost::binder1st< std::multiplies<int> > Function; | 
					
						
							| 
									
										
										
										
											2001-02-11 05:25:19 +00:00
										 |  |  |   typedef boost::transform_iterator_generator<Function, int*>::type doubling_iterator; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-03-29 16:18:41 +00:00
										 |  |  |   doubling_iterator i(x, boost::bind1st(std::multiplies<int>(), 2)), | 
					
						
							|  |  |  |     i_end(x + N, boost::bind1st(std::multiplies<int>(), 2)); | 
					
						
							| 
									
										
										
										
											2001-02-11 05:25:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   std::cout << "multiplying the array by 2:" << std::endl; | 
					
						
							|  |  |  |   while (i != i_end) | 
					
						
							|  |  |  |     std::cout << *i++ << " "; | 
					
						
							|  |  |  |   std::cout << std::endl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   std::cout << "adding 4 to each element in the array:" << std::endl; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-03-29 16:19:34 +00:00
										 |  |  |   std::copy(boost::make_transform_iterator(x, boost::bind1st(std::plus<int>(), 4)), | 
					
						
							| 
									
										
										
										
											2002-02-04 20:29:35 +00:00
										 |  |  |             boost::make_transform_iterator(x + N, boost::bind1st(std::plus<int>(), 4)), | 
					
						
							|  |  |  |             std::ostream_iterator<int>(std::cout, " ")); | 
					
						
							| 
									
										
										
										
											2001-02-11 05:25:19 +00:00
										 |  |  |   std::cout << std::endl; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 |