thread/dispatcher.cc
A 
Glib::Dispatcher example.
#include <glibmm.h>
#include <algorithm>
#include <functional>
#include <iostream>
#include <vector>
namespace
{
class ThreadProgress
{
public:
  explicit ThreadProgress(int id);
  virtual ~ThreadProgress();
  int id() const;
  void launch();
  void join();
  bool unfinished() const;
  sigc::signal<void>& signal_finished();
private:
  enum { ITERATIONS = 100 };
  
  
  
  Glib::Thread* thread_;
  int id_;
  unsigned int progress_;
  Glib::Dispatcher signal_increment_;
  sigc::signal<void> signal_finished_;
  void progress_increment();
  void thread_function();
};
class Application : public sigc::trackable
{
public:
  Application();
  virtual ~Application();
  void run();
private:
  Glib::RefPtr<Glib::MainLoop> main_loop_;
  std::vector<ThreadProgress*> progress_threads_;
  void launch_threads();
  void on_progress_finished(ThreadProgress* thread_progress);
};
template <class T>
class DeletePtr : public std::unary_function<void, T>
{
public:
  void operator()(T ptr) const { delete ptr; }
};
ThreadProgress::ThreadProgress(int id)
:
  thread_ (0),
  id_ (id),
  progress_ (0)
{
  
  signal_increment_.connect(sigc::mem_fun(*this, &ThreadProgress::progress_increment));
}
ThreadProgress::~ThreadProgress()
{
  
  g_return_if_fail(thread_ == 0);
}
int ThreadProgress::id() const
{
  return id_;
}
void ThreadProgress::launch()
{
  
  thread_ = Glib::Thread::create(sigc::mem_fun(*this, &ThreadProgress::thread_function), true);
}
void ThreadProgress::join()
{
  thread_->join();
  thread_ = 0;
}
bool ThreadProgress::unfinished() const
{
  return (progress_ < ITERATIONS);
}
sigc::signal<void>& ThreadProgress::signal_finished()
{
  return signal_finished_;
}
void ThreadProgress::progress_increment()
{
  ++progress_;
  std::cout << "Thread " << id_ << ": " << progress_ << '%' << std::endl;
  if (progress_ >= ITERATIONS)
    signal_finished_();
}
void ThreadProgress::thread_function()
{
  Glib::Rand rand;
  for (int i = 0; i < ITERATIONS; ++i)
  {
    Glib::usleep(rand.get_int_range(2000, 20000));
    
    signal_increment_();
  }
}
Application::Application()
:
  main_loop_ (Glib::MainLoop::create()),
  progress_threads_ (5)
{
  
  
  
#ifdef GLIBMM_EXCEPTIONS_ENABLED
 try
#endif
 {
    for (std::vector<ThreadProgress*>::size_type i = 0; i < progress_threads_.size(); ++i)
    {
      ThreadProgress *const progress = new ThreadProgress(i + 1);
      progress_threads_[i] = progress;
      progress->signal_finished().connect(
          sigc::bind<1>(sigc::mem_fun(*this, &Application::on_progress_finished), progress));
    }
  }
#ifdef GLIBMM_EXCEPTIONS_ENABLED
 catch (...)
  {
    
    
    std::for_each(progress_threads_.begin(), progress_threads_.end(),
                  DeletePtr<ThreadProgress*>());
    throw;
  }
#endif
}
Application::~Application()
{
  std::for_each(progress_threads_.begin(), progress_threads_.end(),
                DeletePtr<ThreadProgress*>());
}
void Application::run()
{
  
  Glib::signal_idle().connect(
      sigc::bind_return(sigc::mem_fun(*this, &Application::launch_threads), false));
  main_loop_->run();
}
void Application::launch_threads()
{
  std::cout << "Launching " << progress_threads_.size() << " threads:" << std::endl;
  std::for_each(progress_threads_.begin(), progress_threads_.end(),
                std::mem_fun(&ThreadProgress::launch));
}
void Application::on_progress_finished(ThreadProgress* thread_progress)
{
  thread_progress->join();
  std::cout << "Thread " << thread_progress->id() << ": finished." << std::endl;
  
  if (std::find_if(progress_threads_.begin(), progress_threads_.end(),
                   std::mem_fun(&ThreadProgress::unfinished)) == progress_threads_.end())
  {
    main_loop_->quit();
  }
}
} 
int main(int, char**)
{
  Glib::thread_init();
  Application application;
  application.run();
  return 0;
}