| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  | /****************************************************************************
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Copyright (C) 2016 The Qt Company Ltd. | 
					
						
							|  |  |  | ** Contact: https://www.qt.io/licensing/
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator. | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Commercial License Usage | 
					
						
							|  |  |  | ** Licensees holding valid commercial Qt licenses may use this file in | 
					
						
							|  |  |  | ** accordance with the commercial license agreement provided with the | 
					
						
							|  |  |  | ** Software or, alternatively, in accordance with the terms contained in | 
					
						
							|  |  |  | ** a written agreement between you and The Qt Company. For licensing terms | 
					
						
							|  |  |  | ** and conditions see https://www.qt.io/terms-conditions. For further
 | 
					
						
							|  |  |  | ** information use the contact form at https://www.qt.io/contact-us.
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** GNU General Public License Usage | 
					
						
							|  |  |  | ** Alternatively, this file may be used under the terms of the GNU | 
					
						
							|  |  |  | ** General Public License version 3 as published by the Free Software | 
					
						
							|  |  |  | ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT | 
					
						
							|  |  |  | ** included in the packaging of this file. Please review the following | 
					
						
							|  |  |  | ** information to ensure the GNU General Public License requirements will | 
					
						
							|  |  |  | ** be met: https://www.gnu.org/licenses/gpl-3.0.html.
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <utils/algorithm.h>
 | 
					
						
							|  |  |  | #include <utils/mapreduce.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-16 11:15:57 +02:00
										 |  |  | #include <QThreadPool>
 | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  | #include <QtTest>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-19 16:21:31 +01:00
										 |  |  | #if !defined(Q_CC_MSVC) || _MSC_VER >= 1900 // MSVC2015
 | 
					
						
							|  |  |  | #define SUPPORTS_MOVE
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  | class tst_MapReduce : public QObject | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Q_OBJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private slots: | 
					
						
							|  |  |  |     void mapReduce(); | 
					
						
							| 
									
										
										
										
											2016-02-15 16:26:11 +01:00
										 |  |  |     void mapReduceRvalueContainer(); | 
					
						
							|  |  |  |     void map(); | 
					
						
							| 
									
										
										
										
											2016-02-23 15:10:01 +01:00
										 |  |  |     void orderedMapReduce(); | 
					
						
							| 
									
										
										
										
											2016-02-19 16:21:31 +01:00
										 |  |  | #ifdef SUPPORTS_MOVE
 | 
					
						
							|  |  |  |     void moveOnlyType(); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int returnxx(int x) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return x*x; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void returnxxThroughFutureInterface(QFutureInterface<int> &fi, int x) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     fi.reportResult(x*x); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void tst_MapReduce::mapReduce() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-05-16 11:15:57 +02:00
										 |  |  |     QThreadPool pool; | 
					
						
							| 
									
										
										
										
											2016-02-25 16:55:58 +01:00
										 |  |  |     const auto initWithFutureInterface = [](QFutureInterface<double> &fi) -> double { | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  |         fi.reportResult(0.); | 
					
						
							|  |  |  |         return 0.; | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2016-02-25 16:55:58 +01:00
										 |  |  |     const auto reduceWithFutureInterface = [](QFutureInterface<double> &fi, double &state, int value) { | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  |         state += value; | 
					
						
							|  |  |  |         fi.reportResult(value); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     const auto reduceWithReturn = [](double &state, int value) -> double { | 
					
						
							|  |  |  |         state += value; | 
					
						
							|  |  |  |         return value; | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2016-02-25 16:55:58 +01:00
										 |  |  |     const auto cleanupWithFutureInterface = [](QFutureInterface<double> &fi, double &state) { | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  |         state /= 2.; | 
					
						
							|  |  |  |         fi.reportResult(state); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  |         // map without future interface
 | 
					
						
							| 
									
										
										
										
											2019-11-12 11:26:56 +01:00
										 |  |  |         QList<double> results = Utils::mapReduce(QVector<int>({1, 2, 3, 4, 5}), | 
					
						
							|  |  |  |                                                  initWithFutureInterface, | 
					
						
							|  |  |  |                                                  returnxx, | 
					
						
							|  |  |  |                                                  reduceWithFutureInterface, | 
					
						
							|  |  |  |                                                  cleanupWithFutureInterface) | 
					
						
							|  |  |  |                                     .results(); | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  |         Utils::sort(results); // mapping order is undefined
 | 
					
						
							|  |  |  |         QCOMPARE(results, QList<double>({0., 1., 4., 9., 16., 25., 27.5})); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  |         // map with future interface
 | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  |         QList<double> results = Utils::mapReduce(QList<int>({1, 2, 3, 4, 5}), | 
					
						
							| 
									
										
										
										
											2016-02-25 16:55:58 +01:00
										 |  |  |                                                  initWithFutureInterface, returnxxThroughFutureInterface, | 
					
						
							|  |  |  |                                                  reduceWithFutureInterface, cleanupWithFutureInterface) | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  |                 .results(); | 
					
						
							|  |  |  |         Utils::sort(results); // mapping order is undefined
 | 
					
						
							|  |  |  |         QCOMPARE(results, QList<double>({0., 1., 4., 9., 16., 25., 27.5})); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  |         // reduce without future interface
 | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  |         QList<double> results = Utils::mapReduce(QList<int>({1, 2, 3, 4, 5}), | 
					
						
							| 
									
										
										
										
											2016-02-25 16:55:58 +01:00
										 |  |  |                                                  initWithFutureInterface, returnxx, | 
					
						
							|  |  |  |                                                  reduceWithReturn, cleanupWithFutureInterface) | 
					
						
							| 
									
										
										
										
											2016-02-19 16:21:31 +01:00
										 |  |  |                 .results(); | 
					
						
							|  |  |  |         Utils::sort(results); // mapping order is undefined
 | 
					
						
							|  |  |  |         QCOMPARE(results, QList<double>({0., 1., 4., 9., 16., 25., 27.5})); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-05-16 11:15:57 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         // reduce with threadpool
 | 
					
						
							|  |  |  |         QList<double> results = Utils::mapReduce(QList<int>({1, 2, 3, 4, 5}), | 
					
						
							|  |  |  |                                                  initWithFutureInterface, returnxx, | 
					
						
							|  |  |  |                                                  reduceWithReturn, cleanupWithFutureInterface, | 
					
						
							|  |  |  |                                                  Utils::MapReduceOption::Unordered, &pool) | 
					
						
							|  |  |  |                 .results(); | 
					
						
							|  |  |  |         Utils::sort(results); // mapping order is undefined
 | 
					
						
							|  |  |  |         QCOMPARE(results, QList<double>({0., 1., 4., 9., 16., 25., 27.5})); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-02-19 16:21:31 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         // lvalue ref container
 | 
					
						
							|  |  |  |         QList<int> container({1, 2, 3, 4, 5}); | 
					
						
							|  |  |  |         QList<double> results = Utils::mapReduce(container, | 
					
						
							| 
									
										
										
										
											2016-02-25 16:55:58 +01:00
										 |  |  |                                                  initWithFutureInterface, returnxx, | 
					
						
							|  |  |  |                                                  reduceWithReturn, cleanupWithFutureInterface) | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  |                 .results(); | 
					
						
							|  |  |  |         Utils::sort(results); // mapping order is undefined
 | 
					
						
							|  |  |  |         QCOMPARE(results, QList<double>({0., 1., 4., 9., 16., 25., 27.5})); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         // std::cref
 | 
					
						
							|  |  |  |         QList<int> container({1, 2, 3, 4, 5}); | 
					
						
							|  |  |  |         QCOMPARE(Utils::mapReduce(std::cref(container), | 
					
						
							|  |  |  |                                   initWithFutureInterface, returnxx, | 
					
						
							|  |  |  |                                   reduceWithReturn, cleanupWithFutureInterface, | 
					
						
							|  |  |  |                                   Utils::MapReduceOption::Ordered).results(), | 
					
						
							|  |  |  |                  QList<double>({0., 1., 4., 9., 16., 25., 27.5})); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-05-16 11:15:57 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         // std::cref with threadpool
 | 
					
						
							|  |  |  |         QList<int> container({1, 2, 3, 4, 5}); | 
					
						
							|  |  |  |         QCOMPARE(Utils::mapReduce(std::cref(container), | 
					
						
							|  |  |  |                                   initWithFutureInterface, returnxx, | 
					
						
							|  |  |  |                                   reduceWithReturn, cleanupWithFutureInterface, | 
					
						
							|  |  |  |                                   Utils::MapReduceOption::Ordered, &pool).results(), | 
					
						
							|  |  |  |                  QList<double>({0., 1., 4., 9., 16., 25., 27.5})); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         // std::ref
 | 
					
						
							|  |  |  |         QList<int> container({1, 2, 3, 4, 5}); | 
					
						
							|  |  |  |         QCOMPARE(Utils::mapReduce(std::ref(container), | 
					
						
							|  |  |  |                                   initWithFutureInterface, returnxx, | 
					
						
							|  |  |  |                                   reduceWithReturn, cleanupWithFutureInterface, | 
					
						
							|  |  |  |                                   Utils::MapReduceOption::Ordered).results(), | 
					
						
							|  |  |  |                  QList<double>({0., 1., 4., 9., 16., 25., 27.5})); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-02-25 16:55:58 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         // init and cleanup without future interface
 | 
					
						
							|  |  |  |         QCOMPARE(Utils::mapReduce(QList<int>({1, 2, 3}), | 
					
						
							|  |  |  |                                   []() { return 0.; }, | 
					
						
							|  |  |  |                                   [](int v) { return v*2; }, | 
					
						
							|  |  |  |                                   [](double &state, int v) { return state += v/4.; }, | 
					
						
							|  |  |  |                                   [](double &) { }, | 
					
						
							|  |  |  |                                   Utils::MapReduceOption::Ordered).results(), | 
					
						
							|  |  |  |                  QList<double>({.5, 1.5, 3.})); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         // simplified map reduce without init and cleanup
 | 
					
						
							|  |  |  |         QCOMPARE(Utils::mapReduce(QList<QString>({QLatin1String("blubb"), QLatin1String("foo"), QLatin1String("blah")}), | 
					
						
							|  |  |  |                                   [](const QString &val) { return val.size(); }, | 
					
						
							|  |  |  |                                   90., | 
					
						
							|  |  |  |                                   [](double &state, int val) { | 
					
						
							|  |  |  |                                       state /= double(val); | 
					
						
							|  |  |  |                                   }, | 
					
						
							|  |  |  |                                   Utils::MapReduceOption::Ordered).result(), | 
					
						
							|  |  |  |                  1.5); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-05-16 11:15:57 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         // simplified map reduce without init and cleanup with threadpool
 | 
					
						
							|  |  |  |         QCOMPARE(Utils::mapReduce(QList<QString>({QLatin1String("blubb"), QLatin1String("foo"), QLatin1String("blah")}), | 
					
						
							|  |  |  |                                   [](const QString &val) { return val.size(); }, | 
					
						
							|  |  |  |                                   90., | 
					
						
							|  |  |  |                                   [](double &state, int val) { | 
					
						
							|  |  |  |                                       state /= double(val); | 
					
						
							|  |  |  |                                   }, | 
					
						
							|  |  |  |                                   Utils::MapReduceOption::Ordered, &pool).result(), | 
					
						
							|  |  |  |                  1.5); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         // simplified map reduce
 | 
					
						
							|  |  |  |         // std::cref
 | 
					
						
							|  |  |  |         QList<int> container({1, 2, 3}); | 
					
						
							|  |  |  |         QCOMPARE(Utils::mapReduce(std::cref(container), [](int val) { return 2*val; }, 10, | 
					
						
							|  |  |  |                                   [](int &state, int val) { state += val; }).result(), | 
					
						
							|  |  |  |                  22); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // simplified map reduce
 | 
					
						
							| 
									
										
										
										
											2017-05-16 11:15:57 +02:00
										 |  |  |         // std::cref with threadpool
 | 
					
						
							|  |  |  |         QList<int> container({1, 2, 3}); | 
					
						
							|  |  |  |         QCOMPARE(Utils::mapReduce(std::cref(container), [](int val) { return 2*val; }, 10, | 
					
						
							|  |  |  |                                   [](int &state, int val) { state += val; }, | 
					
						
							|  |  |  |                                   Utils::MapReduceOption::Unordered, &pool).result(), | 
					
						
							|  |  |  |                  22); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // simplified map reduce
 | 
					
						
							|  |  |  |         // std::ref
 | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  |         QList<int> container({1, 2, 3}); | 
					
						
							|  |  |  |         QCOMPARE(Utils::mapReduce(std::ref(container), [](int &val) { return 2*val; }, 10, | 
					
						
							|  |  |  |                                   [](int &state, int val) { state += val; }).result(), | 
					
						
							|  |  |  |                  22); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // blocking mapReduce = mappedReduced
 | 
					
						
							|  |  |  |         QCOMPARE(Utils::mappedReduced(QList<int>({1, 2, 3}), [](int &val) { return 2*val; }, 10, | 
					
						
							|  |  |  |                                       [](int &state, int val) { state += val; }), | 
					
						
							|  |  |  |                  22); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-05-16 11:15:57 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         // blocking mapReduce = mappedReduced
 | 
					
						
							|  |  |  |         // with threadpool
 | 
					
						
							|  |  |  |         QCOMPARE(Utils::mappedReduced(QList<int>({1, 2, 3}), [](int &val) { return 2*val; }, 10, | 
					
						
							|  |  |  |                                       [](int &state, int val) { state += val; }, | 
					
						
							|  |  |  |                                       Utils::MapReduceOption::Unordered, &pool), | 
					
						
							|  |  |  |                  22); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-15 16:26:11 +01:00
										 |  |  | void tst_MapReduce::mapReduceRvalueContainer() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QFuture<int> future = Utils::mapReduce(QList<int>({1, 2, 3, 4, 5}), | 
					
						
							| 
									
										
										
										
											2016-02-25 16:55:58 +01:00
										 |  |  |                                                  []() { return 0; }, | 
					
						
							| 
									
										
										
										
											2016-02-15 16:26:11 +01:00
										 |  |  |                                                  [](int value) { return value; }, | 
					
						
							| 
									
										
										
										
											2016-02-25 16:55:58 +01:00
										 |  |  |                                                  [](QFutureInterface<int> &, int &state, int value) { state += value; }, | 
					
						
							| 
									
										
										
										
											2016-02-15 16:26:11 +01:00
										 |  |  |                                                  [](QFutureInterface<int> &fi, int &state) { fi.reportResult(state); }); | 
					
						
							|  |  |  |         // here, lifetime of the QList temporary ends
 | 
					
						
							|  |  |  |         QCOMPARE(future.results(), QList<int>({15})); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void tst_MapReduce::map() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-02-23 15:10:01 +01:00
										 |  |  |     QCOMPARE(Utils::map(QList<int>({2, 5, 1}), [](int x) { return x*2.5; }).results(), | 
					
						
							|  |  |  |              QList<double>({5., 12.5, 2.5})); | 
					
						
							| 
									
										
										
										
											2016-02-15 16:26:11 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         // void result
 | 
					
						
							|  |  |  |         QList<int> results; | 
					
						
							|  |  |  |         QMutex mutex; | 
					
						
							|  |  |  |         Utils::map( | 
					
						
							|  |  |  |                     // container
 | 
					
						
							|  |  |  |                     QList<int>({2, 5, 1}), | 
					
						
							|  |  |  |                     // map
 | 
					
						
							|  |  |  |                     [&mutex, &results](int x) { QMutexLocker l(&mutex); results.append(x); } | 
					
						
							|  |  |  |                     ).waitForFinished(); | 
					
						
							| 
									
										
										
										
											2016-02-23 15:10:01 +01:00
										 |  |  |         // Utils::map is "ordered" by default, but that means that result reporting is ordered,
 | 
					
						
							|  |  |  |         // the map function is still called out-of-order
 | 
					
						
							| 
									
										
										
										
											2016-08-12 18:06:49 +03:00
										 |  |  |         Utils::sort(results); | 
					
						
							| 
									
										
										
										
											2016-02-15 16:26:11 +01:00
										 |  |  |         QCOMPARE(results, QList<int>({1, 2, 5})); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-02-19 16:21:31 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         // inplace editing
 | 
					
						
							|  |  |  |         QList<int> container({2, 5, 1}); | 
					
						
							|  |  |  |         Utils::map(std::ref(container), [](int &x) { x *= 2; }).waitForFinished(); | 
					
						
							|  |  |  |         QCOMPARE(container, QList<int>({4, 10, 2})); | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         Utils::map(container.begin(), container.end(), [](int &x) { x *= 2; }, | 
					
						
							| 
									
										
										
										
											2017-05-16 11:15:57 +02:00
										 |  |  |             Utils::MapReduceOption::Unordered, | 
					
						
							|  |  |  |             nullptr, QThread::InheritPriority, 3).waitForFinished(); | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  |         QCOMPARE(container, QList<int>({8, 20, 4})); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // blocking map = mapped
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         const QSet<int> sizes = Utils::mapped<QSet>(QStringList({QLatin1String("foo"), | 
					
						
							|  |  |  |                                                                       QLatin1String("bar"), QLatin1String("blah")}), | 
					
						
							|  |  |  |                                                          [](const QString &s) { return s.size(); }); | 
					
						
							| 
									
										
										
										
											2019-11-12 11:26:56 +01:00
										 |  |  |         QList<int> vals = sizes.values(); | 
					
						
							| 
									
										
										
										
											2016-08-12 18:06:49 +03:00
										 |  |  |         Utils::sort(vals); | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  |         QCOMPARE(vals, QList<int>({3, 4})); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         const QStringList list({QLatin1String("foo"), QLatin1String("bar"), QLatin1String("blah")}); | 
					
						
							|  |  |  |         const QSet<int> sizes = Utils::mapped<QSet>(list.cbegin(), list.cend(), | 
					
						
							|  |  |  |                                                     [](const QString &s) { return s.size(); }); | 
					
						
							| 
									
										
										
										
											2019-11-12 11:26:56 +01:00
										 |  |  |         QList<int> vals = sizes.values(); | 
					
						
							| 
									
										
										
										
											2016-08-12 18:06:49 +03:00
										 |  |  |         Utils::sort(vals); | 
					
						
							| 
									
										
										
										
											2016-02-24 22:24:43 +01:00
										 |  |  |         QCOMPARE(vals, QList<int>({3, 4})); | 
					
						
							| 
									
										
										
										
											2016-02-19 16:21:31 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-23 15:10:01 +01:00
										 |  |  | void tst_MapReduce::orderedMapReduce() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QCOMPARE(Utils::mapReduce(QList<int>({1, 2, 3, 4}), | 
					
						
							| 
									
										
										
										
											2016-02-25 16:55:58 +01:00
										 |  |  |                               []() { return 0; }, | 
					
						
							| 
									
										
										
										
											2016-02-23 15:10:01 +01:00
										 |  |  |                               [](int i) { return i*2; }, | 
					
						
							|  |  |  |                               [](int &state, int val) { state += val; return state; }, | 
					
						
							| 
									
										
										
										
											2016-02-25 16:55:58 +01:00
										 |  |  |                               [](int &) { }, | 
					
						
							| 
									
										
										
										
											2016-02-23 15:10:01 +01:00
										 |  |  |                               Utils::MapReduceOption::Ordered).results(), | 
					
						
							|  |  |  |              QList<int>({2, 6, 12, 20})); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-19 16:21:31 +01:00
										 |  |  | #ifdef SUPPORTS_MOVE
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MoveOnlyType | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2016-03-04 09:10:56 +01:00
										 |  |  |     MoveOnlyType() noexcept {} // <- with GCC 5 the defaulted one is noexcept(false)
 | 
					
						
							| 
									
										
										
										
											2016-02-19 16:21:31 +01:00
										 |  |  |     MoveOnlyType(const MoveOnlyType &) = delete; | 
					
						
							|  |  |  |     MoveOnlyType(MoveOnlyType &&) = default; | 
					
						
							|  |  |  |     MoveOnlyType &operator=(const MoveOnlyType &) = delete; | 
					
						
							|  |  |  |     MoveOnlyType &operator=(MoveOnlyType &&) = default; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MoveOnlyState : public MoveOnlyType | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     int count = 0; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MoveOnlyInit : public MoveOnlyType | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     MoveOnlyState operator()(QFutureInterface<int> &) const { return MoveOnlyState(); } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MoveOnlyMap : public MoveOnlyType | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     int operator()(const MoveOnlyType &) const { return 1; } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MoveOnlyReduce : public MoveOnlyType | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     void operator()(QFutureInterface<int> &, MoveOnlyState &state, int) { ++state.count; } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MoveOnlyList : public std::vector<MoveOnlyType> | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     MoveOnlyList() { emplace_back(MoveOnlyType()); emplace_back(MoveOnlyType()); } | 
					
						
							|  |  |  |     MoveOnlyList(const MoveOnlyList &) = delete; | 
					
						
							|  |  |  |     MoveOnlyList(MoveOnlyList &&) = default; | 
					
						
							|  |  |  |     MoveOnlyList &operator=(const MoveOnlyList &) = delete; | 
					
						
							|  |  |  |     MoveOnlyList &operator=(MoveOnlyList &&) = default; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void tst_MapReduce::moveOnlyType() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QCOMPARE(Utils::mapReduce(MoveOnlyList(), | 
					
						
							|  |  |  |                               MoveOnlyInit(), | 
					
						
							|  |  |  |                               MoveOnlyMap(), | 
					
						
							|  |  |  |                               MoveOnlyReduce(), | 
					
						
							|  |  |  |                               [](QFutureInterface<int> &fi, MoveOnlyState &state) { fi.reportResult(state.count); } | 
					
						
							|  |  |  |                 ).results(), | 
					
						
							|  |  |  |              QList<int>({2})); | 
					
						
							| 
									
										
										
										
											2016-02-15 16:26:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-19 16:21:31 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-08 11:03:28 +01:00
										 |  |  | QTEST_MAIN(tst_MapReduce) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "tst_mapreduce.moc"
 |