| 
									
										
										
										
											2018-05-11 15:14:49 -05:00
										 |  |  | [/ File find_not.qbk] | 
					
						
							| 
									
										
										
										
											2018-05-10 19:01:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-11 15:14:49 -05:00
										 |  |  | [section:find_not find_not ] | 
					
						
							| 
									
										
										
										
											2018-05-10 19:01:12 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | [/license | 
					
						
							|  |  |  | Copyright (c) 2018 T. Zachary Laine | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Distributed under the Boost Software License, Version 1.0. | 
					
						
							|  |  |  | (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) | 
					
						
							|  |  |  | ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The header file 'find_not.hpp' contains a variants of a the stl algorithm | 
					
						
							|  |  |  | `find`. The algorithm finds the first value in the given sequence that is not | 
					
						
							|  |  |  | equal to the given value. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Consider this use of `find()`: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 17:22:49 -05:00
										 |  |  |     std::vector<int> vec = { 1, 1, 2 }; | 
					
						
							| 
									
										
										
										
											2018-05-10 19:01:12 -05:00
										 |  |  |     auto it = std::find(vec.begin(), vec.end(), 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This gives us the first occurance of `1` in `vec`.  What if we want to find | 
					
						
							|  |  |  | the first occurrance of any number besides `1` in `vec`?  We have to write an | 
					
						
							|  |  |  | unfortunate amount of code: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 17:22:49 -05:00
										 |  |  |     std::vector<int> vec = { 1, 1, 2 }; | 
					
						
							| 
									
										
										
										
											2018-05-12 16:47:47 -05:00
										 |  |  |     auto it = std::find_if(vec.begin(), vec.end(), [](int i) { return i != 1; }); | 
					
						
							| 
									
										
										
										
											2018-05-10 19:01:12 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | With `find_not()` the code gets much more terse: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 17:22:49 -05:00
										 |  |  |     std::vector<int> vec = { 1, 1, 2 }; | 
					
						
							| 
									
										
										
										
											2018-05-10 19:01:12 -05:00
										 |  |  |     auto it = find_not(vec.begin(), vec.end(), 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The existing `find` variants are: `find()`, `find_if()`, and `find_if_not()`. | 
					
						
							|  |  |  | It seems natural to also have `find_not()`, for the very reason that we have | 
					
						
							|  |  |  | `find_if_not()` -- to avoid having to write a lambda to wrap the negation of | 
					
						
							| 
									
										
										
										
											2018-05-12 16:47:47 -05:00
										 |  |  | the find condition. | 
					
						
							| 
									
										
										
										
											2018-05-10 19:01:12 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | [heading interface] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-11 15:14:49 -05:00
										 |  |  |     template<typename InputIter, typename Sentinel, typename T>         | 
					
						
							| 
									
										
										
										
											2018-05-12 16:47:47 -05:00
										 |  |  |     InputIter find_not(InputIter first, Sentinel last, const T & x); | 
					
						
							| 
									
										
										
										
											2018-05-10 19:01:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-12 15:25:37 -05:00
										 |  |  |     template<typename Range, typename T> | 
					
						
							| 
									
										
										
										
											2018-05-12 16:47:47 -05:00
										 |  |  |     boost::range_iterator<Range> find_not(Range & r, const T & x); | 
					
						
							| 
									
										
										
										
											2018-05-12 15:25:37 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-12 16:47:47 -05:00
										 |  |  | These overloads of `find_not` return the first value that is not equal to `x` | 
					
						
							|  |  |  | in the sequence `[first, last)` or `r`, respectively. | 
					
						
							| 
									
										
										
										
											2018-05-10 19:01:12 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | [heading Examples] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Given the container `c1` containing `{ 0, 1, 2 }`,  then | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-11 15:14:49 -05:00
										 |  |  |     find_not ( c1.begin(),     c1.end(),    1 ) --> c1.begin() | 
					
						
							|  |  |  |     find_not ( c1.begin(),     c1.end(),    0 ) --> std::next(c1.begin()) | 
					
						
							| 
									
										
										
										
											2018-05-10 19:01:12 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | [heading Iterator Requirements] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-12 16:47:47 -05:00
										 |  |  | `find_not` works on all iterators except output iterators. | 
					
						
							| 
									
										
										
										
											2018-05-10 19:01:12 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | The template parameter `Sentinel` is allowed to be different from `InputIter`, | 
					
						
							|  |  |  | or they may be the same.  For an `InputIter` `it` and a `Sentinel` `end`, `it | 
					
						
							|  |  |  | == end` and `it != end` must be well-formed expressions. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | [heading Complexity] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Linear. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | [heading Exception Safety] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | `find_not` takes its parameters by value and do not depend upon any global | 
					
						
							|  |  |  | state. Therefore, it provides the strong exception guarantee. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-12 16:03:41 -05:00
										 |  |  | [heading Notes] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | `constexpr` in C++14 or later. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-10 19:01:12 -05:00
										 |  |  | [endsect] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | [/ File equal.qbk | 
					
						
							|  |  |  | Copyright 2018 T. Zachary Laine | 
					
						
							|  |  |  | Distributed under the Boost Software License, Version 1.0. | 
					
						
							|  |  |  | (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt). | 
					
						
							|  |  |  | ] |