{"id":465642,"date":"2025-07-02T09:00:06","date_gmt":"2025-07-02T09:00:06","guid":{"rendered":"http:\/\/savepearlharbor.com\/?p=465642"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=465642","title":{"rendered":"<span>C++ Event System \u043e\u0442 \u0438\u0434\u0435\u0438 \u0434\u043e \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438<\/span>"},"content":{"rendered":"<div><!--[--><!--]--><\/div>\n<div id=\"post-content-body\">\n<div>\n<div class=\"article-formatted-body article-formatted-body article-formatted-body_version-2\">\n<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<p>\u0422\u0435 \u0441\u0438\u0441\u0442\u0435\u043c\u044b \u0441\u043e\u0431\u044b\u0442\u0438\u0439, \u0441 \u043a\u043e\u0442\u043e\u0440\u044b\u043c\u0438 \u044f \u0441\u0442\u0430\u043b\u043a\u0438\u0432\u0430\u043b\u0441\u044f, \u0441\u0442\u0440\u0430\u0434\u0430\u043b\u0438 \u043e\u0442 \u0442\u0430\u043a\u0438\u0445 \u043f\u0440\u043e\u0431\u043b\u0435\u043c:<\/p>\n<ol>\n<li>\n<p><strong>\u041f\u0435\u0440\u0435\u0433\u0440\u0443\u0436\u0435\u043d\u043d\u043e\u0441\u0442\u044c \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0430<\/strong> \u2014 \u043c\u0430\u043a\u0440\u043e\u0441\u044b, \u0433\u0440\u043e\u043c\u043e\u0437\u0434\u043a\u0438\u0435 \u0448\u0430\u0431\u043b\u043e\u043d\u044b, \u043d\u0435\u043e\u0447\u0435\u0432\u0438\u0434\u043d\u044b\u0439 \u0441\u0438\u043d\u0442\u0430\u043a\u0441\u0438\u0441, \u043c\u043d\u043e\u0436\u0435\u0441\u0442\u0432\u0435\u043d\u043d\u0430\u044f \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u0438\u0437\u0430\u0446\u0438\u044f;<\/p>\n<\/li>\n<li>\n<p><strong>Broadcast<\/strong> \u2014 \u043a\u0430\u0436\u0434\u043e\u0435 \u0441\u043e\u0431\u044b\u0442\u0438\u0435 \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u0432\u0441\u0435\u043c \u0441\u043b\u0443\u0448\u0430\u0442\u0435\u043b\u044f\u043c, \u0430 \u043e\u043d\u0438 \u0441\u0430\u043c\u0438 \u0440\u0435\u0448\u0430\u044e\u0442, \u043d\u0443\u0436\u043d\u043e \u043b\u0438 \u0438\u043c \u0440\u0435\u0430\u0433\u0438\u0440\u043e\u0432\u0430\u0442\u044c. \u042d\u0442\u043e \u043f\u0440\u043e\u0441\u0442\u043e, \u043d\u043e \u0434\u043e\u0440\u043e\u0433\u043e;<\/p>\n<\/li>\n<li>\n<p><strong>Signal\/Slot \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0430<\/strong>, \u043a\u0430\u043a \u0432 Qt \u2014 \u0442\u0440\u0435\u0431\u0443\u0435\u0442 \u043a\u043e\u0434\u043e\u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u0438 \u0438 \u0442\u044f\u0436\u0435\u043b\u043e \u043e\u0442\u0434\u0435\u043b\u044f\u0435\u0442\u0441\u044f \u043e\u0442 \u0438\u043d\u0444\u0440\u0430\u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u044b.<\/p>\n<\/li>\n<\/ol>\n<p>\u042f \u0437\u0430\u0445\u043e\u0442\u0435\u043b \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u0442\u044c \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u0443\u044e \u0441\u0438\u0441\u0442\u0435\u043c\u0443 \u0441\u043e\u0431\u044b\u0442\u0438\u0439, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u0431\u044b\u043b\u0430 \u0431\u044b:<\/p>\n<ul>\n<li>\n<p><strong>\u043f\u0440\u043e\u0441\u0442\u043e\u0439<\/strong> \u0432 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u0438;<\/p>\n<\/li>\n<li>\n<p><strong>\u043f\u043e\u043d\u044f\u0442\u043d\u043e\u0439<\/strong> \u0432 \u043a\u043e\u0434\u0435;<\/p>\n<\/li>\n<li>\n<p><strong>\u0441\u0438\u043c\u043c\u0435\u0442\u0440\u0438\u0447\u043d\u043e\u0439<\/strong> \u2014 \u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0438 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u0435 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u043e\u0432 \u043f\u043e \u043e\u0434\u0438\u043d\u0430\u043a\u043e\u0432\u043e\u043c\u0443 \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0443;<\/p>\n<\/li>\n<li>\n<p><strong>\u043b\u0435\u0433\u043a\u043e\u0439<\/strong> \u2014 \u043c\u0438\u043d\u0438\u043c\u0443\u043c \u043a\u043e\u0434\u0430;<\/p>\n<\/li>\n<li>\n<p><strong>\u0441\u0430\u043c\u043e\u0434\u043e\u0441\u0442\u0430\u0442\u043e\u0447\u043d\u043e\u0439<\/strong> \u2014 \u0431\u0435\u0437 \u043c\u0430\u043a\u0440\u043e\u0441\u043e\u0432, \u0444\u0440\u0435\u0439\u043c\u0432\u043e\u0440\u043a\u043e\u0432, \u043a\u043e\u0434\u043e\u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u0438 \u0438\u043b\u0438 \u0432\u043d\u0435\u0448\u043d\u0438\u0445 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439;<\/p>\n<\/li>\n<\/ul>\n<h3>\u041f\u0440\u0438\u043c\u0435\u0440 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u044f<\/h3>\n<pre><code class=\"cpp\">class SomeClass { public: Event&lt;&gt; someEvent; Event&lt;int, SomeClass*&gt; otherEvent; private: void dispatchSomeEvent() { someEvent(); }      void dispatchOtherMethod()     {         otherEvent(5, this);     } };  class SomeOtherClass { public:     ~SomeOtherClass()     { m_someClass-&gt;someEvent.RemoveHandler(*this, &amp;SomeOtherClass::onSomeEvent); m_someClass-&gt;otherEvent.RemoveHandler(*this, &amp;SomeOtherClass::onSomeOtherEvent);     } void onSomeClassCreated(SomeClass* someClass) { m_someClass = someClass; m_someClass-&gt;someEvent.AddHandler(*this, &amp;SomeOtherClass::onSomeEvent); m_someClass-&gt;otherEvent.AddHandler(*this, &amp;SomeOtherClass::onSomeOtherEvent); } void onSomeEvent() { \/\/do something and unsubscribe m_someClass-&gt;someEvent.RemoveHandler(*this, &amp;SomeOtherClass::onSomeEvent); }  void onSomeOtherEvent(int val, SomeClass* obj) { } private: SomeClass* m_someClass; }; <\/code><\/pre>\n<h3>\u0420\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f<\/h3>\n<h4>details.inl<\/h4>\n<ol>\n<li>\n<p>\u0424\u0430\u0439\u043b details.inl<\/p>\n<\/li>\n<\/ol>\n<pre><code class=\"cpp\">namespace detail { inline std::size_t hash_combine(std::size_t seed, std::size_t value) noexcept { return seed ^ (value + 0x9e3779b97f4a7c15ULL + (seed &lt;&lt; 6) + (seed &gt;&gt; 2)); }  template&lt;typename... Args&gt; inline std::size_t GenerateID(void (*func)(Args...)) { return std::hash&lt;void*&gt;()(reinterpret_cast&lt;void*&gt;(func)); }  template&lt;typename Obj, typename Meth&gt; inline std::size_t GenerateID(Obj* obj, Meth method) { constexpr std::size_t N = sizeof(Meth); constexpr std::size_t W = sizeof(std::size_t); constexpr std::size_t CHUNKS = (N + W - 1) \/ W;  std::size_t pieces[CHUNKS]{}; std::memcpy(pieces, &amp;method, N);  std::size_t h = std::hash&lt;std::type_index&gt;{}(typeid(Obj)); for (std::size_t v : pieces) h = hash_combine(h, std::hash&lt;std::size_t&gt;{}(v));  h = hash_combine(h, std::hash&lt;void*&gt;{}(static_cast&lt;void*&gt;(obj))); return h; } } <\/code><\/pre>\n<p>\u041f\u0440\u043e\u0441\u0442\u043e \u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u044f \u0445\u0435\u0448\u0430 \u0434\u043b\u044f \u0440\u0430\u0437\u043d\u044b\u0445 \u0442\u0438\u043f\u043e\u0432 \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u043e\u0432. \u0412\u043d\u0438\u043c\u0430\u043d\u0438\u044f \u0441\u0442\u043e\u0438\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u0438\u0435 \u0445\u0435\u0448\u0430 \u0444\u0443\u043d\u043a\u0446\u0438\u0438-\u043c\u0435\u0442\u043e\u0434\u0430. \u0418\u0437-\u0437\u0430 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0433\u043e \u0432\u044b\u0445\u043e\u0434\u0430 \u0440\u0430\u0437\u043c\u0435\u0440\u0430 \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044f \u043c\u0435\u0442\u043e\u0434\u0430 \u0437\u0430 8 \u0431\u0430\u0439\u0442 \u043f\u0440\u0438\u0448\u043b\u043e\u0441\u044c \u0441\u0434\u0435\u043b\u0430\u0442\u044c \u0444\u0443\u043d\u043a\u0446\u0438\u044e \u0447\u0443\u0442\u044c \u0431\u043e\u043b\u0435\u0435 \u0441\u043b\u043e\u0436\u043d\u043e\u0439.<\/p>\n<h4>EventHandler<\/h4>\n<p>\u041f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u043a\u0430\u043a \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u0442\u0441\u044f \u0431\u0430\u0437\u043e\u0432\u044b\u0439 EventHandler:<\/p>\n<pre><code class=\"cpp\">template&lt;typename ...Args&gt; class EventHandler { public: virtual void call(Args&amp;&amp;... args) = 0;  size_t GetHandlerID() const { return m_handlerID; }  void Invalidate() { m_valid = false; } bool IsValid() const { return m_valid; }  protected: size_t m_handlerID; private: bool m_valid = true; };<\/code><\/pre>\n<p>\u042d\u0442\u043e \u0431\u0430\u0437\u0430 \u0434\u043b\u044f \u0440\u0430\u0437\u043d\u044b\u0445 \u0432\u0438\u0434\u043e\u0432 \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u043e\u0432. \u0427\u0438\u0441\u0442\u0430\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u044f \u0432\u044b\u0437\u043e\u0432\u0430.<br \/> \u0422\u0443\u0442 \u0436\u0435 \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u043d\u0430 \u0438\u043d\u0432\u0430\u043b\u0438\u0434\u0430\u0446\u0438\u044f \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u0430 \u0434\u043b\u044f &#171;\u0433\u043e\u0440\u044f\u0447\u0435\u0433\u043e&#187; \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u044f.<\/p>\n<h4>MethodEventHandler<\/h4>\n<p>\u041d\u0430\u0441\u043b\u0435\u0434\u043d\u0438\u043a \u0434\u043b\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u0439-\u0447\u043b\u0435\u043d\u043e\u0432:<\/p>\n<pre><code class=\"cpp\">template&lt;typename Object, typename ...Args&gt; class MethodEventHandler final : public EventHandler&lt;Args...&gt; { public: using MethodType = void(Object::*)(Args...); public: MethodEventHandler(Object&amp; object, MethodType method) : m_object(object), m_method(method) { this-&gt;m_handlerID = detail::GenerateID(&amp;object, method); }  void call(Args&amp;&amp;... args) override { (m_object.*m_method)(std::forward&lt;Args&gt;(args)...); }  private: Object&amp; m_object; MethodType m_method; };<\/code><\/pre>\n<ul>\n<li>\n<p>\u0425\u0440\u0430\u043d\u0438\u0442 \u0441\u0441\u044b\u043b\u043a\u0443 \u043d\u0430 \u043e\u0431\u044a\u0435\u043a\u0442 \u0438 \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044c \u043d\u0430 \u0444\u0443\u043d\u043a\u0446\u0438\u044e.<\/p>\n<\/li>\n<li>\n<p>\u041f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0435\u0442 call, \u0442\u0430\u043a \u043a\u0430\u043a \u0441\u0438\u043d\u0442\u0430\u043a\u0441\u0438\u0441 \u0432\u044b\u0437\u043e\u0432\u0430 \u0441\u043f\u0435\u0446\u0438\u0444\u0438\u0447\u0435\u0441\u043a\u0438\u0439.<\/p>\n<\/li>\n<li>\n<p>\u0413\u0435\u043d\u0435\u0440\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u0430\u0439\u0434\u0438(\u0445\u0435\u0448) \u043d\u0430 \u0431\u0430\u0437\u0435 \u043e\u0431\u044a\u0435\u043a\u0442\u0430 \u0438 \u0444\u0443\u043d\u043a\u0446\u0438\u0438-\u0447\u043b\u0435\u043d\u0430.<\/p>\n<\/li>\n<\/ul>\n<h4>FunctionEventHandler<\/h4>\n<p>\u041d\u0430\u0441\u043b\u0435\u0434\u043d\u0438\u043a \u0434\u043b\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u0438:<\/p>\n<pre><code class=\"cpp\">template&lt;typename ...Args&gt; class FunctionEventHandler : public EventHandler&lt;Args...&gt; { public: using FunctionType = void(*)(Args...); public: FunctionEventHandler(FunctionType function) : m_function(function) { this-&gt;m_handlerID = detail::GenerateID(function); }  void call(Args&amp;&amp;... args) override { (*m_function)(std::forward&lt;Args&gt;(args)...); }  private: FunctionType m_function; };<\/code><\/pre>\n<p>\u0417\u0434\u0435\u0441\u044c \u2014 \u0442\u043e\u0442 \u0436\u0435 \u043f\u043e\u0434\u0445\u043e\u0434, \u0442\u043e\u043b\u044c\u043a\u043e \u0434\u043b\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u0439.<\/p>\n<h4>Event<\/h4>\n<p>\u0420\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f \u0441\u0430\u043c\u043e\u0433\u043e Event:<\/p>\n<pre><code class=\"cpp\">template&lt;typename ...Args&gt; class Event { using HandlerType = EventHandler&lt;Args...&gt;;  public: \/\/\u0417\u0430\u0449\u0438\u0442\u0438\u043c\u0441\u044f \u043e\u0442 \u043a\u043e\u043f\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u0438 \u043f\u0435\u0440\u0435\u043d\u043e\u0441\u0430 - \u044d\u0442\u043e \u043d\u0435\u0434\u043e\u043f\u0443\u0441\u0442\u0438\u043c\u0430\u044f \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u044f(\u0432\u043e \u0432\u0441\u044f\u043a\u043e\u043c \u0441\u043b\u0443\u0447\u0430\u0435 \u043f\u043e\u043a\u0430) Event() = default; Event(const Event&amp;) = delete; Event&amp; operator=(const Event&amp;) = delete; Event(Event&amp;&amp;) = delete; Event&amp; operator=(Event&amp;&amp;) = delete;  template&lt;typename Object&gt; void AddHandler(Object&amp; object, MethodEventHandler&lt;Object, Args...&gt;::MethodType method) { if (HasId(detail::GenerateID(&amp;object, method))) return;  (m_dispatching ? m_added_handlers : m_handlers).emplace_back(std::make_unique&lt;MethodEventHandler&lt;Object, Args...&gt;&gt;(object, method)); }  void AddHandler(FunctionEventHandler&lt;Args...&gt;::FunctionType function) { if (HasId(detail::GenerateID(function))) return;  (m_dispatching ? m_added_handlers : m_handlers).emplace_back(std::make_unique&lt;FunctionEventHandler&lt;Args...&gt;&gt;(function)); }  template&lt;class F&gt; void AddHandler(F) = delete; \/\/ \u041b\u044f\u043c\u0431\u0434\u044b \u0438 \u0444\u0443\u043d\u043a\u0442\u043e\u0440\u044b \u043d\u0435 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u044e\u0442\u0441\u044f \u2014 \u043d\u0435\u043b\u044c\u0437\u044f \u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e \u043e\u0442\u043f\u0438\u0441\u0430\u0442\u044c\u0441\u044f  template&lt;typename Object&gt; void RemoveHandler(Object&amp; object, MethodEventHandler&lt;Object, Args...&gt;::MethodType method) { RemoveById(detail::GenerateID(&amp;object, method)); }  void RemoveHandler(FunctionEventHandler&lt;Args...&gt;::FunctionType function) { RemoveById(detail::GenerateID(function)); }  template&lt;class F&gt; void RemoveHandler(F) = delete; \/\/\u0421\u0438\u043c\u043c\u0435\u0442\u0440\u0438\u044f  void operator()(Args... args) { m_dispatching = true; for (auto&amp; handler : m_handlers) if (handler-&gt;IsValid()) handler-&gt;call(std::forward&lt;Args&gt;(args)...); m_dispatching = false;  \/\/ \u0423\u0434\u0430\u043b\u044f\u0435\u043c \u043d\u0435\u0432\u0430\u043b\u0438\u0434\u043d\u044b\u0435 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0438 \u043f\u043e\u0441\u043b\u0435 \u0437\u0430\u0432\u0435\u0440\u0448\u0435\u043d\u0438\u044f \u0432\u044b\u0437\u043e\u0432\u043e\u0432         if (m_wasHotRemoved)         {     m_handlers.erase(std::remove_if(m_handlers.begin(), m_handlers.end(), [](const auto&amp; handler) { return !handler-&gt;IsValid(); }), m_handlers.end());             m_wasHotRemoved = false;         } for (auto&amp; handler : m_added_handlers) m_handlers.push_back(std::move(handler)); m_added_handlers.clear(); }  private: std::vector&lt;std::unique_ptr&lt;HandlerType&gt;&gt; m_handlers; std::vector&lt;std::unique_ptr&lt;HandlerType&gt;&gt; m_added_handlers;  bool m_dispatching = false;     bool m_wasHotRemoved = false;  bool HasId(std::size_t id) const { auto pred = [&amp;](const auto&amp; handler) { return handler-&gt;GetHandlerID() == id; }; return std::any_of(m_handlers.begin(), m_handlers.end(), pred) || std::any_of(m_added_handlers.begin(), m_added_handlers.end(), pred); }  void RemoveById(std::size_t id) { auto pred = [&amp;](const auto&amp; handler) { return handler-&gt;GetHandlerID() == id; };  if (m_dispatching) { if (auto it = std::find_if(m_handlers.begin(), m_handlers.end(), pred); it != m_handlers.end()) {                 (*it)-&gt;Invalidate();                 m_wasHotRemoved = true;             } } else { m_handlers.erase(std::remove_if(m_handlers.begin(), m_handlers.end(), pred), m_handlers.end()); }  m_added_handlers.erase(std::remove_if(m_added_handlers.begin(), m_added_handlers.end(), pred), m_added_handlers.end()); } };<\/code><\/pre>\n<ul>\n<li>\n<p>\u041f\u0440\u0435\u0434\u043b\u0430\u0433\u0430\u0435\u0442\u0441\u044f 2 \u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u044f \u0438 2 \u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u044f \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u043e\u0432 \u0432 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u043e\u0442 \u0442\u0438\u043f\u0430 \u043a\u043e\u043b\u0431\u0435\u043a\u0430.<\/p>\n<\/li>\n<li>\n<p>\u0417\u0430 \u0441\u0447\u0435\u0442 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0438 \u0438\u043d\u0432\u0430\u043b\u0438\u0434\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0445\u0435\u043d\u0434\u043b\u0435\u0440 \u043c\u043e\u0436\u043d\u043e \u0443\u0434\u0430\u043b\u044f\u0442\u044c \u0432\u043e  \u0432\u0440\u0435\u043c\u044f \u0434\u0438\u0441\u043f\u0430\u0442\u0447\u0438\u043d\u0433\u0430 \u0441\u043e\u0431\u044b\u0442\u0438\u044f.<\/p>\n<\/li>\n<li>\n<p>\u0420\u0430\u0431\u043e\u0442\u0430 \u0441 ID(\u0445\u0435\u0448\u043e\u043c) &#8212; \u0441\u043a\u0440\u044b\u0442\u0430\u044f \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f, \u043f\u043e \u044d\u0442\u043e\u043c\u0443 \u0432\u0441\u0435 \u0443\u0431\u0440\u0430\u043d\u043e \u0432 private \u0437\u043e\u043d\u0443 \u0438 \u0443\u043d\u0438\u0444\u0438\u0446\u0438\u0440\u043e\u0432\u0430\u043d\u043e \u0434\u043b\u044f \u0440\u0430\u0437\u043d\u044b\u0445 \u0442\u0438\u043f\u043e\u0432 \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u043e\u0432.<\/p>\n<\/li>\n<li>\n<p>\u0421\u043f\u0435\u0446\u0438\u0430\u043b\u044c\u043d\u044b\u0439 \u0444\u043b\u0430\u0436\u043e\u043a m_dispatching, \u0447\u0442\u043e\u0431\u044b \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u0442\u044c &#171;\u0433\u043e\u0440\u044f\u0447\u0430\u044f \u0444\u0430\u0437\u0430&#187; \u0438\u043b\u0438 \u043d\u0435\u0442. \u0415\u0441\u043b\u0438 \u0444\u0430\u0437\u0430 \u043d\u0435 \u0433\u043e\u0440\u044f\u0447\u0430\u044f, \u0442\u043e \u0438 \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0442\u044c\/\u0443\u0434\u0430\u043b\u044f\u0442\u044c \u043c\u043e\u0436\u043d\u043e \u043d\u0430\u043f\u0440\u044f\u043c\u0443\u044e.<\/p>\n<\/li>\n<\/ul>\n<h3>\u041d\u0435\u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0442\u043e\u043d\u043a\u043e\u0441\u0442\u0438 \u0433\u043e\u0440\u044f\u0447\u0435\u0439 \u0444\u0430\u0437\u044b<\/h3>\n<p>\u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u0435\/\u0423\u0434\u0430\u043b\u0435\u043d\u0438\u0435 \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u043e\u0432 \u0432\u043e \u0432\u0440\u0435\u043c\u044f \u0434\u0438\u0441\u043f\u0430\u0442\u0447\u0438\u043d\u0433\u0430:<\/p>\n<ul>\n<li>\n<p>\u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u044b\u0439 \u0445\u0435\u043d\u0434\u043b\u0435\u0440 &#8212; \u043d\u0435 \u0431\u0443\u0434\u0435\u0442 \u0432\u044b\u0437\u0432\u0430\u043d \u0432 \u0442\u0435\u043a\u0443\u0449\u0435\u043c \u0434\u0438\u0441\u043f\u0430\u0442\u0447\u0438\u043d\u0433\u0435.<\/p>\n<\/li>\n<li>\n<p>\u0423\u0434\u0430\u043b\u0435\u043d\u043d\u044b\u0439 \u0445\u0435\u043d\u0434\u043b\u0435\u0440 &#8212; \u0431\u0443\u0434\u0435\u0442 \u043f\u043e\u043c\u0435\u0447\u0435\u043d \u043a\u0430\u043a \u043d\u0435\u0432\u0430\u043b\u0438\u0434\u043d\u044b\u0439 \u0438 \u043d\u0435 \u0431\u0443\u0434\u0435\u0442 \u0432\u044b\u0437\u0432\u0430\u043d \u0432 \u0442\u0435\u043a\u0443\u0449\u0435\u043c \u0434\u0438\u0441\u043f\u0430\u0442\u0447\u0438\u043d\u0433\u0435.<\/p>\n<\/li>\n<\/ul>\n<p><strong>\u0423\u0434\u0430\u043b\u0435\u043d\u0438\u0435 \u043c\u0433\u043d\u043e\u0432\u0435\u043d\u043d\u043e\u0435, \u0430 \u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u043e\u0442\u043b\u043e\u0436\u0435\u043d\u043d\u043e\u0435.<\/strong><\/p>\n<h3>\u041f\u0440\u043e\u0431\u043b\u0435\u043c\u044b \u0438 \u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u0438\u044f<\/h3>\n<ol>\n<li>\n<p>\u041c\u043e\u0436\u043d\u043e \u0437\u0430\u0431\u044b\u0442\u044c \u043e\u0442\u043f\u0438\u0441\u0430\u0442\u044c\u0441\u044f. \u0415\u0441\u043b\u0438 \u043e\u0431\u044a\u0435\u043a\u0442 \u0431\u0443\u0434\u0435\u0442 \u0443\u043d\u0438\u0447\u0442\u043e\u0436\u0435\u043d \u0438 \u043d\u0435 \u043e\u0442\u043f\u0438\u0448\u0435\u0442\u0441\u044f \u043e\u0442 \u0441\u043e\u0431\u044b\u0442\u0438\u044f, \u0432\u043e\u0437\u043d\u0438\u043a\u043d\u0435\u0442 UB \u0438\u0437-\u0437\u0430 \u0432\u0438\u0441\u044f\u0447\u0435\u0433\u043e \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044f.<\/p>\n<\/li>\n<li>\n<p>\u041b\u044f\u043c\u0431\u0434\u044b \u0437\u0430\u043f\u0440\u0435\u0449\u0435\u043d\u044b, \u043f\u043e\u0441\u043a\u043e\u043b\u044c\u043a\u0443 \u0438\u0445 \u043d\u0435\u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e \u043e\u0442\u043f\u0438\u0441\u0430\u0442\u044c.<\/p>\n<\/li>\n<li>\n<p>\u041d\u0435 thread-safe.<\/p>\n<\/li>\n<\/ol>\n<h3>\u0414\u043b\u044f \u0447\u0435\u0433\u043e \u044d\u0442\u0430 \u0441\u0438\u0441\u0442\u0435\u043c\u0430 \u043d\u0435 \u043f\u043e\u0434\u043e\u0439\u0434\u0435\u0442<\/h3>\n<ol>\n<li>\n<p>\u00ab\u042f \u0445\u043e\u0447\u0443 \u0441\u043b\u0443\u0448\u0430\u0442\u044c \u0441\u043e\u0431\u044b\u0442\u0438\u0435, \u043d\u043e\u00a0\u044f \u043d\u0435\u00a0\u0437\u043d\u0430\u044e \u043a\u0442\u043e \u0435\u0433\u043e \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u0442\u00bb.<\/p>\n<\/li>\n<li>\n<p>\u00ab\u042f \u043d\u0435\u00a0\u0437\u043d\u0430\u044e(\u0438\u043b\u0438 \u043d\u0435\u00a0\u0445\u043e\u0447\u0443 \u0437\u043d\u0430\u0442\u044c) \u0432\u0440\u0435\u043c\u044f \u0436\u0438\u0437\u043d\u0438 \u043e\u0431\u044a\u0435\u043a\u0442\u0430 \u0441\u00a0\u0441\u043e\u0431\u044b\u0442\u0438\u0435\u043c\u00bb.<\/p>\n<\/li>\n<\/ol>\n<h3>\u0414\u043b\u044f \u0447\u0435\u0433\u043e \u044d\u0442\u0430 \u0441\u0438\u0441\u0442\u0435\u043c\u0430 \u043f\u043e\u0434\u043e\u0439\u0434\u0435\u0442<\/h3>\n<ol>\n<li>\n<p>\u0412\u044b \u0438\u043c\u0435\u0435\u0442\u0435 \u0434\u043e\u0441\u0442\u0443\u043f \u043a\u00a0\u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0443 \u0441\u043e\u0431\u044b\u0442\u0438\u044f.<\/p>\n<\/li>\n<li>\n<p>\u0412\u044b \u0442\u043e\u0447\u043d\u043e \u0437\u043d\u0430\u0435\u0442\u0435 \u0432\u0440\u0435\u043c\u044f \u0436\u0438\u0437\u043d\u0438 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0430 \u0441\u043e\u0431\u044b\u0442\u0438\u044f(\u0438\u043b\u0438 \u043c\u043e\u0436\u0435\u0442\u0435 \u0443\u0431\u0435\u0434\u0438\u0442\u044c\u0441\u044f, \u0447\u0442\u043e\u00a0\u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a \u0436\u0438\u0432)<\/p>\n<\/li>\n<\/ol>\n<h3>\u0412\u043e\u0437\u043c\u043e\u0436\u043d\u044b\u0435 \u0443\u043b\u0443\u0447\u0448\u0435\u043d\u0438\u044f<\/h3>\n<ol>\n<li>\n<p>ScopedEventHandler\u00a0\u2014 RAII\u2011\u043c\u0435\u0445\u0430\u043d\u0438\u0437\u043c, \u043e\u0442\u043f\u0438\u0441\u043a\u0430 \u043f\u043e\u00a0\u0440\u0430\u0437\u0440\u0443\u0448\u0435\u043d\u0438\u044e. \u0420\u0435\u0448\u0438\u0442 \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u0443 \u0437\u0430\u0431\u044b\u0432\u0447\u0438\u0432\u043e\u0441\u0442\u0438, \u0430\u00a0\u0442\u0430\u043a\u00a0\u0436\u0435 \u0434\u0430\u0441\u0442 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u043f\u043e\u0434\u043f\u0438\u0441\u044b\u0432\u0430\u0442\u044c \u043b\u044f\u043c\u0431\u0434\u044b \u0438 \u0444\u0443\u043d\u043a\u0442\u043e\u0440\u044b.<\/p>\n<\/li>\n<li>\n<p>\u041e\u0442\u043a\u0430\u0437 \u043e\u0442\u00a0\u0432\u0438\u0440\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0445 \u0432\u044b\u0437\u043e\u0432\u043e\u0432 \u043c\u043e\u0436\u0435\u0442 \u0434\u0430\u0442\u044c \u043f\u0440\u0438\u0440\u043e\u0441\u0442 \u0437\u0430\u00a0\u0441\u0447\u0451\u0442 \u0443\u0441\u0442\u0440\u0430\u043d\u0435\u043d\u0438\u044f \u043e\u0431\u0440\u0430\u0449\u0435\u043d\u0438\u044f \u043a <code>vtable<\/code>.<\/p>\n<\/li>\n<\/ol>\n<h3>\u0417\u0430\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435<\/h3>\n<p>\u0420\u0435\u0448\u0435\u043d\u0438\u0435 \u0443\u0440\u043e\u0432\u043d\u044f \u00ab\u0434\u043e\u0431\u0430\u0432\u0438\u043b \u0438 \u043f\u043e\u0435\u0445\u0430\u043b\u043e\u00bb. \u0422\u0440\u0435\u0431\u0443\u0435\u0442 \u0432\u043d\u0438\u043c\u0430\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u0438, \u043f\u043e\u0441\u043a\u043e\u043b\u044c\u043a\u0443, \u0437\u0430\u0431\u044b\u0432 \u043e\u0442\u043f\u0438\u0441\u0430\u0442\u044c\u0441\u044f, \u043c\u043e\u0436\u043d\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c UB. \u041f\u043e\u0434\u0445\u043e\u0434\u0438\u0442 \u0434\u043b\u044f\u00a0\u043e\u0434\u043d\u043e\u043f\u043e\u0442\u043e\u0447\u043d\u043e\u0439 \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u044b, \u0433\u0434\u0435 \u044f\u0432\u043d\u043e \u0438\u0437\u0432\u0435\u0441\u0442\u043d\u044b \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438 \u0441\u043e\u0431\u044b\u0442\u0438\u0439 \u0438 \u0438\u0445 \u0436\u0438\u0437\u043d\u0435\u043d\u043d\u044b\u0439 \u0446\u0438\u043a\u043b.<\/p>\n<p>\u0415\u0441\u043b\u0438 \u0431\u0443\u0434\u0435\u0442 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e \u0432\u00a0\u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0435\u043c \u043f\u043e\u0441\u0442\u0435 \u043d\u0430\u043f\u0438\u0448\u0443:<\/p>\n<ol>\n<li>\n<p>ScopedEventHandler<\/p>\n<\/li>\n<li>\n<p>\u0420\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044e \u0431\u0435\u0437 virtual<\/p>\n<\/li>\n<\/ol>\n<p>\u041a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u0438 \u0438 \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u0438\u0432\u043d\u0430\u044f \u043a\u0440\u0438\u0442\u0438\u043a\u0430 \u043f\u0440\u0438\u0432\u0435\u0442\u0441\u0442\u0432\u0443\u0435\u0442\u0441\u044f. \u0411\u0443\u0434\u0435\u0442 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e \u043c\u043d\u0435\u043d\u0438\u0435 \u044d\u043a\u0441\u043f\u0435\u0440\u0442\u043e\u0432.<\/p>\n<p><a href=\"https:\/\/github.com\/2pizza\/cpp-event-system\" rel=\"noopener noreferrer nofollow\">GitHub<\/a><\/p>\n<\/div>\n<\/div>\n<\/div>\n<p><!----><!----><\/div>\n<p><!----><!----><br \/> \u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"https:\/\/habr.com\/ru\/articles\/924138\/\"> https:\/\/habr.com\/ru\/articles\/924138\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<div><!--[--><!--]--><\/div>\n<div id=\"post-content-body\">\n<div>\n<div class=\"article-formatted-body article-formatted-body article-formatted-body_version-2\">\n<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<p>\u0422\u0435 \u0441\u0438\u0441\u0442\u0435\u043c\u044b \u0441\u043e\u0431\u044b\u0442\u0438\u0439, \u0441 \u043a\u043e\u0442\u043e\u0440\u044b\u043c\u0438 \u044f \u0441\u0442\u0430\u043b\u043a\u0438\u0432\u0430\u043b\u0441\u044f, \u0441\u0442\u0440\u0430\u0434\u0430\u043b\u0438 \u043e\u0442 \u0442\u0430\u043a\u0438\u0445 \u043f\u0440\u043e\u0431\u043b\u0435\u043c:<\/p>\n<ol>\n<li>\n<p><strong>\u041f\u0435\u0440\u0435\u0433\u0440\u0443\u0436\u0435\u043d\u043d\u043e\u0441\u0442\u044c \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0430<\/strong> \u2014 \u043c\u0430\u043a\u0440\u043e\u0441\u044b, \u0433\u0440\u043e\u043c\u043e\u0437\u0434\u043a\u0438\u0435 \u0448\u0430\u0431\u043b\u043e\u043d\u044b, \u043d\u0435\u043e\u0447\u0435\u0432\u0438\u0434\u043d\u044b\u0439 \u0441\u0438\u043d\u0442\u0430\u043a\u0441\u0438\u0441, \u043c\u043d\u043e\u0436\u0435\u0441\u0442\u0432\u0435\u043d\u043d\u0430\u044f \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u0438\u0437\u0430\u0446\u0438\u044f;<\/p>\n<\/li>\n<li>\n<p><strong>Broadcast<\/strong> \u2014 \u043a\u0430\u0436\u0434\u043e\u0435 \u0441\u043e\u0431\u044b\u0442\u0438\u0435 \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u0432\u0441\u0435\u043c \u0441\u043b\u0443\u0448\u0430\u0442\u0435\u043b\u044f\u043c, \u0430 \u043e\u043d\u0438 \u0441\u0430\u043c\u0438 \u0440\u0435\u0448\u0430\u044e\u0442, \u043d\u0443\u0436\u043d\u043e \u043b\u0438 \u0438\u043c \u0440\u0435\u0430\u0433\u0438\u0440\u043e\u0432\u0430\u0442\u044c. \u042d\u0442\u043e \u043f\u0440\u043e\u0441\u0442\u043e, \u043d\u043e \u0434\u043e\u0440\u043e\u0433\u043e;<\/p>\n<\/li>\n<li>\n<p><strong>Signal\/Slot \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0430<\/strong>, \u043a\u0430\u043a \u0432 Qt \u2014 \u0442\u0440\u0435\u0431\u0443\u0435\u0442 \u043a\u043e\u0434\u043e\u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u0438 \u0438 \u0442\u044f\u0436\u0435\u043b\u043e \u043e\u0442\u0434\u0435\u043b\u044f\u0435\u0442\u0441\u044f \u043e\u0442 \u0438\u043d\u0444\u0440\u0430\u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u044b.<\/p>\n<\/li>\n<\/ol>\n<p>\u042f \u0437\u0430\u0445\u043e\u0442\u0435\u043b \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u0442\u044c \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u0443\u044e \u0441\u0438\u0441\u0442\u0435\u043c\u0443 \u0441\u043e\u0431\u044b\u0442\u0438\u0439, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u0431\u044b\u043b\u0430 \u0431\u044b:<\/p>\n<ul>\n<li>\n<p><strong>\u043f\u0440\u043e\u0441\u0442\u043e\u0439<\/strong> \u0432 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u0438;<\/p>\n<\/li>\n<li>\n<p><strong>\u043f\u043e\u043d\u044f\u0442\u043d\u043e\u0439<\/strong> \u0432 \u043a\u043e\u0434\u0435;<\/p>\n<\/li>\n<li>\n<p><strong>\u0441\u0438\u043c\u043c\u0435\u0442\u0440\u0438\u0447\u043d\u043e\u0439<\/strong> \u2014 \u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0438 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u0435 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u043e\u0432 \u043f\u043e \u043e\u0434\u0438\u043d\u0430\u043a\u043e\u0432\u043e\u043c\u0443 \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0443;<\/p>\n<\/li>\n<li>\n<p><strong>\u043b\u0435\u0433\u043a\u043e\u0439<\/strong> \u2014 \u043c\u0438\u043d\u0438\u043c\u0443\u043c \u043a\u043e\u0434\u0430;<\/p>\n<\/li>\n<li>\n<p><strong>\u0441\u0430\u043c\u043e\u0434\u043e\u0441\u0442\u0430\u0442\u043e\u0447\u043d\u043e\u0439<\/strong> \u2014 \u0431\u0435\u0437 \u043c\u0430\u043a\u0440\u043e\u0441\u043e\u0432, \u0444\u0440\u0435\u0439\u043c\u0432\u043e\u0440\u043a\u043e\u0432, \u043a\u043e\u0434\u043e\u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u0438 \u0438\u043b\u0438 \u0432\u043d\u0435\u0448\u043d\u0438\u0445 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439;<\/p>\n<\/li>\n<\/ul>\n<h3>\u041f\u0440\u0438\u043c\u0435\u0440 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u044f<\/h3>\n<pre><code class=\"cpp\">class SomeClass { public: Event&lt;&gt; someEvent; Event&lt;int, SomeClass*&gt; otherEvent; private: void dispatchSomeEvent() { someEvent(); }      void dispatchOtherMethod()     {         otherEvent(5, this);     } };  class SomeOtherClass { public:     ~SomeOtherClass()     { m_someClass-&gt;someEvent.RemoveHandler(*this, &amp;SomeOtherClass::onSomeEvent); m_someClass-&gt;otherEvent.RemoveHandler(*this, &amp;SomeOtherClass::onSomeOtherEvent);     } void onSomeClassCreated(SomeClass* someClass) { m_someClass = someClass; m_someClass-&gt;someEvent.AddHandler(*this, &amp;SomeOtherClass::onSomeEvent); m_someClass-&gt;otherEvent.AddHandler(*this, &amp;SomeOtherClass::onSomeOtherEvent); } void onSomeEvent() { \/\/do something and unsubscribe m_someClass-&gt;someEvent.RemoveHandler(*this, &amp;SomeOtherClass::onSomeEvent); }  void onSomeOtherEvent(int val, SomeClass* obj) { } private: SomeClass* m_someClass; }; <\/code><\/pre>\n<h3>\u0420\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f<\/h3>\n<h4>details.inl<\/h4>\n<ol>\n<li>\n<p>\u0424\u0430\u0439\u043b details.inl<\/p>\n<\/li>\n<\/ol>\n<pre><code class=\"cpp\">namespace detail { inline std::size_t hash_combine(std::size_t seed, std::size_t value) noexcept { return seed ^ (value + 0x9e3779b97f4a7c15ULL + (seed &lt;&lt; 6) + (seed &gt;&gt; 2)); }  template&lt;typename... Args&gt; inline std::size_t GenerateID(void (*func)(Args...)) { return std::hash&lt;void*&gt;()(reinterpret_cast&lt;void*&gt;(func)); }  template&lt;typename Obj, typename Meth&gt; inline std::size_t GenerateID(Obj* obj, Meth method) { constexpr std::size_t N = sizeof(Meth); constexpr std::size_t W = sizeof(std::size_t); constexpr std::size_t CHUNKS = (N + W - 1) \/ W;  std::size_t pieces[CHUNKS]{}; std::memcpy(pieces, &amp;method, N);  std::size_t h = std::hash&lt;std::type_index&gt;{}(typeid(Obj)); for (std::size_t v : pieces) h = hash_combine(h, std::hash&lt;std::size_t&gt;{}(v));  h = hash_combine(h, std::hash&lt;void*&gt;{}(static_cast&lt;void*&gt;(obj))); return h; } } <\/code><\/pre>\n<p>\u041f\u0440\u043e\u0441\u0442\u043e \u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u044f \u0445\u0435\u0448\u0430 \u0434\u043b\u044f \u0440\u0430\u0437\u043d\u044b\u0445 \u0442\u0438\u043f\u043e\u0432 \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u043e\u0432. \u0412\u043d\u0438\u043c\u0430\u043d\u0438\u044f \u0441\u0442\u043e\u0438\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u0438\u0435 \u0445\u0435\u0448\u0430 \u0444\u0443\u043d\u043a\u0446\u0438\u0438-\u043c\u0435\u0442\u043e\u0434\u0430. \u0418\u0437-\u0437\u0430 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0433\u043e \u0432\u044b\u0445\u043e\u0434\u0430 \u0440\u0430\u0437\u043c\u0435\u0440\u0430 \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044f \u043c\u0435\u0442\u043e\u0434\u0430 \u0437\u0430 8 \u0431\u0430\u0439\u0442 \u043f\u0440\u0438\u0448\u043b\u043e\u0441\u044c \u0441\u0434\u0435\u043b\u0430\u0442\u044c \u0444\u0443\u043d\u043a\u0446\u0438\u044e \u0447\u0443\u0442\u044c \u0431\u043e\u043b\u0435\u0435 \u0441\u043b\u043e\u0436\u043d\u043e\u0439.<\/p>\n<h4>EventHandler<\/h4>\n<p>\u041f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u043a\u0430\u043a \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u0442\u0441\u044f \u0431\u0430\u0437\u043e\u0432\u044b\u0439 EventHandler:<\/p>\n<pre><code class=\"cpp\">template&lt;typename ...Args&gt; class EventHandler { public: virtual void call(Args&amp;&amp;... args) = 0;  size_t GetHandlerID() const { return m_handlerID; }  void Invalidate() { m_valid = false; } bool IsValid() const { return m_valid; }  protected: size_t m_handlerID; private: bool m_valid = true; };<\/code><\/pre>\n<p>\u042d\u0442\u043e \u0431\u0430\u0437\u0430 \u0434\u043b\u044f \u0440\u0430\u0437\u043d\u044b\u0445 \u0432\u0438\u0434\u043e\u0432 \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u043e\u0432. \u0427\u0438\u0441\u0442\u0430\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u044f \u0432\u044b\u0437\u043e\u0432\u0430.<br \/> \u0422\u0443\u0442 \u0436\u0435 \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u043d\u0430 \u0438\u043d\u0432\u0430\u043b\u0438\u0434\u0430\u0446\u0438\u044f \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u0430 \u0434\u043b\u044f &#171;\u0433\u043e\u0440\u044f\u0447\u0435\u0433\u043e&#187; \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u044f.<\/p>\n<h4>MethodEventHandler<\/h4>\n<p>\u041d\u0430\u0441\u043b\u0435\u0434\u043d\u0438\u043a \u0434\u043b\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u0439-\u0447\u043b\u0435\u043d\u043e\u0432:<\/p>\n<pre><code class=\"cpp\">template&lt;typename Object, typename ...Args&gt; class MethodEventHandler final : public EventHandler&lt;Args...&gt; { public: using MethodType = void(Object::*)(Args...); public: MethodEventHandler(Object&amp; object, MethodType method) : m_object(object), m_method(method) { this-&gt;m_handlerID = detail::GenerateID(&amp;object, method); }  void call(Args&amp;&amp;... args) override { (m_object.*m_method)(std::forward&lt;Args&gt;(args)...); }  private: Object&amp; m_object; MethodType m_method; };<\/code><\/pre>\n<ul>\n<li>\n<p>\u0425\u0440\u0430\u043d\u0438\u0442 \u0441\u0441\u044b\u043b\u043a\u0443 \u043d\u0430 \u043e\u0431\u044a\u0435\u043a\u0442 \u0438 \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044c \u043d\u0430 \u0444\u0443\u043d\u043a\u0446\u0438\u044e.<\/p>\n<\/li>\n<li>\n<p>\u041f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0435\u0442 call, \u0442\u0430\u043a \u043a\u0430\u043a \u0441\u0438\u043d\u0442\u0430\u043a\u0441\u0438\u0441 \u0432\u044b\u0437\u043e\u0432\u0430 \u0441\u043f\u0435\u0446\u0438\u0444\u0438\u0447\u0435\u0441\u043a\u0438\u0439.<\/p>\n<\/li>\n<li>\n<p>\u0413\u0435\u043d\u0435\u0440\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u0430\u0439\u0434\u0438(\u0445\u0435\u0448) \u043d\u0430 \u0431\u0430\u0437\u0435 \u043e\u0431\u044a\u0435\u043a\u0442\u0430 \u0438 \u0444\u0443\u043d\u043a\u0446\u0438\u0438-\u0447\u043b\u0435\u043d\u0430.<\/p>\n<\/li>\n<\/ul>\n<h4>FunctionEventHandler<\/h4>\n<p>\u041d\u0430\u0441\u043b\u0435\u0434\u043d\u0438\u043a \u0434\u043b\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u0438:<\/p>\n<pre><code class=\"cpp\">template&lt;typename ...Args&gt; class FunctionEventHandler : public EventHandler&lt;Args...&gt; { public: using FunctionType = void(*)(Args...); public: FunctionEventHandler(FunctionType function) : m_function(function) { this-&gt;m_handlerID = detail::GenerateID(function); }  void call(Args&amp;&amp;... args) override { (*m_function)(std::forward&lt;Args&gt;(args)...); }  private: FunctionType m_function; };<\/code><\/pre>\n<p>\u0417\u0434\u0435\u0441\u044c \u2014 \u0442\u043e\u0442 \u0436\u0435 \u043f\u043e\u0434\u0445\u043e\u0434, \u0442\u043e\u043b\u044c\u043a\u043e \u0434\u043b\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u0439.<\/p>\n<h4>Event<\/h4>\n<p>\u0420\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f \u0441\u0430\u043c\u043e\u0433\u043e Event:<\/p>\n<pre><code class=\"cpp\">template&lt;typename ...Args&gt; class Event { using HandlerType = EventHandler&lt;Args...&gt;;  public: \/\/\u0417\u0430\u0449\u0438\u0442\u0438\u043c\u0441\u044f \u043e\u0442 \u043a\u043e\u043f\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u0438 \u043f\u0435\u0440\u0435\u043d\u043e\u0441\u0430 - \u044d\u0442\u043e \u043d\u0435\u0434\u043e\u043f\u0443\u0441\u0442\u0438\u043c\u0430\u044f \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u044f(\u0432\u043e \u0432\u0441\u044f\u043a\u043e\u043c \u0441\u043b\u0443\u0447\u0430\u0435 \u043f\u043e\u043a\u0430) Event() = default; Event(const Event&amp;) = delete; Event&amp; operator=(const Event&amp;) = delete; Event(Event&amp;&amp;) = delete; Event&amp; operator=(Event&amp;&amp;) = delete;  template&lt;typename Object&gt; void AddHandler(Object&amp; object, MethodEventHandler&lt;Object, Args...&gt;::MethodType method) { if (HasId(detail::GenerateID(&amp;object, method))) return;  (m_dispatching ? m_added_handlers : m_handlers).emplace_back(std::make_unique&lt;MethodEventHandler&lt;Object, Args...&gt;&gt;(object, method)); }  void AddHandler(FunctionEventHandler&lt;Args...&gt;::FunctionType function) { if (HasId(detail::GenerateID(function))) return;  (m_dispatching ? m_added_handlers : m_handlers).emplace_back(std::make_unique&lt;FunctionEventHandler&lt;Args...&gt;&gt;(function)); }  template&lt;class F&gt; void AddHandler(F) = delete; \/\/ \u041b\u044f\u043c\u0431\u0434\u044b \u0438 \u0444\u0443\u043d\u043a\u0442\u043e\u0440\u044b \u043d\u0435 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u044e\u0442\u0441\u044f \u2014 \u043d\u0435\u043b\u044c\u0437\u044f \u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e \u043e\u0442\u043f\u0438\u0441\u0430\u0442\u044c\u0441\u044f  template&lt;typename Object&gt; void RemoveHandler(Object&amp; object, MethodEventHandler&lt;Object, Args...&gt;::MethodType method) { RemoveById(detail::GenerateID(&amp;object, method)); }  void RemoveHandler(FunctionEventHandler&lt;Args...&gt;::FunctionType function) { RemoveById(detail::GenerateID(function)); }  template&lt;class F&gt; void RemoveHandler(F) = delete; \/\/\u0421\u0438\u043c\u043c\u0435\u0442\u0440\u0438\u044f  void operator()(Args... args) { m_dispatching = true; for (auto&amp; handler : m_handlers) if (handler-&gt;IsValid()) handler-&gt;call(std::forward&lt;Args&gt;(args)...); m_dispatching = false;  \/\/ \u0423\u0434\u0430\u043b\u044f\u0435\u043c \u043d\u0435\u0432\u0430\u043b\u0438\u0434\u043d\u044b\u0435 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0438 \u043f\u043e\u0441\u043b\u0435 \u0437\u0430\u0432\u0435\u0440\u0448\u0435\u043d\u0438\u044f \u0432\u044b\u0437\u043e\u0432\u043e\u0432         if (m_wasHotRemoved)         {     m_handlers.erase(std::remove_if(m_handlers.begin(), m_handlers.end(), [](const auto&amp; handler) { return !handler-&gt;IsValid(); }), m_handlers.end());             m_wasHotRemoved = false;         } for (auto&amp; handler : m_added_handlers) m_handlers.push_back(std::move(handler)); m_added_handlers.clear(); }  private: std::vector&lt;std::unique_ptr&lt;HandlerType&gt;&gt; m_handlers; std::vector&lt;std::unique_ptr&lt;HandlerType&gt;&gt; m_added_handlers;  bool m_dispatching = false;     bool m_wasHotRemoved = false;  bool HasId(std::size_t id) const { auto pred = [&amp;](const auto&amp; handler) { return handler-&gt;GetHandlerID() == id; }; return std::any_of(m_handlers.begin(), m_handlers.end(), pred) || std::any_of(m_added_handlers.begin(), m_added_handlers.end(), pred); }  void RemoveById(std::size_t id) { auto pred = [&amp;](const auto&amp; handler) { return handler-&gt;GetHandlerID() == id; };  if (m_dispatching) { if (auto it = std::find_if(m_handlers.begin(), m_handlers.end(), pred); it != m_handlers.end()) {                 (*it)-&gt;Invalidate();                 m_wasHotRemoved = true;             } } else { m_handlers.erase(std::remove_if(m_handlers.begin(), m_handlers.end(), pred), m_handlers.end()); }  m_added_handlers.erase(std::remove_if(m_added_handlers.begin(), m_added_handlers.end(), pred), m_added_handlers.end()); } };<\/code><\/pre>\n<ul>\n<li>\n<p>\u041f\u0440\u0435\u0434\u043b\u0430\u0433\u0430\u0435\u0442\u0441\u044f 2 \u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u044f \u0438 2 \u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u044f \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u043e\u0432 \u0432 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u043e\u0442 \u0442\u0438\u043f\u0430 \u043a\u043e\u043b\u0431\u0435\u043a\u0430.<\/p>\n<\/li>\n<li>\n<p>\u0417\u0430 \u0441\u0447\u0435\u0442 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0438 \u0438\u043d\u0432\u0430\u043b\u0438\u0434\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0445\u0435\u043d\u0434\u043b\u0435\u0440 \u043c\u043e\u0436\u043d\u043e \u0443\u0434\u0430\u043b\u044f\u0442\u044c \u0432\u043e  \u0432\u0440\u0435\u043c\u044f \u0434\u0438\u0441\u043f\u0430\u0442\u0447\u0438\u043d\u0433\u0430 \u0441\u043e\u0431\u044b\u0442\u0438\u044f.<\/p>\n<\/li>\n<li>\n<p>\u0420\u0430\u0431\u043e\u0442\u0430 \u0441 ID(\u0445\u0435\u0448\u043e\u043c) &#8212; \u0441\u043a\u0440\u044b\u0442\u0430\u044f \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f, \u043f\u043e \u044d\u0442\u043e\u043c\u0443 \u0432\u0441\u0435 \u0443\u0431\u0440\u0430\u043d\u043e \u0432 private \u0437\u043e\u043d\u0443 \u0438 \u0443\u043d\u0438\u0444\u0438\u0446\u0438\u0440\u043e\u0432\u0430\u043d\u043e \u0434\u043b\u044f \u0440\u0430\u0437\u043d\u044b\u0445 \u0442\u0438\u043f\u043e\u0432 \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u043e\u0432.<\/p>\n<\/li>\n<li>\n<p>\u0421\u043f\u0435\u0446\u0438\u0430\u043b\u044c\u043d\u044b\u0439 \u0444\u043b\u0430\u0436\u043e\u043a m_dispatching, \u0447\u0442\u043e\u0431\u044b \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u0442\u044c &#171;\u0433\u043e\u0440\u044f\u0447\u0430\u044f \u0444\u0430\u0437\u0430&#187; \u0438\u043b\u0438 \u043d\u0435\u0442. \u0415\u0441\u043b\u0438 \u0444\u0430\u0437\u0430 \u043d\u0435 \u0433\u043e\u0440\u044f\u0447\u0430\u044f, \u0442\u043e \u0438 \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0442\u044c\/\u0443\u0434\u0430\u043b\u044f\u0442\u044c \u043c\u043e\u0436\u043d\u043e \u043d\u0430\u043f\u0440\u044f\u043c\u0443\u044e.<\/p>\n<\/li>\n<\/ul>\n<h3>\u041d\u0435\u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0442\u043e\u043d\u043a\u043e\u0441\u0442\u0438 \u0433\u043e\u0440\u044f\u0447\u0435\u0439 \u0444\u0430\u0437\u044b<\/h3>\n<p>\u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u0435\/\u0423\u0434\u0430\u043b\u0435\u043d\u0438\u0435 \u0445\u0435\u043d\u0434\u043b\u0435\u0440\u043e\u0432 \u0432\u043e \u0432\u0440\u0435\u043c\u044f \u0434\u0438\u0441\u043f\u0430\u0442\u0447\u0438\u043d\u0433\u0430:<\/p>\n<ul>\n<li>\n<p>\u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u044b\u0439 \u0445\u0435\u043d\u0434\u043b\u0435\u0440 &#8212; \u043d\u0435 \u0431\u0443\u0434\u0435\u0442 \u0432\u044b\u0437\u0432\u0430\u043d \u0432 \u0442\u0435\u043a\u0443\u0449\u0435\u043c \u0434\u0438\u0441\u043f\u0430\u0442\u0447\u0438\u043d\u0433\u0435.<\/p>\n<\/li>\n<li>\n<p>\u0423\u0434\u0430\u043b\u0435\u043d\u043d\u044b\u0439 \u0445\u0435\u043d\u0434\u043b\u0435\u0440 &#8212; \u0431\u0443\u0434\u0435\u0442 \u043f\u043e\u043c\u0435\u0447\u0435\u043d \u043a\u0430\u043a \u043d\u0435\u0432\u0430\u043b\u0438\u0434\u043d\u044b\u0439 \u0438 \u043d\u0435 \u0431\u0443\u0434\u0435\u0442 \u0432\u044b\u0437\u0432\u0430\u043d \u0432 \u0442\u0435\u043a\u0443\u0449\u0435\u043c \u0434\u0438\u0441\u043f\u0430\u0442\u0447\u0438\u043d\u0433\u0435.<\/p>\n<\/li>\n<\/ul>\n<p><strong>\u0423\u0434\u0430\u043b\u0435\u043d\u0438\u0435 \u043c\u0433\u043d\u043e\u0432\u0435\u043d\u043d\u043e\u0435, \u0430 \u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u043e\u0442\u043b\u043e\u0436\u0435\u043d\u043d\u043e\u0435.<\/strong><\/p>\n<h3>\u041f\u0440\u043e\u0431\u043b\u0435\u043c\u044b \u0438 \u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u0438\u044f<\/h3>\n<ol>\n<li>\n<p>\u041c\u043e\u0436\u043d\u043e \u0437\u0430\u0431\u044b\u0442\u044c \u043e\u0442\u043f\u0438\u0441\u0430\u0442\u044c\u0441\u044f. \u0415\u0441\u043b\u0438 \u043e\u0431\u044a\u0435\u043a\u0442 \u0431\u0443\u0434\u0435\u0442 \u0443\u043d\u0438\u0447\u0442\u043e\u0436\u0435\u043d \u0438 \u043d\u0435 \u043e\u0442\u043f\u0438\u0448\u0435\u0442\u0441\u044f \u043e\u0442 \u0441\u043e\u0431\u044b\u0442\u0438\u044f, \u0432\u043e\u0437\u043d\u0438\u043a\u043d\u0435\u0442 UB \u0438\u0437-\u0437\u0430 \u0432\u0438\u0441\u044f\u0447\u0435\u0433\u043e \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044f.<\/p>\n<\/li>\n<li>\n<p>\u041b\u044f\u043c\u0431\u0434\u044b \u0437\u0430\u043f\u0440\u0435\u0449\u0435\u043d\u044b, \u043f\u043e\u0441\u043a\u043e\u043b\u044c\u043a\u0443 \u0438\u0445 \u043d\u0435\u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e \u043e\u0442\u043f\u0438\u0441\u0430\u0442\u044c.<\/p>\n<\/li>\n<li>\n<p>\u041d\u0435 thread-safe.<\/p>\n<\/li>\n<\/ol>\n<h3>\u0414\u043b\u044f \u0447\u0435\u0433\u043e \u044d\u0442\u0430 \u0441\u0438\u0441\u0442\u0435\u043c\u0430 \u043d\u0435 \u043f\u043e\u0434\u043e\u0439\u0434\u0435\u0442<\/h3>\n<ol>\n<li>\n<p>\u00ab\u042f \u0445\u043e\u0447\u0443 \u0441\u043b\u0443\u0448\u0430\u0442\u044c \u0441\u043e\u0431\u044b\u0442\u0438\u0435, \u043d\u043e\u00a0\u044f \u043d\u0435\u00a0\u0437\u043d\u0430\u044e \u043a\u0442\u043e \u0435\u0433\u043e \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u0442\u00bb.<\/p>\n<\/li>\n<li>\n<p>\u00ab\u042f \u043d\u0435\u00a0\u0437\u043d\u0430\u044e(\u0438\u043b\u0438 \u043d\u0435\u00a0\u0445\u043e\u0447\u0443 \u0437\u043d\u0430\u0442\u044c) \u0432\u0440\u0435\u043c\u044f \u0436\u0438\u0437\u043d\u0438 \u043e\u0431\u044a\u0435\u043a\u0442\u0430 \u0441\u00a0\u0441\u043e\u0431\u044b\u0442\u0438\u0435\u043c\u00bb.<\/p>\n<\/li>\n<\/ol>\n<h3>\u0414\u043b\u044f \u0447\u0435\u0433\u043e \u044d\u0442\u0430 \u0441\u0438\u0441\u0442\u0435\u043c\u0430 \u043f\u043e\u0434\u043e\u0439\u0434\u0435\u0442<\/h3>\n<ol>\n<li>\n<p>\u0412\u044b \u0438\u043c\u0435\u0435\u0442\u0435 \u0434\u043e\u0441\u0442\u0443\u043f \u043a\u00a0\u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0443 \u0441\u043e\u0431\u044b\u0442\u0438\u044f.<\/p>\n<\/li>\n<li>\n<p>\u0412\u044b \u0442\u043e\u0447\u043d\u043e \u0437\u043d\u0430\u0435\u0442\u0435 \u0432\u0440\u0435\u043c\u044f \u0436\u0438\u0437\u043d\u0438 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0430 \u0441\u043e\u0431\u044b\u0442\u0438\u044f(\u0438\u043b\u0438 \u043c\u043e\u0436\u0435\u0442\u0435 \u0443\u0431\u0435\u0434\u0438\u0442\u044c\u0441\u044f, \u0447\u0442\u043e\u00a0\u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a \u0436\u0438\u0432)<\/p>\n<\/li>\n<\/ol>\n<h3>\u0412\u043e\u0437\u043c\u043e\u0436\u043d\u044b\u0435 \u0443\u043b\u0443\u0447\u0448\u0435\u043d\u0438\u044f<\/h3>\n<ol>\n<li>\n<p>ScopedEventHandler\u00a0\u2014 RAII\u2011\u043c\u0435\u0445\u0430\u043d\u0438\u0437\u043c, \u043e\u0442\u043f\u0438\u0441\u043a\u0430 \u043f\u043e\u00a0\u0440\u0430\u0437\u0440\u0443\u0448\u0435\u043d\u0438\u044e. \u0420\u0435\u0448\u0438\u0442 \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u0443 \u0437\u0430\u0431\u044b\u0432\u0447\u0438\u0432\u043e\u0441\u0442\u0438, \u0430\u00a0\u0442\u0430\u043a\u00a0\u0436\u0435 \u0434\u0430\u0441\u0442 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u043f\u043e\u0434\u043f\u0438\u0441\u044b\u0432\u0430\u0442\u044c \u043b\u044f\u043c\u0431\u0434\u044b \u0438 \u0444\u0443\u043d\u043a\u0442\u043e\u0440\u044b.<\/p>\n<\/li>\n<li>\n<p>\u041e\u0442\u043a\u0430\u0437 \u043e\u0442\u00a0\u0432\u0438\u0440\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0445 \u0432\u044b\u0437\u043e\u0432\u043e\u0432 \u043c\u043e\u0436\u0435\u0442 \u0434\u0430\u0442\u044c \u043f\u0440\u0438\u0440\u043e\u0441\u0442 \u0437\u0430\u00a0\u0441\u0447\u0451\u0442 \u0443\u0441\u0442\u0440\u0430\u043d\u0435\u043d\u0438\u044f \u043e\u0431\u0440\u0430\u0449\u0435\u043d\u0438\u044f \u043a <code>vtable<\/code>.<\/p>\n<\/li>\n<\/ol>\n<h3>\u0417\u0430\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435<\/h3>\n<p>\u0420\u0435\u0448\u0435\u043d\u0438\u0435 \u0443\u0440\u043e\u0432\u043d\u044f \u00ab\u0434\u043e\u0431\u0430\u0432\u0438\u043b \u0438 \u043f\u043e\u0435\u0445\u0430\u043b\u043e\u00bb. \u0422\u0440\u0435\u0431\u0443\u0435\u0442 \u0432\u043d\u0438\u043c\u0430\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u0438, \u043f\u043e\u0441\u043a\u043e\u043b\u044c\u043a\u0443, \u0437\u0430\u0431\u044b\u0432 \u043e\u0442\u043f\u0438\u0441\u0430\u0442\u044c\u0441\u044f, \u043c\u043e\u0436\u043d\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c UB. \u041f\u043e\u0434\u0445\u043e\u0434\u0438\u0442 \u0434\u043b\u044f\u00a0\u043e\u0434\u043d\u043e\u043f\u043e\u0442\u043e\u0447\u043d\u043e\u0439 \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u044b, \u0433\u0434\u0435 \u044f\u0432\u043d\u043e \u0438\u0437\u0432\u0435\u0441\u0442\u043d\u044b \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438 \u0441\u043e\u0431\u044b\u0442\u0438\u0439 \u0438 \u0438\u0445 \u0436\u0438\u0437\u043d\u0435\u043d\u043d\u044b\u0439 \u0446\u0438\u043a\u043b.<\/p>\n<p>\u0415\u0441\u043b\u0438 \u0431\u0443\u0434\u0435\u0442 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e \u0432\u00a0\u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0435\u043c \u043f\u043e\u0441\u0442\u0435 \u043d\u0430\u043f\u0438\u0448\u0443:<\/p>\n<ol>\n<li>\n<p>ScopedEventHandler<\/p>\n<\/li>\n<li>\n<p>\u0420\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044e \u0431\u0435\u0437 virtual<\/p>\n<\/li>\n<\/ol>\n<p>\u041a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u0438 \u0438 \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u0438\u0432\u043d\u0430\u044f \u043a\u0440\u0438\u0442\u0438\u043a\u0430 \u043f\u0440\u0438\u0432\u0435\u0442\u0441\u0442\u0432\u0443\u0435\u0442\u0441\u044f. \u0411\u0443\u0434\u0435\u0442 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e \u043c\u043d\u0435\u043d\u0438\u0435 \u044d\u043a\u0441\u043f\u0435\u0440\u0442\u043e\u0432.<\/p>\n<p><a href=\"https:\/\/github.com\/2pizza\/cpp-event-system\" rel=\"noopener noreferrer nofollow\">GitHub<\/a><\/p>\n<\/div>\n<\/div>\n<\/div>\n<p><!----><!----><\/div>\n<p><!----><!----><br \/> \u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"https:\/\/habr.com\/ru\/articles\/924138\/\"> https:\/\/habr.com\/ru\/articles\/924138\/<\/a><br \/><\/br><\/br><\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[],"tags":[],"class_list":["post-465642","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/465642","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=465642"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/465642\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=465642"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=465642"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=465642"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}