С Новым годом! Форум программистов, компьютерный форум, киберфорум
Boost C++
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.78/18: Рейтинг темы: голосов - 18, средняя оценка - 4.78
 Аватар для DiffEreD
1458 / 795 / 257
Регистрация: 21.06.2011
Сообщений: 1,740
Записей в блоге: 2

boost::asio::io_service::stop()

09.01.2014, 22:12. Показов 3774. Ответов 6
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Читаю тут одну книгу и в ней есть такой пример по использованию boost::asio::io_service примерно вот такого содержимого:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
#include <iostream>
#include <memory>
#include <boost/asio/io_service.hpp>
 
boost::asio::io_service io;
 
template <typename F>
void push_task(const F& task) {
   io.post(task);
}
 
int g_val = 0;
 
void foo()
{
   ++g_val;
   if (g_val == 90)
      io.stop();
}
 
int main()
{
   std::unique_ptr<boost::asio::io_service::work> pwork(new boost::asio::io_service::work(io));
 
   static const std::size_t tasks_count = 100;
   for (std::size_t i =0; i < tasks_count; ++i) {
      push_task(&foo);
   }
 
   io.run();
   
   std::cout << "g_val = " << g_val << "\n"; // должно было бы быть 90, а равно 100
 
   return 0;
}
У меня почему то в ф-ции foo() не срабатывает io.stop(). Полный пример (тот что в книге) ведет себя аналогично. В чем тут может быть проблема?
P.S. Я только начал вникать в boost::asio.
0
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
09.01.2014, 22:12
Ответы с готовыми решениями:

Хелп! Резолвинг и деструктор boost::asio::io_service
Добрый день! Boost::asio начал использовать не так давно. Столкнулся с проблемой. Вернее проблемы две. Написал я простой...

