Timer functions, test, comments and readme
This commit is contained in:
		
							parent
							
								
									d3c7b3a91b
								
							
						
					
					
						commit
						8fdf56d1bb
					
				
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -0,0 +1 @@ | |||||||
|  | test/test | ||||||
							
								
								
									
										216
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										216
									
								
								README.md
									
									
									
									
									
								
							| @ -0,0 +1,216 @@ | |||||||
|  | 
 | ||||||
|  | # Asynco | ||||||
|  | 
 | ||||||
|  | A C++ library for event-driven asynchronous multi-threaded programming. | ||||||
|  | 
 | ||||||
|  | ## Features | ||||||
|  | 
 | ||||||
|  | - Object oriented | ||||||
|  | - Small and easy to integrate | ||||||
|  | - Header only | ||||||
|  | - Asynchronous launch functions | ||||||
|  | - Multithread parallel execution of tasks | ||||||
|  | - Timer functions: interval, timeout | ||||||
|  | - Events (on, emit) | ||||||
|  | - Event loop | ||||||
|  | 
 | ||||||
|  | ## Installation | ||||||
|  | 
 | ||||||
|  | Just download the latest release and unzip it into your project.  | ||||||
|  | 
 | ||||||
|  | ```c++ | ||||||
|  | #include "asynco/lib/asynco.hpp"    // asynco(), wait() | ||||||
|  | #include "asynco/lib/event.hpp"     // event | ||||||
|  | #include "asynco/lib/rotor.hpp"     // interval, timeout | ||||||
|  | #include "asynco/lib/runner.hpp"    // on_async | ||||||
|  | using namespace marcelb; | ||||||
|  | 
 | ||||||
|  | #ifndef ON_RUNNER | ||||||
|  | #define ON_RUNNER | ||||||
|  | runner on_async; | ||||||
|  | #endif | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ## Usage | ||||||
|  | 
 | ||||||
|  | Time asynchronous functions | ||||||
|  | 
 | ||||||
|  | ```c++ | ||||||
|  | // start interval | ||||||
|  | interval inter1 ([&]() { | ||||||
|  |      cout << "Interval 1" << endl; | ||||||
|  | }, 1000); | ||||||
|  | 
 | ||||||
|  | // stop interval | ||||||
|  | inter1.clear(); | ||||||
|  | 
 | ||||||
|  | // start timeout | ||||||
|  | timeout time1 ( [&] () { | ||||||
|  |     cout << "Timeout 1 " << endl; | ||||||
|  | }, 10000); | ||||||
|  | 
 | ||||||
|  | // stop timeout | ||||||
|  | time1.clear(); | ||||||
|  | ``` | ||||||
|  | Make functions asynchronous | ||||||
|  | 
 | ||||||
|  | ```c++ | ||||||
|  | /** | ||||||
|  | * Put task directly and get returned value - it is not recommended to use it | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | auto res1 = on_async.put_task( [] () { | ||||||
|  |     cout << "Jebiga " <<endl; | ||||||
|  |     throw string ("jebiga!!"); | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | try { | ||||||
|  |     res1.get(); | ||||||
|  | } catch (const string except) { | ||||||
|  |     cout << except << endl; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  | * Run an function asyncronic | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | asynco( []() { | ||||||
|  |     sleep_for(2s);   // only for simulate log duration function | ||||||
|  |     cout << "asynco" << endl; | ||||||
|  |     return 5; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  | * Wait after runned as async | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | auto a = asynco( []() { | ||||||
|  |     sleep_for(2s);   // only for simulate log duration function | ||||||
|  |     cout << "asynco" << endl; | ||||||
|  |     return 5; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | cout << wait(move(a)) << endl; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  | * Wait async function call and use i cout | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | cout << wait(asynco( [] () { | ||||||
|  |     sleep_for(chrono::seconds(1)); // only for simulate log duration function | ||||||
|  |     cout << "wait end" << endl; | ||||||
|  |     return 4; | ||||||
|  | })) << endl; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  | * Sleep with timeout sleep implement | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | void sleep_to (int _time) { | ||||||
|  |     promise<void> _promise; | ||||||
|  |     timeout t( [&]() { | ||||||
|  |         _promise.set_value(); | ||||||
|  |     }, _time); | ||||||
|  | 
 | ||||||
|  |     return _promise.get_future().get(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | sleep_to(3000); | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  | * Catch promise reject | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | void promise_reject (int _time) { | ||||||
|  |     promise<void> _promise; | ||||||
|  |     timeout t( [&]() { | ||||||
|  |         try { | ||||||
|  |             // simulate except | ||||||
|  |             throw runtime_error("Error simulation"); | ||||||
|  |             _promise.set_value(); | ||||||
|  |         } catch (...) { | ||||||
|  |             _promise.set_exception(current_exception()); | ||||||
|  |         } | ||||||
|  |     }, _time); | ||||||
|  | 
 | ||||||
|  |     return _promise.get_future().get(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | try { | ||||||
|  |     promise_reject(3000); | ||||||
|  | } catch (runtime_error err) { | ||||||
|  |     cout<< err.what() << endl; | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  | Events | ||||||
|  | 
 | ||||||
|  | ```c++ | ||||||
|  | /** | ||||||
|  | * initialization of typed events | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | event<int, int> ev2int; | ||||||
|  | event<int, string> evintString; | ||||||
|  | event<> evoid; | ||||||
|  | 
 | ||||||
|  | ev2int.on("sum", [](int a, int b) { | ||||||
|  |     cout << "Sum " << a+b << endl; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | evintString.on("substract", [](int a, string b) { | ||||||
|  |     cout << "Substract " << a-stoi(b) << endl; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | evoid.on("void", []() { | ||||||
|  |     cout << "Void emited" << endl; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | sleep(1); | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  | * Emit | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | ev2int.emit("sum", 5, 8); | ||||||
|  | 
 | ||||||
|  | sleep(1); | ||||||
|  | evintString.emit("substract", 3, to_string(2)); | ||||||
|  | 
 | ||||||
|  | sleep(1); | ||||||
|  | evoid.emit("void"); | ||||||
|  | ``` | ||||||
|  | Extend own class whit events | ||||||
|  | 
 | ||||||
|  | ```c++ | ||||||
|  | class myOwnClass : public event<int> { | ||||||
|  |     public: | ||||||
|  |     myOwnClass() : event() {}; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | myOwnClass myclass; | ||||||
|  | 
 | ||||||
|  | timeout t( [&] { | ||||||
|  |     myclass.emit("constructed", 1); | ||||||
|  | }, 200); | ||||||
|  | 
 | ||||||
|  | myclass.on("constructed", [] (int i) { | ||||||
|  |     cout << "Constructed " << i  << endl; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | ## License | ||||||
|  | 
 | ||||||
|  | [APACHE 2.0](http://www.apache.org/licenses/LICENSE-2.0/) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ## Support & Feedback | ||||||
|  | 
 | ||||||
|  | For support and any feedback, contact the address: marcelb96@yahoo.com. | ||||||
|  | 
 | ||||||
|  | ## Contributing | ||||||
|  | 
 | ||||||
|  | Contributions are always welcome! | ||||||
|  | 
 | ||||||
|  | Feel free to fork and start working with or without a later pull request. Or contact for suggest and request an option. | ||||||
|  | 
 | ||||||
| @ -1,164 +0,0 @@ | |||||||
| #ifndef _ASYNCO_ |  | ||||||
| #define _ASYNCO_ |  | ||||||
| 
 |  | ||||||
| #include <vector> |  | ||||||
| #include <mutex> |  | ||||||
| #include <future> |  | ||||||
| #include <thread> |  | ||||||
| 
 |  | ||||||
| using namespace std; |  | ||||||
| 
 |  | ||||||
| namespace marcelb { |  | ||||||
| 
 |  | ||||||
| class interval; |  | ||||||
| class timeout; |  | ||||||
| 
 |  | ||||||
| class loop_core { |  | ||||||
|     vector<interval> intervals; |  | ||||||
|     vector<timeout> timeouts; |  | ||||||
|     time_t sampling; |  | ||||||
|     mutex i_m, t_m; |  | ||||||
|     future<void> bot; |  | ||||||
| 
 |  | ||||||
|     loop_core() { |  | ||||||
|         bot = async(launch::async, [this] () { |  | ||||||
|             loop(); |  | ||||||
|         }); |  | ||||||
|         // on_async.put_task( [this] () {
 |  | ||||||
|         //     loop();
 |  | ||||||
|         // });
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void run(interval& _interval)  { |  | ||||||
|         lock_guard<mutex> lock(i_m); |  | ||||||
|         intervals.push_back(_interval); |  | ||||||
|         update_sampling(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void run(timeout& _timeout)  { |  | ||||||
|         lock_guard<mutex> lock(t_m); |  | ||||||
|         timeouts.push_back(_timeout); |  | ||||||
|         update_sampling(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void loop() { |  | ||||||
|         while (true) { |  | ||||||
|             for (auto& _interval : intervals) { |  | ||||||
|                 int64_t now = chrono::duration_cast<chrono::milliseconds>(chrono::system_clock::now().time_since_epoch()).count(); |  | ||||||
|                 if (now - _interval.execute >= _interval._duration) { |  | ||||||
|                     _interval.callback(); |  | ||||||
|                     _interval.execute = now; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             for (int i=0; i<timeouts.size(); i++) { |  | ||||||
|                 int64_t now = chrono::duration_cast<chrono::milliseconds>(chrono::system_clock::now().time_since_epoch()).count(); |  | ||||||
|                 if (now - timeouts[i]._construct >= timeouts[i]._duration) { |  | ||||||
|                     auto& _timeout = timeouts[i]; |  | ||||||
|                     { |  | ||||||
|                         lock_guard<mutex> lock(t_m); |  | ||||||
|                         timeouts.erase(timeouts.begin() + i); |  | ||||||
|                     } |  | ||||||
|                     _timeout.callback(); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             this_thread::sleep_for(chrono::milliseconds(sampling)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void update_sampling() { |  | ||||||
|         sampling = 0; |  | ||||||
|         for (auto& _interval : intervals)  { |  | ||||||
|             sampling += _interval._duration; |  | ||||||
|         } |  | ||||||
|         for (auto& _timeout : timeouts)  { |  | ||||||
|             sampling += _timeout._duration; |  | ||||||
|         } |  | ||||||
|         sampling /= (intervals.size() + timeouts.size())*2; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| loop_core co_loop; |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| class interval { |  | ||||||
|     public: |  | ||||||
|     bool run = true; |  | ||||||
|     function<void()> callback; |  | ||||||
|     const time_t _duration; |  | ||||||
|     time_t execute = 0; |  | ||||||
|      |  | ||||||
|     // public:
 |  | ||||||
|     interval(function<void()> func, const time_t duration): callback(func), _duration(duration) { |  | ||||||
|         #ifndef ON_ASYNC |  | ||||||
|             throw string("Not on_async defined!"); |  | ||||||
|         #endif |  | ||||||
| 
 |  | ||||||
|         auto task = [&] () { |  | ||||||
|             while (run) { |  | ||||||
|                 // this_thread::sleep_for(chrono::milliseconds(_duration));
 |  | ||||||
|                 // if (run) {
 |  | ||||||
|                     callback(); |  | ||||||
|                 // }
 |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         // on_async.put_task(task);
 |  | ||||||
|         co_loop.run(this*); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void clear() { |  | ||||||
|         run = false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     ~interval() { |  | ||||||
|         clear(); |  | ||||||
|     } |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| class timeout { |  | ||||||
|     public: |  | ||||||
|     bool run = true; |  | ||||||
|     function<void()> callback; |  | ||||||
|     const time_t _duration; |  | ||||||
|     int64_t _construct = |  | ||||||
|         chrono::duration_cast<chrono::milliseconds>(chrono::system_clock::now().time_since_epoch()) |  | ||||||
|         .count(); |  | ||||||
|      |  | ||||||
|     // public:
 |  | ||||||
|     timeout(function<void()> f, const time_t duration): callback(f), _duration(duration) { |  | ||||||
|         #ifndef ON_ASYNC |  | ||||||
|             throw string("Not on_async defined!"); |  | ||||||
|         #endif |  | ||||||
| 
 |  | ||||||
|         auto task = [&] () { |  | ||||||
|             // int64_t _start = 
 |  | ||||||
|             //     chrono::duration_cast<chrono::milliseconds>(chrono::system_clock::now().time_since_epoch())
 |  | ||||||
|             //     .count();
 |  | ||||||
|             // this_thread::sleep_for(chrono::milliseconds(_duration - (_start - _construct)));
 |  | ||||||
|             if (run) { |  | ||||||
|                 callback(); |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         // on_async.put_task(task);
 |  | ||||||
|         co_loop.run(this*); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void clear() { |  | ||||||
|         run = false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     ~timeout() { |  | ||||||
|         clear(); |  | ||||||
|     } |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
							
								
								
									
										159
									
								
								lib/asynco.hpp
									
									
									
									
									
								
							
							
						
						
									
										159
									
								
								lib/asynco.hpp
									
									
									
									
									
								
							| @ -1,162 +1,20 @@ | |||||||
| #ifndef _ASYNCO_ | #ifndef _ASYNCO_ | ||||||
| #define _ASYNCO_ | #define _ASYNCO_ | ||||||
| 
 | 
 | ||||||
| #include "loop.hpp" | #include "runner.hpp" | ||||||
| 
 | 
 | ||||||
| using namespace std; | using namespace std; | ||||||
| 
 | 
 | ||||||
| namespace marcelb { | namespace marcelb { | ||||||
| 
 | 
 | ||||||
| #ifndef ON_ASYNC | #ifndef ON_RUNNER | ||||||
| #define ON_ASYNC | #define ON_RUNNER | ||||||
| AsyncLoop on_async; | runner on_async; | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| class interval; | /**
 | ||||||
| class timeout; |  * Run the function asynchronously | ||||||
| 
 | */ | ||||||
| class loop_core { |  | ||||||
|    static vector<interval> intervals; |  | ||||||
|    static vector<timeout> timeouts; |  | ||||||
|     time_t sampling; |  | ||||||
|     mutex i_m, t_m; |  | ||||||
| 
 |  | ||||||
|     loop_core() { |  | ||||||
|         on_async.put_task( [this] () { |  | ||||||
|             loop(); |  | ||||||
|         }); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void run(interval& _interval)  { |  | ||||||
|         lock_guard<mutex> lock(i_m); |  | ||||||
|         intervals.push_back(_interval); |  | ||||||
|         update_sampling(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void run(timeout& _timeout)  { |  | ||||||
|         lock_guard<mutex> lock(t_m); |  | ||||||
|         timeouts.push_back(_timeout); |  | ||||||
|         update_sampling(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void loop() { |  | ||||||
|         while (true) { |  | ||||||
|             for (auto& _interval : intervals) { |  | ||||||
|                 int64_t now = chrono::duration_cast<chrono::milliseconds>(chrono::system_clock::now().time_since_epoch()).count(); |  | ||||||
|                 if (now - _interval.execute >= _interval._duration) { |  | ||||||
|                     _interval.callback(); |  | ||||||
|                     _interval.execute = now; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             for (int i=0; i<timeouts.size(); i++) { |  | ||||||
|                 int64_t now = chrono::duration_cast<chrono::milliseconds>(chrono::system_clock::now().time_since_epoch()).count(); |  | ||||||
|                 if (now - timeouts[i]._construct >= timeouts[i]._duration) { |  | ||||||
|                     auto& _timeout = timeouts[i]; |  | ||||||
|                     { |  | ||||||
|                         lock_guard<mutex> lock(t_m); |  | ||||||
|                         timeouts.erase(timeouts.begin() + i); |  | ||||||
|                     } |  | ||||||
|                     _timeout.callback(); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             sleep_for(chrono::milliseconds(sampling)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void update_sampling() { |  | ||||||
|         sampling = 0; |  | ||||||
|         for (auto& _interval : intervals)  { |  | ||||||
|             sampling += _interval._duration; |  | ||||||
|         } |  | ||||||
|         for (auto& _timeout : timeouts)  { |  | ||||||
|             sampling += _timeout._duration; |  | ||||||
|         } |  | ||||||
|         sampling /= (intervals.size() + timeouts.size())*2; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| loop_core co_loop; |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| class interval { |  | ||||||
|     public: |  | ||||||
|     bool run = true; |  | ||||||
|     function<void()> callback; |  | ||||||
|     const time_t _duration; |  | ||||||
|     time_t execute = 0; |  | ||||||
|      |  | ||||||
|     // public:
 |  | ||||||
|     interval(function<void()> func, const time_t duration): callback(func), _duration(duration) { |  | ||||||
|         #ifndef ON_ASYNC |  | ||||||
|             throw string("Not on_async defined!"); |  | ||||||
|         #endif |  | ||||||
| 
 |  | ||||||
|         auto task = [&] () { |  | ||||||
|             while (run) { |  | ||||||
|                 // this_thread::sleep_for(chrono::milliseconds(_duration));
 |  | ||||||
|                 // if (run) {
 |  | ||||||
|                     callback(); |  | ||||||
|                 // }
 |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         // on_async.put_task(task);
 |  | ||||||
|         co_loop.run(this*); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void clear() { |  | ||||||
|         run = false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     ~interval() { |  | ||||||
|         clear(); |  | ||||||
|     } |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| class timeout { |  | ||||||
|     public: |  | ||||||
|     bool run = true; |  | ||||||
|     function<void()> callback; |  | ||||||
|     const time_t _duration; |  | ||||||
|     int64_t _construct = |  | ||||||
|         chrono::duration_cast<chrono::milliseconds>(chrono::system_clock::now().time_since_epoch()) |  | ||||||
|         .count(); |  | ||||||
|      |  | ||||||
|     // public:
 |  | ||||||
|     timeout(function<void()> f, const time_t duration): callback(f), _duration(duration) { |  | ||||||
|         #ifndef ON_ASYNC |  | ||||||
|             throw string("Not on_async defined!"); |  | ||||||
|         #endif |  | ||||||
| 
 |  | ||||||
|         auto task = [&] () { |  | ||||||
|             // int64_t _start = 
 |  | ||||||
|             //     chrono::duration_cast<chrono::milliseconds>(chrono::system_clock::now().time_since_epoch())
 |  | ||||||
|             //     .count();
 |  | ||||||
|             // this_thread::sleep_for(chrono::milliseconds(_duration - (_start - _construct)));
 |  | ||||||
|             if (run) { |  | ||||||
|                 callback(); |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         // on_async.put_task(task);
 |  | ||||||
|         co_loop.run(this*); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void clear() { |  | ||||||
|         run = false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     ~timeout() { |  | ||||||
|         clear(); |  | ||||||
|     } |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| template<class F, class... Args> | template<class F, class... Args> | ||||||
| auto asynco(F&& f, Args&&... args) -> future<typename result_of<F(Args...)>::type> { | auto asynco(F&& f, Args&&... args) -> future<typename result_of<F(Args...)>::type> { | ||||||
|     using return_type = typename result_of<F(Args...)>::type; |     using return_type = typename result_of<F(Args...)>::type; | ||||||
| @ -165,6 +23,9 @@ auto asynco(F&& f, Args&&... args) -> future<typename result_of<F(Args...)>::typ | |||||||
|     return res; |     return res; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * Block until the asynchronous call completes | ||||||
|  | */ | ||||||
| template<typename T> | template<typename T> | ||||||
| T wait(future<T> r) { | T wait(future<T> r) { | ||||||
|     return r.get(); |     return r.get(); | ||||||
|  | |||||||
| @ -5,28 +5,38 @@ | |||||||
| #include <map> | #include <map> | ||||||
| #include <string> | #include <string> | ||||||
| #include <functional> | #include <functional> | ||||||
| #include "loop.hpp" | #include "runner.hpp" | ||||||
| 
 | 
 | ||||||
| using namespace std; | using namespace std; | ||||||
| 
 | 
 | ||||||
| namespace marcelb { | namespace marcelb { | ||||||
| 
 | 
 | ||||||
| #ifndef ON_ASYNC | #ifndef ON_RUNNER | ||||||
| #define ON_ASYNC | #define ON_RUNNER | ||||||
| AsyncLoop on_async; | runner on_async; | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| 
 | /**
 | ||||||
|  |  * Event class, for event-driven programming. | ||||||
|  |  * These events are typed according to the arguments of the callback function | ||||||
|  | */ | ||||||
| template<typename... T> | template<typename... T> | ||||||
| class event { | class event { | ||||||
|     private: |     private: | ||||||
|     unordered_map<string, function<void(T...)>> events; |     unordered_map<string, function<void(T...)>> events; | ||||||
| 
 | 
 | ||||||
|     public: |     public: | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Defines event by key, and callback function | ||||||
|  |     */ | ||||||
|     void on(const string& key, function<void(T...)> callback) { |     void on(const string& key, function<void(T...)> callback) { | ||||||
|         events[key] = callback; |         events[key] = callback; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /**
 | ||||||
|  |      * It emits an event and sends a callback function saved according to the key with the passed parameters | ||||||
|  |     */ | ||||||
|     template<typename... Args> |     template<typename... Args> | ||||||
|     void emit(const string& key, Args... args) { |     void emit(const string& key, Args... args) { | ||||||
|         auto it = events.find(key); |         auto it = events.find(key); | ||||||
|  | |||||||
							
								
								
									
										212
									
								
								lib/rotor.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										212
									
								
								lib/rotor.hpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,212 @@ | |||||||
|  | #ifndef _ROTOR_ | ||||||
|  | #define _ROTOT_ | ||||||
|  | 
 | ||||||
|  | #include "runner.hpp" | ||||||
|  | #include "chrono" | ||||||
|  | 
 | ||||||
|  | #include "iostream" | ||||||
|  | 
 | ||||||
|  | using namespace std; | ||||||
|  | using namespace marcelb; | ||||||
|  | 
 | ||||||
|  | #ifndef ON_RUNNER | ||||||
|  | #define ON_RUNNER | ||||||
|  | runner on_async; | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | namespace marcelb { | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Get the time in ms from the epoch | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | int64_t rtime_ms() { | ||||||
|  |     return chrono::duration_cast<chrono::milliseconds>(chrono::system_clock::now() | ||||||
|  |         .time_since_epoch()) | ||||||
|  |         .count(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Structure for time events | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | struct time_event { | ||||||
|  |     function<void()> callback; | ||||||
|  |     int64_t init; | ||||||
|  |     int64_t time; | ||||||
|  |     bool repeat; | ||||||
|  |     bool stop; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Event loop for time events | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | class rotor { | ||||||
|  |     vector<struct time_event *> tevents; | ||||||
|  |     mutex te_m; | ||||||
|  |     bool rotating = true; | ||||||
|  |     int64_t sampling; | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Loop method, started by the constructor in a separate runner | ||||||
|  |      * It checks the events on the stack and sends the expired ones to the runner | ||||||
|  |     */ | ||||||
|  |     void loop() { | ||||||
|  |         while (rotating) { | ||||||
|  |             for (int i=0; i<tevents.size(); i++) { | ||||||
|  | 
 | ||||||
|  |                 if (tevents[i]->stop) { | ||||||
|  |                     remove(i); | ||||||
|  |                     i--; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 else if (expired(tevents[i])) { | ||||||
|  |                     on_async.put_task(tevents[i]->callback); | ||||||
|  |                     if (tevents[i]->repeat) { | ||||||
|  |                         tevents[i]->init = rtime_ms(); | ||||||
|  |                     } | ||||||
|  |                     else { | ||||||
|  |                         remove(i); | ||||||
|  |                         i--; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             this_thread::sleep_for(chrono::milliseconds(sampling)); | ||||||
|  |         }  | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * The method checks whether the time event has expired | ||||||
|  |     */ | ||||||
|  |     bool expired(struct time_event *tevent) { | ||||||
|  |         return rtime_ms() - tevent->init >= tevent->time; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * The method deletes a non-repeating or stopped event from the stack | ||||||
|  |     */ | ||||||
|  |     void remove(const int& position) { | ||||||
|  |         lock_guard<mutex> lock(te_m); | ||||||
|  |         tevents.erase(tevents.begin()+position); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Updates the idle time of the loop, according to twice the frequency of available events | ||||||
|  |     */ | ||||||
|  |     void update_sampling() { | ||||||
|  |         sampling = tevents[0]->time; | ||||||
|  |         for (int i=0; i<tevents.size(); i++) { | ||||||
|  |             if (sampling > tevents[i]->time) { | ||||||
|  |                 sampling = tevents[i]->time; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         sampling /= tevents.size()*2; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public: | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Constructor for the rotor, starts the given loop by occupying one runner | ||||||
|  |     */ | ||||||
|  |     rotor() { | ||||||
|  |         on_async.put_task( [&] () { | ||||||
|  |             loop(); | ||||||
|  |         }); | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Adds a time event to the stack | ||||||
|  |     */ | ||||||
|  |     void insert(struct time_event *tevent) { | ||||||
|  |         lock_guard<mutex> lock(te_m); | ||||||
|  |         tevents.push_back(tevent); | ||||||
|  |         update_sampling(); | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Returns the number of active events | ||||||
|  |     */ | ||||||
|  |     int active() { | ||||||
|  |         return tevents.size(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Stops all active events and stops the rotor | ||||||
|  |     */ | ||||||
|  |     ~rotor() { | ||||||
|  |         for (int i=0; i<tevents.size(); i++) { | ||||||
|  |             tevents[i]->stop = true; | ||||||
|  |         } | ||||||
|  |         rotating = false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * It is intended that there is only one global declaration | ||||||
|  | */ | ||||||
|  | rotor _rotor; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * A class for all timer functions | ||||||
|  | */ | ||||||
|  | class timer_core { | ||||||
|  |     public: | ||||||
|  |     struct time_event t_event; | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Timer constructor, receives a callback function and time | ||||||
|  |     */ | ||||||
|  |     timer_core( function<void()> _callback, int64_t _time): | ||||||
|  |         t_event({ _callback, rtime_ms(), _time, false, false }) { | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  |     /**
 | ||||||
|  |      * Stop timer | ||||||
|  |     */ | ||||||
|  |     void clear() { | ||||||
|  |         t_event.stop = true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Destruktor of timer, call stop | ||||||
|  |     */ | ||||||
|  |     ~timer_core() { | ||||||
|  |         clear(); | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Class interval for periodic execution of the callback in time in ms | ||||||
|  | */ | ||||||
|  | class interval : public timer_core { | ||||||
|  |     public: | ||||||
|  |      | ||||||
|  |     /**
 | ||||||
|  |      * The constructor receives a callback function and an interval time | ||||||
|  |     */ | ||||||
|  |     interval( function<void()> _callback, int64_t _time): timer_core(_callback, _time) { | ||||||
|  |         t_event.repeat = true; | ||||||
|  |         _rotor.insert(&t_event); | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Class interval for delayed callback execution in ms | ||||||
|  | */ | ||||||
|  | class timeout : public timer_core { | ||||||
|  |     public: | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * The constructor receives a callback function and a delay time | ||||||
|  |     */ | ||||||
|  |     timeout( function<void()> _callback, int64_t delay): timer_core(_callback, delay) { | ||||||
|  |         t_event.repeat = false; | ||||||
|  |         _rotor.insert(&t_event); | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
| @ -1,5 +1,5 @@ | |||||||
| #ifndef _LOOP_ | #ifndef _RUNNER_ | ||||||
| #define _LOOP_ | #define _RUNNER_ | ||||||
| 
 | 
 | ||||||
| #include <thread> | #include <thread> | ||||||
| #include <vector> | #include <vector> | ||||||
| @ -13,22 +13,31 @@ using namespace std; | |||||||
| 
 | 
 | ||||||
| namespace marcelb { | namespace marcelb { | ||||||
| 
 | 
 | ||||||
| #ifdef ON_ASYNC | #ifdef ON_RUNNER | ||||||
| extern AsyncLoop on_async; | extern runner on_async; | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| class AsyncLoop { | /**
 | ||||||
|  |  * The runner class implements multithread, task stack and event loop for asynchronous execution of tasks | ||||||
|  | */ | ||||||
|  | class runner { | ||||||
|     private: |     private: | ||||||
|     vector<thread> workers; |     vector<thread> runners; | ||||||
|     queue<function<void()>> tasks; |     queue<function<void()>> tasks; | ||||||
|     mutex q_io; |     mutex q_io; | ||||||
|     condition_variable cv; |     condition_variable cv; | ||||||
|     bool stop; |     bool stop; | ||||||
| 
 | 
 | ||||||
|     public: |     public: | ||||||
|     AsyncLoop(size_t pool_size = thread::hardware_concurrency()) : stop(false) { | 
 | ||||||
|  |     /**
 | ||||||
|  |      * The constructor starts as many threads as the system has cores,  | ||||||
|  |      * and runs an event loop inside each one.  | ||||||
|  |      * Each event loop waits for tasks from the stack and executes them. | ||||||
|  |     */ | ||||||
|  |     runner(size_t pool_size = thread::hardware_concurrency()) : stop(false) { | ||||||
|         for (size_t i = 0; i < pool_size; ++i) { |         for (size_t i = 0; i < pool_size; ++i) { | ||||||
|             workers.emplace_back([this] { |             runners.emplace_back( thread([this] { | ||||||
|                 while (true) { |                 while (true) { | ||||||
|                     function<void()> task; |                     function<void()> task; | ||||||
|                     { |                     { | ||||||
| @ -41,10 +50,13 @@ class AsyncLoop { | |||||||
|                     } |                     } | ||||||
|                     task(); |                     task(); | ||||||
|                 } |                 } | ||||||
|             }); |             })); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /**
 | ||||||
|  |      * With the method, we send the callback function and its arguments to the task stack | ||||||
|  |     */ | ||||||
|     template<class F, class... Args> |     template<class F, class... Args> | ||||||
|     auto put_task(F&& f, Args&&... args) |     auto put_task(F&& f, Args&&... args) | ||||||
|         -> future<typename result_of<F(Args...)>::type> { |         -> future<typename result_of<F(Args...)>::type> { | ||||||
| @ -66,22 +78,31 @@ class AsyncLoop { | |||||||
|         return res; |         return res; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Returns the number of tasks the runner has to perform | ||||||
|  |     */ | ||||||
|     unsigned int count_tasks() { |     unsigned int count_tasks() { | ||||||
|         return tasks.size(); |         return tasks.size(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Returns the number of threads used by the runner | ||||||
|  |     */ | ||||||
|     unsigned int count_threads() { |     unsigned int count_threads() { | ||||||
|         return workers.size(); |         return runners.size(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     ~AsyncLoop() { |     /**
 | ||||||
|  |      * The destructor stops all loops and stops all threads | ||||||
|  |     */ | ||||||
|  |     ~runner() { | ||||||
|         { |         { | ||||||
|             unique_lock<mutex> lock(q_io); |             unique_lock<mutex> lock(q_io); | ||||||
|             stop = true; |             stop = true; | ||||||
|         } |         } | ||||||
|         cv.notify_all(); |         cv.notify_all(); | ||||||
|         for (thread& worker : workers) { |         for (thread& runner : runners) { | ||||||
|             worker.join(); |             runner.join(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
							
								
								
									
										190
									
								
								test/test.cpp
									
									
									
									
									
								
							
							
						
						
									
										190
									
								
								test/test.cpp
									
									
									
									
									
								
							| @ -1,8 +1,9 @@ | |||||||
| 
 | 
 | ||||||
| // #include "../lib/loop.hpp"
 | #include "../lib/runner.hpp" | ||||||
| // #include "../lib/asynco.hpp"
 | #include "../lib/asynco.hpp" | ||||||
| // #include "../example/asynco.hpp"
 |  | ||||||
| #include "../lib/event.hpp" | #include "../lib/event.hpp" | ||||||
|  | #include "../lib/rotor.hpp" | ||||||
|  | 
 | ||||||
| #include <iostream> | #include <iostream> | ||||||
| #include <unistd.h> | #include <unistd.h> | ||||||
| 
 | 
 | ||||||
| @ -10,49 +11,85 @@ using namespace std; | |||||||
| using namespace marcelb; | using namespace marcelb; | ||||||
| using namespace this_thread; | using namespace this_thread; | ||||||
| 
 | 
 | ||||||
| #ifndef ON_ASYNC | #ifndef ON_RUNNER | ||||||
| #define ON_ASYNC | #define ON_RUNNER | ||||||
| AsyncLoop on_async(8); | runner on_async; | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | void sleep_to (int _time) { | ||||||
|  |     promise<void> _promise; | ||||||
|  |     timeout t( [&]() { | ||||||
|  |         _promise.set_value(); | ||||||
|  |     }, _time); | ||||||
|  | 
 | ||||||
|  |     return _promise.get_future().get(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void promise_reject (int _time) { | ||||||
|  |     promise<void> _promise; | ||||||
|  |     timeout t( [&]() { | ||||||
|  |         try { | ||||||
|  |             // simulate except
 | ||||||
|  |             throw runtime_error("Error simulation"); | ||||||
|  |             _promise.set_value(); | ||||||
|  |         } catch (...) { | ||||||
|  |             _promise.set_exception(current_exception()); | ||||||
|  |         } | ||||||
|  |     }, _time); | ||||||
|  | 
 | ||||||
|  |     return _promise.get_future().get(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // ------------------ EXTEND OWN CLASS WITH EVENTS -------------------
 | ||||||
|  | 
 | ||||||
|  | class myOwnClass : public event<int> { | ||||||
|  |     public: | ||||||
|  |     myOwnClass() : event() {}; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| int main () { | int main () { | ||||||
| 
 | 
 | ||||||
|     // auto start = chrono::high_resolution_clock::now();
 |     auto start = rtime_ms(); | ||||||
|  | 
 | ||||||
|  |     // --------------- TIME ASYNCHRONOUS FUNCTIONS --------------
 | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Init interval and timeout; clear interval and timeout | ||||||
|  |     */ | ||||||
| 
 | 
 | ||||||
|     // interval inter1 ([&]() {
 |     // interval inter1 ([&]() {
 | ||||||
|     //     cout << "interval prvi " << chrono::duration_cast<chrono::milliseconds>(chrono::high_resolution_clock::now() - start).count() << endl;
 |     //     cout << "interval prvi " << rtime_ms() - start << endl;
 | ||||||
|     // }, 1000);
 |     // }, 1000);
 | ||||||
| 
 | 
 | ||||||
|     // interval inter2 ([&]() {
 |     // interval inter2 ([&]() {
 | ||||||
|     //     cout << "interval drugi " << chrono::duration_cast<chrono::milliseconds>(chrono::high_resolution_clock::now() - start).count() << endl;
 |     //     cout << "interval drugi " << rtime_ms() - start << endl;
 | ||||||
|     // }, 2000);
 |     // }, 2000);
 | ||||||
| 
 | 
 | ||||||
|     // interval inter3 ([&]() {
 |     // interval inter3 ([&]() {
 | ||||||
|     //     cout << "interval treći " << chrono::duration_cast<chrono::milliseconds>(chrono::high_resolution_clock::now() - start).count() << endl;
 |     //     cout << "interval treći " << rtime_ms() - start << endl;
 | ||||||
|     // }, 3000);
 |     // }, 3000);
 | ||||||
| 
 | 
 | ||||||
|     // timeout time1 ( [&] () {
 |     // timeout time1 ( [&] () {
 | ||||||
|     //     cout << "Close interval 1 i 2 " << chrono::duration_cast<chrono::milliseconds>(chrono::high_resolution_clock::now() - start).count() << endl;
 |     //     cout << "Close interval 1 i 2 " << rtime_ms() - start << endl;
 | ||||||
|     //     inter1.clear();
 |     //     inter1.clear();
 | ||||||
|     //     inter2.clear();
 |     //     inter2.clear();
 | ||||||
|     // }, 10000);
 |     // }, 10000);
 | ||||||
| 
 | 
 | ||||||
|     // timeout time2 ([&] () {
 |     // timeout time2 ([&] () {
 | ||||||
|     //     cout << "Close interval 3 " << chrono::duration_cast<chrono::milliseconds>(chrono::high_resolution_clock::now() - start).count() << endl;
 |     //     cout << "Close interval 3 " << rtime_ms() - start << endl;
 | ||||||
|     //     inter3.clear();
 |     //     inter3.clear();
 | ||||||
|  |     //     time1.clear();
 | ||||||
|     // }, 2000);
 |     // }, 2000);
 | ||||||
| 
 | 
 | ||||||
|     // cout << "zadataka: " << on_async.count_tasks() << " niti: " << on_async.count_threads() << endl;
 |     // ------------------------ MAKE FUNCTIONS ASYNCHRONOUS -------------------------
 | ||||||
| 
 | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Put task directly and get returned value - it is not recommended to use it | ||||||
|  |     */ | ||||||
| 
 | 
 | ||||||
|     //  for (int i = 0; i < 8; ++i) {
 |     // auto res1 = on_async.put_task( [] () {
 | ||||||
|     //     pool.put_task( [&] (int id) {
 |  | ||||||
|     //         this_thread::sleep_for(chrono::seconds(1));
 |  | ||||||
|     //         cout << a*i << endl;
 |  | ||||||
|     //     }, i);
 |  | ||||||
|     // }
 |  | ||||||
| 
 |  | ||||||
|     // auto res1 = pool.put_task( [] () {
 |  | ||||||
|     //     cout << "Jebiga " <<endl;
 |     //     cout << "Jebiga " <<endl;
 | ||||||
|     //     throw string ("jebiga!!");
 |     //     throw string ("jebiga!!");
 | ||||||
|     // });
 |     // });
 | ||||||
| @ -63,33 +100,80 @@ int main () { | |||||||
|     //     cout << except << endl;
 |     //     cout << except << endl;
 | ||||||
|     // }
 |     // }
 | ||||||
| 
 | 
 | ||||||
|     // auto res = pool.put_task( []() {
 |     /**
 | ||||||
|     //     this_thread::sleep_for(chrono::seconds(1));
 |      * Run an function asyncronic | ||||||
|     //     return 4;
 |     */ | ||||||
|     // });
 |  | ||||||
| 
 |  | ||||||
|     // cout << wait(asynco( [] () {
 |  | ||||||
|     //     sleep_for(chrono::seconds(1));
 |  | ||||||
|     //     cout << "RETURN" << endl;
 |  | ||||||
|     //     return 4;
 |  | ||||||
|     // })) << endl;
 |  | ||||||
| 
 | 
 | ||||||
|     // asynco( []() {
 |     // asynco( []() {
 | ||||||
|     //     sleep_for(2s);
 |     //     sleep_for(2s);   // only for simulate log duration function
 | ||||||
|     //     cout << "RETURN 2" << endl;
 |     //     cout << "asynco" << endl;
 | ||||||
|     //     return 5;
 |     //     return 5;
 | ||||||
|     // });
 |     // });
 | ||||||
| 
 | 
 | ||||||
|     // cout << wait(pool.put_task( []() {
 | 
 | ||||||
|     //     this_thread::sleep_for(chrono::seconds(1));
 |     /**
 | ||||||
|     //     return 7;
 |      * Wait after runned as async | ||||||
|     // })); 
 |      */ | ||||||
|  | 
 | ||||||
|  |     // auto a = asynco( []() {
 | ||||||
|  |     //     sleep_for(2s);   // only for simulate log duration function
 | ||||||
|  |     //     cout << "asynco" << endl;
 | ||||||
|  |     //     return 5;
 | ||||||
|  |     // });
 | ||||||
|  | 
 | ||||||
|  |     // cout << wait(move(a)) << endl;
 | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Wait async function call and use i cout | ||||||
|  |     */ | ||||||
|  | 
 | ||||||
|  |     // cout << wait(asynco( [] () {
 | ||||||
|  |     //     sleep_for(chrono::seconds(1)); // only for simulate log duration function
 | ||||||
|  |     //     cout << "wait end" << endl;
 | ||||||
|  |     //     return 4;
 | ||||||
|  |     // })) << endl;
 | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Sleep with timeout sleep implement | ||||||
|  |     */ | ||||||
|  | 
 | ||||||
|  |     // sleep_to(3000);
 | ||||||
|  |     // cout << "sleep_to " << rtime_ms() - start << endl;
 | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Catch promise reject | ||||||
|  |     */ | ||||||
|  | 
 | ||||||
|  |     // try {
 | ||||||
|  |     //     promise_reject(3000);
 | ||||||
|  |     // } catch (runtime_error err) {
 | ||||||
|  |     //     cout<< err.what() << endl;
 | ||||||
|  |     // }
 | ||||||
|  | 
 | ||||||
|  |     // cout << "promise_reject " << rtime_ms() - start << endl;
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|     // cout << wait(run1) << endl;
 |     /**
 | ||||||
|  |      * Nested asynchronous invocation | ||||||
|  |     */ | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     // asynco( [] {
 | ||||||
|  |     //     cout << "idemo ..." << endl;
 | ||||||
|  |     //     asynco( [] {
 | ||||||
|  |     //         cout << "ugdnježdena async funkcija " << endl;
 | ||||||
|  |     //     });
 | ||||||
|  |     // });
 | ||||||
|  | 
 | ||||||
|  |     // --------------- EVENTS -------------------
 | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * initialization of typed events | ||||||
|  |     */ | ||||||
| 
 | 
 | ||||||
|     // event<int, int> ev2int;
 |     // event<int, int> ev2int;
 | ||||||
|     // event<int, string> evintString;
 |     // event<int, string> evintString;
 | ||||||
|  |     // event<> evoid;
 | ||||||
| 
 | 
 | ||||||
|     // ev2int.on("sum", [](int a, int b) {
 |     // ev2int.on("sum", [](int a, int b) {
 | ||||||
|     //     cout << "Sum " << a+b << endl;
 |     //     cout << "Sum " << a+b << endl;
 | ||||||
| @ -99,15 +183,39 @@ int main () { | |||||||
|     //     cout << "Substract " << a-stoi(b) << endl;
 |     //     cout << "Substract " << a-stoi(b) << endl;
 | ||||||
|     // });
 |     // });
 | ||||||
| 
 | 
 | ||||||
|     // sleep(5);
 |     // evoid.on("void", []() {
 | ||||||
|  |     //     cout << "Void emited" << endl;
 | ||||||
|  |     // });
 | ||||||
|  | 
 | ||||||
|  |     // sleep(1);
 | ||||||
|  | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Emit | ||||||
|  |     */ | ||||||
| 
 | 
 | ||||||
|     // ev2int.emit("sum", 5, 8);
 |     // ev2int.emit("sum", 5, 8);
 | ||||||
| 
 | 
 | ||||||
|     // sleep(2);
 |     // sleep(1);
 | ||||||
|     // evintString.emit("substract", 3, to_string(2));
 |     // evintString.emit("substract", 3, to_string(2));
 | ||||||
| 
 | 
 | ||||||
|  |     // sleep(1);
 | ||||||
|  |     // evoid.emit("void");
 | ||||||
| 
 | 
 | ||||||
|     sleep(1000); |     /**
 | ||||||
|  |      * Own class  | ||||||
|  |     */ | ||||||
|  | 
 | ||||||
|  |     // myOwnClass myclass;
 | ||||||
|  | 
 | ||||||
|  |     // timeout t( [&] {
 | ||||||
|  |     //     myclass.emit("constructed", 1);
 | ||||||
|  |     // }, 200);
 | ||||||
|  | 
 | ||||||
|  |     // myclass.on("constructed", [] (int i) {
 | ||||||
|  |     //     cout << "Constructed " << i  << endl;
 | ||||||
|  |     // });
 | ||||||
|  | 
 | ||||||
|  |     sleep(10000); // only for testing
 | ||||||
| 
 | 
 | ||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user