Boost.asio io_service.run не дожидается выполнения асинхронного чтения
Добрый день! Имеется такая задача: - нужно отправялться HTTP запрос и получать ответ. все операции асинхронные (читаю через...

boost::asio
начал ковырять буст. Возник вопрос с созданием клинт-серверного приложения. Для ознакомления хочу посмотреть простой пример, где сервер...

6
Ушел с форума
Эксперт С++
 Аватар для Убежденный
16481 / 7444 / 1187
Регистрация: 02.05.2013
Сообщений: 11,616
Записей в блоге: 1
09.01.2014, 22:41
Мне кажется, все работает именно так, как должно. Вы поставили в очередь
100 операций, затем из одного обработчика вызываете stop. Этот stop, видимо,
тоже ставится в очередь, и пока до его обработки доходит дело, все операции
успевают завершиться. Вот если здесь создать несколько рабочих потоков,
которые крутились бы в io_service::run, то картина могла быть иной.

Не забываем, что на Windows (кстати, ОС какая ?) Boost.Asio использует порты
завершения ввода-вывода, а сигнал ставится в очередь порта с помощью
функции PostQueuedCompletionStatus и, надо полагать, обрабатывается с
таким же приоритетом, как и остальные I/O-операции. Т.е. последовательно.

Кстати, в самом Boost.Asio по поводу io_service::stop пишут:
This function does not block, but instead simply signals the io_service to stop.
All invocations of its run() or run_one() member functions should return as soon as possible.
"As soon as possible" - т.е. поток выйдет из io_service::run как только сможет.
Необязательно сразу.
1
 Аватар для DiffEreD
1458 / 795 / 257
Регистрация: 21.06.2011
Сообщений: 1,740
Записей в блоге: 2
10.01.2014, 00:33  [ТС]
Я тогда просто недоумеваю, зачем этот пример тогда в книге дается? На всякий случай вот полный пример:
tasks_processor_base.hpp
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
#ifndef BOOK_CHAPTER6_TASK_PROCESSOR_BASE_HPP
#define BOOK_CHAPTER6_TASK_PROCESSOR_BASE_HPP
 
#if defined(_MSC_VER) && (_MSC_VER >= 1200)
# pragma once
#endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
 
#include <boost/thread/thread.hpp>
#include <boost/asio/io_service.hpp>
 
namespace detail {
 
template <class T>
struct task_wrapped {
private:
    T task_unwrapped_;
 
public:
    explicit task_wrapped(const T& task_unwrapped)
        : task_unwrapped_(task_unwrapped)
    {}
 
    void operator()() const {
        // resetting interruption
        try {
            boost::this_thread::interruption_point();
        } catch(const boost::thread_interrupted&){}
 
        try {
            // Executing task
            task_unwrapped_();
        } catch (const std::exception& e) {
            std::cerr<< "Exception: " << e.what() << '\n';
        } catch (const boost::thread_interrupted&) {
            std::cerr<< "Thread interrupted\n";
        } catch (...) {
            std::cerr<< "Unknown exception\n";
        }
    }
};
 
template <class T>
inline task_wrapped<T> make_task_wrapped(const T& task_unwrapped) {
    return task_wrapped<T>(task_unwrapped);
}
 
} // namespace detail
 
 
namespace tp_base {
 
class tasks_processor: private boost::noncopyable {
protected:
    boost::asio::io_service         ios_;
    boost::asio::io_service::work   work_;
 
    tasks_processor()
        : ios_()
        , work_(ios_)
    {}
 
public:
    static tasks_processor& get();
 
    template <class T>
    inline void push_task(const T& task_unwrapped) {
        ios_.post(detail::make_task_wrapped(task_unwrapped));
    }
 
    void start() {
        ios_.run();
    }
 
    void stop() {
        ios_.stop();
    }
}; // tasks_processor
 
} // namespace base::
 
#endif // BOOK_CHAPTER6_TASK_PROCESSOR_BASE_HPP
main.cpp
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
// Amost all the code for this example is in this header
#include "tasks_processor_base.hpp"
using namespace tp_base;
 
// Part of tasks_processor class from
// tasks_processor_base.hpp, that must be defined
// Somewhere in source file
tasks_processor& tasks_processor::get() {
    static tasks_processor proc;
    return proc;
}
 
int g_val = 0;
void func_test() {
    ++ g_val;
    if (g_val == 3) {
        throw std::logic_error("Just checking");
    }
    
    boost::this_thread::interruption_point();
    if (g_val == 10) {
        // Emulation of thread interruption.
        // Will be catched and won't stop execution.
        throw boost::thread_interrupted();
    }
 
    if (g_val == 90) {
        tasks_processor::get().stop();
    }
}
 
int main () {
    static const std::size_t tasks_count = 100;
    // stop() is called at 90
    BOOST_STATIC_ASSERT(tasks_count > 90);
 
    for (std::size_t i =0; i < tasks_count; ++i) {
        tasks_processor::get().push_task(&func_test);
    }
 
    // We can also use result of boost::bind call
    // as a task.
    tasks_processor::get().push_task(
        boost::bind(std::plus<int>(), 2, 2) // counting 2 + 2
    );
 
    // Processing was not started.
    assert(g_val == 0);
 
    // Will not throw, but blocks till
    // one of the tasks it is owning
    // calls stop().
    tasks_processor::get().start();
    assert(g_val == 90);
}
В конце автор дает понять что g_val должно быть равно 90. Возможно, это его ошибка, но тем не менее, раз уж я тему создал, то как тогда правильно данной функцией пользоваться? Как я понял нет никакой гарантии что она сработает.
P.S. Если это может зависить от ОС, то у меня Windows 8.1 64bit
0
Ушел с форума
Эксперт С++
 Аватар для Убежденный
16481 / 7444 / 1187
Регистрация: 02.05.2013
Сообщений: 11,616
Записей в блоге: 1
10.01.2014, 01:08
Boost.Asio - это кросс-платформенная библиотека, и она в своих "потрохах"
использует разные платформо-зависимые вещи. На Windows используются порты
завершения ввода-вывода (I/O Completion Ports, IOCP), на других системах -
epoll/select и т.д. В зависимости от выбранного метода поведение может
немного варьироваться. Думаю, данный случай - как раз из таких.
Автор, скорее всего, не проверял свой код на Windows. Кстати, что за книга ?

Если Вы почитаете в MSDN про IOCP, в особенности о работе таких функций,
как GetQueuedCompletionStatus и PostQueuedCompletionStatus, то станет ясно,
почему io_service::stop не останавливает работу io_service::run немедленно.
Сигнал ставится в конец очереди и рабочий поток выгребает его оттуда только
после того, как обработает все I/O-операции, стоящие в очереди перед ним.

Кстати, в Boost можно отключить IOCP, тогда библиотека будет использовать
менее эффективную модель select. Я ради интереса поставил в самом начале
заголовка tasks_processor_base.hpp такую строчку:
C++
1
#define BOOST_ASIO_DISABLE_IOCP 1
И после этого ассерт перестал срабатывать...

Цитата Сообщение от DiffEreD Посмотреть сообщение
Как я понял нет никакой гарантии что она сработает.
Она сработает, просто не гарантируется, что это произойдет сразу.

Добавлено через 7 минут
И еще хочу добавить, что данный пример несколько надуман.
В реальных условиях такое бывает очень редко, чтобы io_service сначала
нагрузили большим количеством операций, а затем их бы "разматывал"
один поток. Как правило, потоков несколько (thread pool), а I/O-операции
поступают и обрабатываются постепенно: запустили на сокете операцию
чтения, получили сигнал завершения, запустили запись, снова ждем
завершения, затем снова чтение и т.д. В этом случае очередь io_service
поддерживается в состоянии, близком к пустому, поэтому если кто-то
вызывает io_service::stop, остановка происходит практически без задержки.
1
 Аватар для DiffEreD
1458 / 795 / 257
Регистрация: 21.06.2011
Сообщений: 1,740
Записей в блоге: 2
10.01.2014, 01:17  [ТС]
Да, этот макрос помог.
А книга "Antony Polukhin - Boost C++ Application Development Cookbook".
0
503 / 352 / 94
Регистрация: 22.03.2011
Сообщений: 1,112
10.01.2014, 19:25
Цитата Сообщение от Убежденный Посмотреть сообщение
И еще хочу добавить, что данный пример несколько надуман.
В реальных условиях такое бывает очень редко, чтобы io_service сначала
нагрузили большим количеством операций, а затем их бы "разматывал"
один поток. Как правило, потоков несколько (thread pool), а I/O-операции
поступают и обрабатываются постепенно: запустили на сокете операцию
чтения, получили сигнал завершения, запустили запись, снова ждем
завершения, затем снова чтение и т.д. В этом случае очередь io_service
поддерживается в состоянии, близком к пустому, поэтому если кто-то
вызывает io_service::stop, остановка происходит практически без задержки.
Нет. Если в очереди не будет задач произойдет выход.

Цитата Сообщение от Убежденный Посмотреть сообщение
Сигнал ставится в конец очереди и рабочий поток выгребает его оттуда только
после того, как обработает все I/O-операции, стоящие в очереди перед ним.
Вот это верно.
0
Ушел с форума
Эксперт С++
 Аватар для Убежденный
16481 / 7444 / 1187
Регистрация: 02.05.2013
Сообщений: 11,616
Записей в блоге: 1
10.01.2014, 19:40
Цитата Сообщение от stima Посмотреть сообщение
Если в очереди не будет задач произойдет выход.
Так я вроде обратного не утверждал.
А так да, очередь пуста - io_service::run возвращает управление.
Если только не используется какой-нибудь io_service::work.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
10.01.2014, 19:40
Помогаю со студенческими работами здесь

Boost Asio C++
Добрый день. Юзаю этот ( http://www.boost.org/ ) проект. Нужна помощь с документаций...

boost/asio
Кто тестил asio в Вuilder-е? Код использовал отсюда https://www.cyberforum.ru/cpp-beginners/thread329576.html И напоролся сразу...

Boost.asio
Добрый день. Я только начал знакомство с boost.asio и при использовании примеров с офф.сайта никак не могу разобраться в их работе. Вот...

boost::asio пример
Привет! Можно, пожалуйста, простейший пример работы с boost::asio - клиент-серверная программа, клиент которой должен отсылать...

boost::asio cессии
Здравствуйте. Хотелось бы узнать ответ на вопрос, который не где не нашол. Сервер реализован на Boost:asio, к нему подключаются клиенты,...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
7
Ответ Создать тему
Новые блоги и статьи
Советы по крайней бережливости. Внимание, это ОЧЕНЬ длинный пост.
Programma_Boinc 28.12.2025
Советы по крайней бережливости. Внимание, это ОЧЕНЬ длинный пост. Налог на собак: https:/ / **********/ gallery/ V06K53e Финансовый отчет в Excel: https:/ / **********/ gallery/ bKBkQFf Пост отсюда. . .
Кто-нибудь знает, где можно бесплатно получить настольный компьютер или ноутбук? США.
Programma_Boinc 26.12.2025
Нашел на реддите интересную статью под названием Anyone know where to get a free Desktop or Laptop? Ниже её машинный перевод. После долгих разбирательств я наконец-то вернула себе. . .
Thinkpad X220 Tablet — это лучший бюджетный ноутбук для учёбы, точка.
Programma_Boinc 23.12.2025
Рецензия / Мнение/ Перевод Нашел на реддите интересную статью под названием The Thinkpad X220 Tablet is the best budget school laptop period . Ниже её машинный перевод. Thinkpad X220 Tablet —. . .
PhpStorm 2025.3: WSL Terminal всегда стартует в ~
and_y87 14.12.2025
PhpStorm 2025. 3: WSL Terminal всегда стартует в ~ (home), игнорируя директорию проекта Симптом: После обновления до PhpStorm 2025. 3 встроенный терминал WSL открывается в домашней директории. . .
Как объединить две одинаковые БД Access с разными данными
VikBal 11.12.2025
Помогите пожалуйста !! Как объединить 2 одинаковые БД Access с разными данными.
Новый ноутбук
volvo 07.12.2025
Всем привет. По скидке в "черную пятницу" взял себе новый ноутбук Lenovo ThinkBook 16 G7 на Амазоне: Ryzen 5 7533HS 64 Gb DDR5 1Tb NVMe 16" Full HD Display Win11 Pro
Музыка, написанная Искусственным Интеллектом
volvo 04.12.2025
Всем привет. Некоторое время назад меня заинтересовало, что уже умеет ИИ в плане написания музыки для песен, и, собственно, исполнения этих самых песен. Стихов у нас много, уже вышли 4 книги, еще 3. . .
От async/await к виртуальным потокам в Python
IndentationError 23.11.2025
Армин Ронахер поставил под сомнение async/ await. Создатель Flask заявляет: цветные функции - провал, виртуальные потоки - решение. Не threading-динозавры, а новое поколение лёгких потоков. Откат?. . .
Поиск "дружественных имён" СОМ портов
Argus19 22.11.2025
Поиск "дружественных имён" СОМ портов На странице: https:/ / norseev. ru/ 2018/ 01/ 04/ comportlist_windows/ нашёл схожую тему. Там приведён код на С++, который показывает только имена СОМ портов, типа,. . .
Сколько Государство потратило денег на меня, обеспечивая инсулином.
Programma_Boinc 20.11.2025
Сколько Государство потратило денег на меня, обеспечивая инсулином. Вот решила сделать интересный приблизительный подсчет, сколько государство потратило на меня денег на покупку инсулинов. . . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru