Многопоточное программирование одна из самых сложных тем в программировании, особенно в C++. Трудно избежать при этом ошибок. К счастью большую часть удаётся отловить на этапе проверки кода или тестирования. Но особо коварные проникают в рабочие системы и исправлять их достаточно затруднительно.
В этой статье собраны и переведены самые значимые по мнению автора заметки ошибочные ситуации. Если у вас есть свои любимые ошибки или варианты их решения, оставьте, пожалуйста, их в комментариях.
Все примеры успешно компилируются и исполняются в Ubuntu 16.04 LTS:
g++ -std=c++14 -O2 -Wall -pedantic -pthread main.cpp && ./a.out
#1 Отсутствие join() или detach() перед завершением
Если забыть вызвать join() или detach() перед завершением программы, это может привести к её аварийному завершению.
#include <iostream>
#include <thread>
void foo()
{
std::cout << "foo" << std::endl;
}
int main(int argc, char *argv[])
{
std::thread t(foo);
return 0;
}
В конце функции main() объект t выходит из области видимости и вызывается деструктор. Внутри деструктора выполняется проверка на подключаемость потока. Подключаемый поток - это поток который может или уже выполняется. В данном случае это именно так поэтому будет вызвана функция std::terminate().
В зависимости желаемого поведения следует либо подождать завершения потока:
int main(int argc, char *argv[])
{
std::thread t(foo);
t.join();
return 0;
}
либо разорвать с ним связь
int main(int argc, char *argv[])
{
std::thread t(foo);
t.detach();
return 0;
}
#2 Попытка дождаться завершения неподключаемого потока
Для объектов std::thread которые были перемещены, завершены join() или брошены detach() нельзя дождаться завершения.
#include <iostream>
#include <thread>
void foo()
{
std::cout << "foo" << std::endl;
}
int main(int argc, char *argv[])
{
std::thread t(foo);
t.detach();
// ... какая-то логика ...
t.join();
return 0;
}
В таких случаях следует проверять, а можно ли в принципе подключить поток, и только потом уже вызывать join().
if (t.joinable())
{
t.join();
}
#3 Вызов join() блокирует вызывающий поток
В реальных приложениях потоки могут обслуживать достаточно длительные операции связанные с сетевым вводом/выводом или реакцией пользователя в пользовательском интерфейсе. Попытка дождаться завершения таких потоков в основном потоке или в потоке отвечающим за интерфейс может привести к "заморозке". Лучше найти другое решение.
Например, для десктопного приложения вспомогательный поток перед своим завершением может послать сообщение в поток отвечающий за интерфейс. Последний как правило организован как обработчик очереди сообщений. Чаще всего это сообщения от элементов интерфейса, но могут быть и нажатия клавиш и даже перемещения мыши. В этом потоке точно так же можно получить сообщение от вспомогательного потока и среагировать на завершение, не дожидаясь его буквально, и не блокируя ожиданием обработку событий.
#4 Не учитывать особенности передачи аргументов в поток
Аргументы в функцию потока перемещаются или копируются по значению. Пример ниже даже не скомпилируется.
#include <iostream>
#include <thread>
void foo(int &s)
{
s = 42;
}
int main(int argc, char *argv[])
{
int answer = 0;
std::thread t(foo, answer);
t.join();
return 0;
}
Для успешной компиляции ссылка должна быть передана через std::ref
std::thread t(foo, std::ref(answer));
#5 Игнорировать общий доступ к ресурсам
В условиях многопоточности несколько потоков могут одновременно использовать общие ресурсы или данные. Это может приводить не только к труднопрогнозируемому поведению но и к краху программы. В таких случаях необходимо упорядочивать и контролировать доступ.
В качестве примера рассмотрим вывод в консоль в несколько потоков - основной и шесть дополнительных.
#include <iostream>
#include <thread>
void foo(const std::string &message)
{
std::cout << "thread " << std::this_thread::get_id() << ", message " << message << std::endl;
}
int main(int argc, char *argv[])
{
std::thread t1(foo, "каждый");
std::thread t2(foo, "охотник");
std::thread t3(foo, "желает");
foo("знать");
std::thread t4(foo, "где");
std::thread t5(foo, "сидит");
std::thread t6(foo, "фазан");
t1.join();
t2.join();
t3.join();
t4.join();
t5.join();
t6.join();
return 0;
}
В результате получим мешанину из слов:
thread thread thread 140597982512960, message знать140597965162240
thread 140597939984128, message где
thread 140597931591424, message 140597956769536, message thread каждый140597948376832, message желает, message
сидит
охотник
thread 140597923198720, message фазан
Дело в том, что консоль одна, а семь потоков пытаются выводить на неё одновременно. Чтобы сделать вывод более предсказуемым необходимо ограничить одновременный доступ. Сделаем это через std::mutex - заблокируем до вывода и освободим после.
#include <mutex>
std::mutex cout_guard;
void foo(const std::string &message)
{
cout_guard.lock();
std::cout << "thread " << std::this_thread::get_id() << ", message " << message << std::endl;
cout_guard.unlock();
}
Вывод получится читаемый:
thread 140710040659776, message знать
thread 140710023309056, message каждый
thread 140710006523648, message желает
thread 140709989738240, message сидит
thread 140709981345536, message фазан
thread 140710014916352, message охотник
thread 140709998130944, message где
Мешанины из фрагментов строк уже нет. Порядок захвата нами не управляется, поэтому сами сообщения появляются в произвольном порядке.
#6 Забыть вызвать unlock()
В предыдущем примере для разделения доступа к ресурсу использовался std::mutex. Это не самый удачный способ, поскольку вызова unlock() мы можем и не достичь, если вообще не забыли его вызвать.
void foo(const std::string &message)
{
cout_guard.lock();
std::cout << "thread " << std::this_thread::get_id() << ", message " << message << std::endl;
// cout_guard.unlock();
}
После появления первого же сообщения программа зависнет:
thread 140569688782656, message знать
Чтобы защититься от ошибок такого рода воспользуемся std::lock_guard, который манипулирует временем жизни блокировки в стиле RAII.
В конструкторе захватывает, в деструкторе освобождает. По какой бы причине мы не покинули область видимости - блокировка будет снята.
void foo(const std::string &message)
{
std::lock_guard<std::mutex> lock(cout_guard);
std::cout << "thread " << std::this_thread::get_id() << ", message " << message << std::endl;
}
#7 Пренебрежение размером защищённой секции
Пока мы находимся внутри защищённой секции все остальные потоки рвущиеся её выполнить заблокированы. Старайтесь делать заблокированный участок как можно меньше.
#include <chrono>
#include <iostream>
#include <mutex>
#include <thread>
using namespace std::chrono_literals;
std::mutex cout_mutex;
void foo()
{
std::lock_guard<std::mutex> lock(cout_mutex);
std::this_thread::sleep_for(1s); // на самом деле что-то очень полезное и безопасное
std::cout << "foo" << std::endl;
}
int main(int argc, char *argv[])
{
std::thread t1(foo);
std::thread t2(foo);
t1.join();
t2.join();
return 0;
}
Безопасный вариант программы будет исполняться почти две секунды.
Если нам необходимо защитить только вывод в консоль, то нет необходимости в секции такого размера. Мы можем переместить блокировки ближе к выводу.
void foo()
{
std::this_thread::sleep_for(1s); // на самом деле что-то очень полезное и безопасное
std::lock_guard<std::mutex> lock(cout_mutex);
std::cout << "foo" << std::endl;
}
Такой вариант будет выполняться уже около одной секунды и при этом останется безопасным.
#8 Взаимные блокировки
Как правило такие блокировки уже навсегда из-за чего получили название deadlock. Типичная ситуация такой блокировки представлена ниже. Функция sleep_for даёт нам 100% шанс попасть в вечную блокировку, без неё скорее всего на любой машине этот код выполнился бы без зависания.
#include <chrono>
#include <iostream>
#include <mutex>
#include <thread>
using namespace std::chrono_literals;
std::mutex cerr_mutex;
std::mutex cout_mutex;
void foo()
{
cerr_mutex.lock();
std::cerr << "use cerr in foo" << std::endl;
std::this_thread::sleep_for(1s);
cout_mutex.lock();
std::cout << "use cout in foo" << std::endl;
cout_mutex.unlock();
cerr_mutex.unlock();
}
void bar()
{
cout_mutex.lock();
std::cout << "use cout in bar" << std::endl;
std::this_thread::sleep_for(1s);
cerr_mutex.lock();
std::cerr << "use cerr in bar" << std::endl;
cerr_mutex.unlock();
cout_mutex.unlock();
}
int main(int argc, char *argv[])
{
std::thread t1(foo);
std::thread t2(bar);
t1.join();
t2.join();
return 0;
}
Причина зависания кроется в перекрёстной блокировке. Когда оба потока начинают работать каждый из них блокирует свой mutex. То есть t1 захватил cerr_mutex, а t2 - cout_mutex. После вызова sleep_for потоки пытаются захватить их наоборот, еще не освободив занятые. Для того чтобы освободить свой mutex потоку приходится ждать пока это сделает второй, а у второго ситуация ровно такая же.
Самое простое решение использовать std::lock для захвата обоих mutex.
void foo()
{
std::lock(cerr_mutex, cout_mutex);
std::cerr << "use cerr in foo" << std::endl;
std::this_thread::sleep_for(1s);
std::cout << "use cout in foo" << std::endl;
cout_mutex.unlock();
cerr_mutex.unlock();
}
Если условия позволяют, можно использовать std::timed_mutex. Для выхода из перекрёстной блокировки достаточно, чтобы одну из них можно было нарушить.
void foo()
{
cerr_mutex.lock();
std::cerr << "use cerr in foo" << std::endl;
std::this_thread::sleep_for(1s);
if (cout_mutex.try_lock_for(1s))
{
std::cout << "use cout in foo" << std::endl;
cout_mutex.unlock();
}
cerr_mutex.unlock();
}
С одной стороны мы успешно избежали блокировки, но с другой стороны нам пришлось взять на себя обработку этой ситуации.
#9 Повторный захват std::mutex
Эта некоторая вариация на тему перекрёстного захвата с тем лишь отличием, что для такой блокировки достаточно одного std::mutex. Даже дополнительные потоки не нужны. Тут можно было бы привести в качестве примера страшную банальщину типа:
void foo()
{
std::lock_guard<std::mutex> lock1(cerr_mutex);
std::lock_guard<std::mutex> lock2(cerr_mutex);
std::cerr << "foo" << std::endl;
}
Да, это именно такого рода ошибка, вот только в жизни она встречается в несколько более изощрённой форме. В примере ниже нет ни потоков, ни рекурсии и всего один mutex.
#include <iostream>
#include <mutex>
std::mutex cerr_mutex;
int bar()
{
std::lock_guard<std::mutex> lock(cerr_mutex);
std::cerr << "bar" << std::endl;
return 42;
}
void foo()
{
std::lock_guard<std::mutex> lock(cerr_mutex);
std::cerr << "foo, bar = " << bar() << std::endl;
}
int main(int argc, char *argv[])
{
foo();
return 0;
}
Вызов foo приводит к захвату mutex, но в процессе вызова bar возникает необходимость блокировки ресурса повторно. Это такой же deadlock, только теперь основной поток ждёт сам себя.
Существует очевидный способ решить проблему - заменить обычный std::mutex на рекурсивный std::recursive_mutex и это решит нашу проблему, но решит её ой каким опасным способом. Тысячу раз подумайте всё ли будет в порядке при таком подходе, не удастся ли найти решение более элегантное
void foo()
{
auto b = bar();
std::lock_guard<std::mutex> lock(cerr_mutex);
std::cerr << "foo, bar = " << b << std::endl;
}
#10 Излишняя предосторожность
Когда возникает необходимость модифицировать простые типы наподобие bool или int использование 'std::atomic' почти всегда более эффективно в сравнении с использованием mutex.
#include <mutex>
std::mutex counter_mutex;
int counter;
void foo()
{
std::lock_guard<std::mutex> lock(counter_mutex);
++counter;
}
Та же самая логика без использования mutex и использованием atomic.
#include <atomic>
std::atomic<int> counter;
void foo()
{
++counter;
}
#11 Частое создание потоков без использования пулов
Создание и удаление потоков дорогое удовольствия с точки зрения затрат CPU. Часто создавать и удалять потоки в приложении, которое само по активно занимается вычислениями значит мешать ему. Вместо того, чтобы часто создавать и удалять потоки лучше создать пул предварительно запущенных потоков и распределять между ними задания.
Использование пула потоков позволяет сократить объём кода и количество потенциальных ошибок связанных с запуском и остановкой потоков. Позволит избежать избыточного количества потоков, которое может негативно сказаться на производительности.
Существуют готовые реализации такого рода пулов. Например, TBB
#12 Не обработанные исключения в потоке
Исключения брошенные в одном потоке не могут быть перехвачены другим. Представим себе функцию, которая может бросить исключение. Если мы выполним эту функцию в отдельном потоке, то попытка поймать исключение в основном не сработает.
#include <iostream>
#include <stdexcept>
#include <thread>
void foo()
{
throw std::runtime_error("foo");
}
int main(int argc, char *argv[])
{
try
{
std::thread t(foo);
t.join();
}
catch (const std::exception &e)
{
std::cout << "error" << e.what() << std::endl;
}
return 0;
}
Программа аварийно завершится так и не вызвав обработчик исключения. Решением может быть перехват исключения в потоке и передача информации о нём через экземпляр std::exception_ptr в родительский поток и повторного бросания уже за пределами породившего исключения потока.
#include <iostream>
#include <stdexcept>
#include <thread>
static std::exception_ptr eptr = nullptr;
void foo()
{
try
{
throw std::runtime_error("foo");
}
catch (...)
{
eptr = std::current_exception();
}
}
int main(int argc, char *argv[])
{
std::thread t(foo);
t.join();
if (eptr)
{
try
{
std::rethrow_exception(eptr);
}
catch (const std::exception &e)
{
std::cout << "error" << e.what() << std::endl;
}
}
return 0;
}
#13 Имитация асинхронной работы без std::async
Когда нужно выполнить часть кода независимо от основного потока отличным выбором будет использование std::async для запуска. Это тоже самое, что создать ещё один поток и передать ему на выполнение функцию или лямбду. Правда при этом за жизненным циклом потока и исключений в нём тоже придётся следить самостоятельно. В случае использования std::async можно не заботиться об этом да ещё и существенно сократить вероятность блокировки.
Еще одно важно преимущество заключается в возможности получить результат работы функции через std::future. Функция int foo(), будучи выполнена как асинхронная задача, заранее установит результат своей работы. А получим мы его тогда, когда нам это будет удобно.
#include <iostream>
#include <cmath>
#include <future>
int main(int argc, char *argv[])
{
auto f = std::async(sqrt, 9.0);
std::cout << f.get() << std::endl;
return 0;
}
Использование потоков напрямую делает получение результатов чуть более громоздким. Это может быть:
Передача ссылки на результирующую переменную в поток, в котором необходимо сохранить результат.
#include <iostream>
#include <cmath>
#include <thread>
void foo(double i, double &r)
{
r = std::sqrt(i);
}
int main(int argc, char *argv[])
{
double result = 0.0;
auto t = std::thread(foo, 9, std::ref(result));
t.join();
std::cout << result << std::endl;
return 0;
}
Сохранение результата в переменной класса функционального объекта и чтение его после завершения работы.
#include <iostream>
#include <cmath>
#include <thread>
template<typename T>
class foo
{
T r;
public:
T get()
{
return r;
}
void operator()(T i)
{
r = std::sqrt(i);
}
};
int main(int argc, char *argv[])
{
auto f = foo<double>();
auto t = std::thread(std::ref(f), 9);
t.join();
std::cout << f.get() << std::endl;
return 0;
}
Курт Гантерот в своей книге утверждает, что создание потоков в 14 раз дороже использования std::async.
Короче говоря, пока не доказано обратное использовать следует std::async.
#14 Опускание std::launch::async когда это действительно необходимо
Название std::async может ввести в заблуждение, потому что функция, которая будет передана для запуска по умолчанию может и не запуститься отдельно от вызываемого потока!
Существует два способа запуска:
- std::launch::async. Задача будет немедленно запущена в отдельном потоке.
- std::launch::deferred. Выполнение задачи будет отложено до вызова .get() или .wait() возвращаемого объекта std::future. При этом выполнение осуществляется синхронно!
Без явного указания способа запуска предполагается комбинация этих вариантов и фактически предсказать как именно будет запущена задача невозможно. Существуют связанные с этим сложности, например невозможно предсказать корректно ли будет обращение к переменным потока, невозможно предсказать будет ли выполнена функция вообще, если до выполнения функций .get() или .wait() дело так и не дошло ну и цикл ожидания готовности future никогда не закончится для отложенного сценария, ждать бесполезно.
Чтобы избежать недоразумений явно указывайте std::launch::async при запуске std::async.
Непредсказуемо:
auto f = std::async(sqrt, 9.0);
Гарантировано в другом потоке:
auto f = std::async(std::launch::async, sqrt, 9.0);
#15 Использование .get() может привести к ожиданию
#include <chrono>
#include <future>
#include <iostream>
int main(int argc, char *argv[])
{
std::future<int> f = std::async(std::launch::async, [](){
std::this_thread::sleep_for(std::chrono::seconds(1));
return 42;
});
while (true)
{
// ...
std::cout << f.get() << std::endl;
// ...
}
return 0;
}
Несмотря на то, что лямбда явно будет запущена в отдельном потоке вызов метода .get() может привести к нежелательному ожиданию. Более того, на следующей итерации код вообще аварийно завершится, потому что первый результат уже был выведен на консоль а никакого другого во future нет.
Обе проблемы можно решить проверив future на готовность.
if (f.valid())
{
std::cout << f.get() << std::endl;
}
#16 Исключение из задачи перетечёт во future
Исключение брошенное в асинхронной задаче должно быть обработано так, будто оно возникло в вызываемом потоке. В данном случае пример себя поведёт так, будто исключение никто не обработал.
#include <future>
#include <iostream>
int main(int argc, char *argv[])
{
std::future<int> f = std::async(std::launch::async, [](){
throw std::runtime_error("error");
return 42;
});
std::cout << f.get() << std::endl;
return 0;
}
Программа аварийно завершится. Если в задаче было брошено исключение, оно распространится и на вызов .get(). Если до конца жизни future .get() так и не будет вызван исключение просто проигнорируется.
Для таких задач имеет смысл использование обычной конструкции try/catch.
try
{
std::cout << f.get() << std::endl;
}
catch (const std::exception &e)
{
std::cout << e.what() << std::endl;
}
#17 Использование std::async там, где нужен тонкий контроль потоков
В большинстве случаев достаточно использования std::async, кроме ситуаций когда возникает необходимость в полном контроле над исполняемым потоком.
Например изменить параметры для планировщика:
#include <iostream>
#include <thread>
void foo()
{
std::cout << "foo" <<std::endl;
}
int main(int argc, char *argv[])
{
auto t = std::thread(foo);
sched_param sch;
int policy;
pthread_getschedparam(t.native_handle(), &policy, &sch);
sch.sched_priority = 20;
pthread_setschedparam(t.native_handle(), SCHED_FIFO, &sch);
t.join();
return 0;
}
Это возможно благодаря наличию метода .native_handle() у std::thread, значение которого можно использовать в POSIX системах. Использование этого метода полезно всегда, когда не хватает функциональности ни std::async ни std::thread. Использование std::async скрывает детали реализации и непригодно для такой тонкой работы.
#18 Пренебрежение анализом нагрузки на CPU
В любой момент времени потоки можно разделить на две группы - те которые что-то делают и те, который спят.
Потоки которые что-то делают занимают ядра процессора на которые их отправил планировщик. Для потоков которые занимаются активными вычислениями важно иметь в своём распоряжении свободные ядра, в противном случае большое количество потоков не даст никакого прироста в производительности. Даже скорее наоборот снизит производительность за счёт дополнительных переключений контекста потока.
Потоки которые преимущественно находятся в режиме ожидания в таком внимании процессора не нуждаются и могут находится в системе в гораздо большем количестве. И в случае с вводом/выводом даже помогают увеличить пропускную способность.
Я рассмотрел два крайних варианта, но наш конечно же будет посередине. Да, есть метод std::thread::hardware_concurrency(), которая сообщит нам сколько ядер доступно планировщику с учётом физических и логических.
Но это не помогает ответить правильно на вопрос - сколько же потоков можно запустить одновременно? Число ядер помогает понять сколько одновременно потоков, которые непрерывно длительное время активно потребляют процессор.
Если сценарий использования вычислений именно такой,то количество потоков должно быть как можно ближе к количеству ядер, а то и меньше, чтобы исключить распределение на логических ядрах.
Если потоки преимущественно заблокированы мьютексами или вводом/выводом, то ограничивать количество потоков ради экономии процессора не имеет большого смысла.
В остальных случаях необходимо нагрузочное тестирование и мониторинг с регулировкой количества потоков. Иными словами подбирается экспериментально.
#19 Использование квалификатора volatile для синхронизации
Использование этого квалификатора указание компилятору того, что изменения объекта могут происходить без контроля на этапе компиляции. Не в том смысле, что они происходят из разных потоков, а в том, что вообще за пределами кода, грубо говоря самопроизвольно. Это очень низкоуровневое указание и это никак не помогает в одновременном доступе внутри процесса.
Для синхронизации следует использовать atomic, mutex, и condition_variable.
#20 Неоправданное использование lock-free алгоритмов
Программирование без необходимости блокировок звучит очень привлекательно в сравнении с обычными механизмами синхронизации. Возможно, в случае жёсткого ограничения вычислительных ресурсов применение подобных алгоритмов может быть оправдано. В остальных случаях выглядит скорее преждевременной оптимизацией, которая, к тому же, может обернуться сложными ошибками в самое неподходящее время (и без coredump тут не обойтись).
Прежде чем приступить к использованию свободных от блокировок алгоритмов следует подумать над тремя вопросами:
- Пробовали ли спроектировать код без необходимости синхронизации?
- Выполняли ли анализ производительности, поиск и оптимизацию узких мест?
- Можно ли ограничиться горизонтальным масштабированием?
Использование lock-free алгоритмов оправдано, когда никаких других решений просто не осталось.
Надеюсь, чтение этого материала было так же полезно, как его перевод и публикация.
<!doctype html>
<html lang="ru" class="no-js no-touch ">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width,minimum-scale=1,initial-scale=1">
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
<meta name="yandex-verification" content="3019a35aeda6b45d" />
<script type="text/javascript">
var html = document.getElementsByTagName('html')[0];
html.className = html.className.replace('no-js', '');
window.React = {};
window.metrics = [];
window.TEST_ENV = false;
window.isSuperuser = false;
window.isStaff = false;
window.isHeadTeacher = false;
</script>
<script type="text/javascript">
window.DEBUG_COUNTERS = true;
var yaParams = {
'course_title': 'b',
'ab': 'b',
'features': JSON.parse('{"category-catalog-redirect": true, "landing-price-mode-switcher": true, "register-instead-of-start-test": true, "main-page-redesign": true, "subscription_genus_basic": true, "adblender": true, "greenlight": true, "phone": true, "professions": true, "course_enrol": true, "new_lessons_page": true, "jivosite": false, "course-page-single-screen": true, "right-price": false, "course_page_header_footer_new": true, "submit-application": false, "prof_dev_certificate": false, "assessment-react": true, "new-events-calendar": true, "new-pre-assessment-screen": true, "recommended_courses": true, "tinkoff_payment": true, "payment-page-3-front-refactor": true, "installment-calculator": true, "boomstream-player": true, "finsystems": true, "new-reviews": true}')
};
</script>
<!-- MindBox JavaScript SDK --->
<script>
mindbox = window.mindbox || function() { mindbox.queue.push(arguments); };
mindbox.queue = mindbox.queue || [];
mindbox('create');
</script>
<script src="https://api.mindbox.ru/scripts/v1/tracker.js" async></script>
<!-- End MindBox JavaScript SDK --->
<!-- Yandex.Metrika counter -->
<script type="text/javascript">
(function (ids) {
function yamInit(d, w, c, id) {
(w[c] = w[c] || []).push(function () {
try {
const metrika = new Ya.Metrika2({
id: id,
params: window.yaParams || {},
clickmap: true,
trackLinks: true,
accurateTrackBounce: true,
webvisor: true
});
w.yaMetriks.push(metrika);
w['yaCounter' + id] = metrika;
} catch (e) {
}
});
var n = d.getElementsByTagName('script')[0],
s = d.createElement('script'),
f = function () {
n.parentNode.insertBefore(s, n);
};
s.type = 'text/javascript';
s.async = true;
s.src = 'https://mc.yandex.ru/metrika/tag.js';
if (w.opera == '[object Opera]') {
d.addEventListener('DOMContentLoaded', f, false);
} else {
f();
}
}
window['yaMetriks'] = [];
(Array.isArray(ids) ? ids : [ids]).forEach(id => {
if (id) {
yamInit(document, window, 'yandex_metrika_callbacks2', id)
}
})
})([34531570, 82755226, 93715742])
</script>
<!-- /Yandex.Metrika counter -->
<script type="text/javascript" id="advcakeAsync">
(function (a) {
var b = a.createElement('script');
b.async = 1;
b.src = '//0gs25f.ru/';
a = a.getElementsByTagName('script')[0];
a.parentNode.insertBefore(b, a)
})(document);
window.advcake_data = window.advcake_data || [];
</script>
<script>
!(function (w, d, t) {
w.TiktokAnalyticsObject = t;
var ttq = (w[t] = w[t] || []);
(ttq.methods = [
'page',
'track',
'identify',
'instances',
'debug',
'on',
'off',
'once',
'ready',
'alias',
'group',
'enableCookie',
'disableCookie'
]),
(ttq.setAndDefer = function (t, e) {
t[e] = function () {
t.push([e].concat(Array.prototype.slice.call(arguments, 0)));
};
});
for (var i = 0; i < ttq.methods.length; i++)
ttq.setAndDefer(ttq, ttq.methods[i]);
(ttq.instance = function (t) {
for (var e = ttq._i[t] || [], n = 0; n < ttq.methods.length; n++)
ttq.setAndDefer(e, ttq.methods[n]);
return e;
}),
(ttq.load = function (e, n) {
var i = 'https://analytics.tiktok.com/i18n/pixel/events.js';
(ttq._i = ttq._i || {}),
(ttq._i[e] = []),
(ttq._i[e]._u = i),
(ttq._t = ttq._t || {}),
(ttq._t[e] = +new Date()),
(ttq._o = ttq._o || {}),
(ttq._o[e] = n || {});
var o = document.createElement('script');
(o.type = 'text/javascript'),
(o.async = !0),
(o.src = i + '?sdkid=' + e + '&lib=' + t);
var a = document.getElementsByTagName('script')[0];
a.parentNode.insertBefore(o, a);
});
ttq.load("C4IDL5C17T561FR1EMKG");
ttq.page();
})(window, document, 'ttq');
</script>
<script>
window.vkAsyncInit = function () {
VK.Retargeting.Init("VK-RTRG-410987-bLXUv");
VK.Retargeting.Hit();
}
</script>
<script src="//vk.com/js/api/openapi.js?159" async></script>
<noscript>
<img src="https://vk.com/rtrg?p=VK-RTRG-410987-bLXUv"
style="position:fixed; left:-999px;" alt="" />
</noscript>
<!-- rick.ai/q -->
<script type="text/javascript">
(function(e) {
var t = e.createElement("script");
t.src = "https://store-b2b.ru/tag.js?id=wsse7xcbtr07r1";
t.type = "module";
t.async = true;
t.crossorigin = "anonymous";
e.head.appendChild(t)
})(document)
</script>
<script type="text/javascript" nomodule src="https://store-b2b.ru/tag.js?id=wsse7xcbtr07r1&nomodule"></script>
<!-- end rick.ai/q -->
<script type="text/javascript">
!(function (n, e, t, r, a, s) {
function i(n, r) {
const a = e.createElement(t),
s = e.getElementsByTagName(t)[0];
(a.async = 1),
(a.src = n),
(a.onerror = r),
s.parentNode.insertBefore(a, s);
}
(n.SalesNinja = ['init', 'start', 'onPersonalization', 'reachGoal'].reduce(
(e, t) => {
return (
(e[t] = function () {
const e = Array.prototype.slice.call(arguments);
e.unshift(t), n[r].apply(0, e);
}),
e
);
},
{ k: r, ready: !1 }
)),
(n[r] = function () {
let e,
t,
a = new Promise((n, r) => {
(e = n), (t = r);
});
return (
(n[r].r = n[r].r || []).push({ s: e, f: t }),
(n[r].c = n[r].c || []).push(arguments),
a
);
}),
i(a, () => {
i(s);
});
})(
window,
document,
'script',
'ninja',
'https://cdn.sales-ninja.me/userBundle.js',
'https://bundle.sales-ninja.me/userBundle.js'
);
ninja('init', 'c20cc0ff-6d2f-42ac-9b66-1103c735a13a');
ninja('start');
</script>
<script type="text/javascript">
window.TMR_PIXEL_ID = 3316675;
var _tmr = window._tmr || (window._tmr = []);
_tmr.push({id: "3316675", type: "pageView", start: (new Date()).getTime()});
(function (d, w, id) {
if (d.getElementById(id)) return;
var ts = d.createElement("script"); ts.type = "text/javascript"; ts.async = true; ts.id = id;
ts.src = "https://top-fwz1.mail.ru/js/code.js";
var f = function () {var s = d.getElementsByTagName("script")[0]; s.parentNode.insertBefore(ts, s);};
if (w.opera == "[object Opera]") { d.addEventListener("DOMContentLoaded", f, false); } else { f(); }
})(document, window, "tmr-code");
</script>
<noscript>
<div>
<img src="https://top-fwz1.mail.ru/counter?id=3316675;js=na"
style="position:absolute;left:-9999px;" alt="Top.Mail.Ru" />
</div>
</noscript>
<!-- Pixel Tag Code -->
<script type="text/javascript">
(function (t, l, g, r, m) {
t[g] ||
((g = t[g] =
function () {
g.run ? g.run.apply(g, arguments) : g.queue.push(arguments);
}),
(g.queue = []),
(t = l.createElement(r)),
(t.async = !0),
(t.src = m),
(l = l.getElementsByTagName(r)[0]),
l.parentNode.insertBefore(t, l));
})(window, document, 'tgp', 'script', 'https://telegram.org/js/pixel.js');
tgp('init', '4bxSybss');
</script>
<!-- End Pixel Tag Code -->
<!-- GTM is no more -->
<link rel="canonical" href="https://otus.ru/nest/post/145/"/>
<link rel="amphtml" href="https://otus.ru/nest/post/145/?amp"/>
<title>Ошибки многопоточности в C++: 20 типичных примеров</title>
<meta property="og:type" content="website"/>
<meta property="og:url" content="https://otus.ru/nest/post/145/"/>
<meta property="og:title" content="Ошибки многопоточности в C++: 20 типичных примеров">
<meta property="og:description" content="20 типичных ошибок многопоточности в C++ в OTUS, только интересные посты!">
<meta name="description" content="20 типичных ошибок многопоточности в C++ в OTUS, только интересные посты!">
<meta property="og:site_name" content="Otus">
<meta property="fb:app_id" content="486413851704844"/>
<meta property="og:image" content="/static/img/favicons/android-chrome-537x240.jpg?nocache">
<meta property="og:image:width" content="537">
<meta property="og:image:height" content="240">
<link href="/static/img/favicons/android-chrome-537x240.jpg" rel="image_src"/>
<meta property="vk:image" content="/static/img/favicons/android-chrome-537x240.jpg">
<link href="/static/img/favicons/apple-touch-icon.png" rel="apple-touch-icon"/>
<link href="/static/img/favicons/apple-touch-icon-57x57.png" sizes="57x57" rel="apple-touch-icon"/>
<link href="/static/img/favicons/apple-touch-icon-60x60.png" sizes="60x60" rel="apple-touch-icon"/>
<link href="/static/img/favicons/apple-touch-icon-72x72.png" sizes="72x72" rel="apple-touch-icon"/>
<link href="/static/img/favicons/apple-touch-icon-76x76.png" sizes="76x76" rel="apple-touch-icon"/>
<link href="/static/img/favicons/apple-touch-icon-114x114.png" sizes="114x114" rel="apple-touch-icon"/>
<link href="/static/img/favicons/apple-touch-icon-120x120.png" sizes="120x120" rel="apple-touch-icon"/>
<link href="/static/img/favicons/apple-touch-icon-144x144.png" sizes="144x144" rel="apple-touch-icon"/>
<link href="/static/img/favicons/apple-touch-icon-152x152.png" sizes="152x152" rel="apple-touch-icon"/>
<link href="/static/img/favicons/apple-touch-icon-180x180.png" sizes="180x180" rel="apple-touch-icon"/>
<link type="image/png" href="/static/img/favicons/favicon-32x32.png" sizes="32x32" rel="icon"/>
<link type="image/png" href="/static/img/favicons/favicon-16x16.png" sizes="16x16" rel="icon"/>
<link type="image/x-icon" href="/static/img/favicons/favicon.ico" rel="shortcut icon"/>
<link rel="mask-icon" href="/static/img/favicons/safari-pinned-tab.svg" color="#000000"/>
<meta name="msapplication-TileColor" content="#ffffff">
<meta name="msapplication-TileImage" content="/static/img/favicons/mstile-144x144.png">
<meta name="theme-color" content="#FFFFFF"/>
<link rel="manifest" href="/manifest.json" crossorigin="use-credentials">
<meta name="sw" content="https://otus.ru/static/js/service-worker.8ed2b.js"/>
<meta name="csrf" id="meta-csrf" content="lXYhBGOxOepaErVDt9EvPCNH9v8oLtyRgmCExQIGsHTWoKQEjexlmkk5IShqMan6"/>
<meta name="auth" content="false"/>
<meta name="phone_confirmed" content="false"/>
<meta name="next" content='{"value": "/nest/post/145/"}'/>
<link href="https://otus.ru/static/css/vendor.react.dd7f4.css" rel="stylesheet" />
<link href="https://otus.ru/static/css/vendor.common.5ac2f.css" rel="stylesheet" />
<link href="https://otus.ru/static/css/otus-react:header-search.37c7a.css" rel="stylesheet" />
<link href="https://otus.ru/static/css/fonts.211eb.css" rel="stylesheet" />
<link href="https://otus.ru/static/css/otus-icons.e3e2d.css" rel="stylesheet" />
<link href="https://otus.ru/static/css/vendor.common.5ac2f.css" rel="stylesheet" />
<link href="https://otus.ru/static/css/vendor.otus-scss.59b5e.css" rel="stylesheet" />
<link href="https://otus.ru/static/css/otus-scss.2de41.css" rel="stylesheet" />
<script type="application/ld+json">
{
"@context": "https://schema.org",
"@type": "Organization",
"name": "Otus",
"url": "https://otus.ru",
"logo": "https://otus.ru/__new_static__/img/meta-image.png",
"sameAs": [
"https://vk.com/otusru",
"https://t.me/Otusjava"
]
}
</script>
</head>
<body class=" body-header3">
<script type="application/ld+json">
{
"@context" : "http://schema.org",
"@type" : "Organization",
"name" : "OTUS",
"url" : "https://otus.ru",
"logo": "https://otus.ru/static/img/favicons/apple-touch-icon-180x180.png",
"sameAs": [
"",
"https://www.youtube.com/channel/UCetgtvy93o3i3CvyGXKFU3g",
"https://www.instagram.com/otus.ru/"
],
"contactPoint": [
{
"@type": "ContactPoint",
"telephone": "+7-499-938-92-02",
"contactType": "customer service",
"areaServed": "RU"
}
]
}
</script>
<!-- TODO FIXME DRY -->>
<script
src="https://smartcaptcha.yandexcloud.net/captcha.js?render=onload&onload=smartCaptchaInit"
defer
></script>
<script>
function smartCaptchaInit() {
const sitekey = 'ysc1_cM9ClhSx0kwuG9QxSMfFmxHnC1gsW7Axbyddkmzref6982c0';
const test = false;
if (!window.smartCaptcha || !sitekey) {
return;
}
const widgetId = window.smartCaptcha.render('captcha-container', {
sitekey,
invisible: true,
test,
hideShield: true,
});
}
</script>
<div id="captcha-container"></div>
<script src="https://otus.ru/js-18n"></script>
<script>
window.texts = {
companyEmail: "help@otus.ru",
separateQuestionCount: " из "
}
window.language = 'ru-ru'
window.config = {
isEnablePhoneConfirm: true
}
</script>
<script src="https://otus.ru/js-18n"></script>
<script>
window.texts = {
companyEmail: "help@otus.ru",
separateQuestionCount: " из "
}
window.language = 'ru-ru'
</script>
<script type="application/ld+json">
{
"@context": "https://schema.org",
"@type": "BlogPosting",
"mainEntityOfPage": {
"@type": "WebPage",
"@id": "https://otus.ru/nest/post/145/"
},
"headline": "20 типичных ошибок многопоточности в C++",
"image": "",
"author": {
"@type": "Person",
"name": "Дмитрий",
"url": "https://otus.ru/profile/4868/"
},
"publisher": {
"@type": "Organization",
"name": "OTUS",
"logo": {
"@type": "ImageObject",
"url": "https://otus.ru/static/img/favicons/apple-touch-icon-180x180.png"
}
},
"datePublished": "2018-05-16T23:30:34.553062+03:00",
"dateModified": "2018-09-25T15:35:45+03:00"
}
</script>
<div class="body-wrapper">
<div class="body body_header3 drawer body_not-subscribed drawer--right blog-drawer ">
<div class="before-header-ui">
<div class="before-header-ui__ellipse1"></div>
<div class="before-header-ui__ellipse2"></div>
<div class="before-header-ui__container">
<div class="before-header-ui__img before-header-ui__img_sales"></div>
<div class="before-header-ui__content">
<div class="before-header-ui__title hide-phone">Курсы по нейросетям со скидкой до 30%</div>
<div class="before-header-ui__title show-phone">Курсы по нейросетям со скидкой до 30%</div>
</div>
<a href="https://otus.ru/catalog/courses?categories=neural_networks&utm_source=internal&utm_medium=free&utm_campaign=otus&utm_term=chank&utm_content=sla_sale_20-02-2026-10-04-2026" rel="nofollow noreferrer noopener" target="_blank" class="before-header-ui__button">Выбрать курс</a>
</div>
</div>
<header class="header3 js-header3">
<div class="header3__container">
<a class="header3__logo" href="/">
<img
class="header3__logo-img"
src="/static/img/logos/logo-2022-without-text.svg"
width="82"
height="42"
alt="Logo"
/>
</a>
<nav class="header3__nav">
<div id="headerSearch" class="header3__nav-item header3__nav-item-search">
<div class="header-search-icon">
<svg
class="header-search-icon__icon"
viewBox="0 0 24 24"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<circle
cx="11.767"
cy="11.767"
r="8.989"
stroke="currentColor"
stroke-width="1.5"
stroke-linecap="round"
stroke-linejoin="round"
></circle>
<path
d="M18.018 18.485 21.542 22"
stroke="currentColor"
stroke-width="1.5"
stroke-linecap="round"
stroke-linejoin="round"
></path>
</svg>
</div>
</div>
<div data-name="learning" class="header3__nav-item js-header3-popup-trigger header3__nav-item_only-desktop header3__nav-item_with-hover " >
<span title="Обучение" class="header3__nav-item-arrow-title">Обучение</span>
<div class="header3__nav-item-arrow-container js-header3-popup-arrow">
<svg
width="10"
height="5"
viewBox="0 0 10 5"
fill="none"
xmlns="http://www.w3.org/2000/svg"
class="header3__nav-item-arrow"
>
<path
fill-rule="evenodd"
clip-rule="evenodd"
d="M1.10067 0.378818C1.29593 0.183556 1.61251 0.183555 1.80778 0.378818L5.00023 3.57127L8.19272 0.378777C8.38798 0.183515 8.70457 0.183515 8.89983 0.378777C9.09509 0.574039 9.09509 0.890622 8.89983 1.08588L5.3643 4.62142C5.26426 4.72146 5.13237 4.77024 5.00127 4.76777C4.8695 4.77079 4.73676 4.72202 4.6362 4.62146L1.10067 1.08592C0.905408 0.890663 0.905408 0.57408 1.10067 0.378818Z"
fill="currentColor"
></path>
</svg>
</div>
</div>
<div class="header3__nav-item-popup-wrapper js-header3-popup" data-name="learning" style="display: none;">
<div class="header3__nav-item-popup-container js-header3-popup-container">
<div class="header3__nav-item-popup-content">
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title ">Направления</p>
<div class="header3__nav-section-items header3__nav-section-items_learning header3__nav-section-items_learning_rows-8">
<a
class="header3__nav-section-item"
href="/categories/programming/"
>
Программирование (117)
</a>
<a
class="header3__nav-section-item"
href="/categories/architecture/"
>
Архитектура (17)
</a>
<a
class="header3__nav-section-item"
href="/categories/data-science/"
>
Data Science (27)
</a>
<a
class="header3__nav-section-item"
href="/categories/operations/"
>
Инфраструктура (58)
</a>
<a
class="header3__nav-section-item"
href="/categories/gamedev/"
>
GameDev (10)
</a>
<a
class="header3__nav-section-item"
href="/categories/information-security-courses/"
>
Безопасность (15)
</a>
<a
class="header3__nav-section-item"
href="/categories/marketing-business/"
>
Управление (46)
</a>
<a
class="header3__nav-section-item"
href="/categories/analytics/"
>
Аналитика и анализ (25)
</a>
<a
class="header3__nav-section-item"
href="/categories/business-product/"
>
Бизнес и продукт в IT (26)
</a>
<a
class="header3__nav-section-item"
href="/categories/import-substitution/"
>
Импортозамещение (15)
</a>
<a
class="header3__nav-section-item"
href="/categories/testing/"
>
Тестирование (12)
</a>
<a
class="header3__nav-section-item"
href="/categories/neural_networks/"
>
Нейросети (9)
</a>
<a
class="header3__nav-section-item"
href="/categories/it-bez-programmirovanija/"
>
IT без программирования (19)
</a>
<a
class="header3__nav-section-item"
href="/categories/corporate/"
>
Корпоративные курсы (27)
</a>
</div>
</div>
</div>
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title ">События</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item"
href="/lessons/calendar/2026/"
>
Календарь запуска курсов
</a>
<a
class="header3__nav-section-item"
href="/events/near/"
>
Календарь мероприятий
</a>
</div>
</div>
<div>
<p class="header3__nav-section-title ">Другое</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item"
href="/categories/spec/"
>
Специализации (13)
</a>
<a
class="header3__nav-section-item"
href="/categories/online/"
>
Подготовительные курсы (14)
</a>
<a
class="header3__nav-section-item header3__nav-section-item_bold"
href="/subscription"
>
Подписка на курсы
</a>
<a
class="header3__nav-section-item"
href="/tests"
>
Проверьте свои знания
</a>
</div>
</div>
</div>
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title header3__nav-section-items-recommendation-title">OTUS рекомендует</p>
<div class="header3__nav-section-items header3__nav-section-items_not-items header3__nav-section-items-recommendation">
<a
href="/lessons/ai-dlya-analitiki-i-raboty-s-dannymi/"
class="header3__card-recommendation"
>
<div
class="header3__card-recommendation-background"
style="background: linear-gradient( 90deg,#0A4489, #00316B);"
></div>
<div class="header3__card-recommendation-header">
<div class="header3__card-recommendation-header-photo-wrapper" style="background: linear-gradient( 90deg,#0A4489, #00316B);">
<img
src="data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iODAiIGhlaWdodD0iODAiIHZpZXdCb3g9IjAgMCA4MCA4MCIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KICAgIDxwYXRoCiAgICAgICAgZD0iTTM1LjAwNDcgNDUuODcyNUMzNS4wMDQ3IDQ2LjkzMDYgMzYuMTA5IDQ3LjU3NTcgMzYuOTYwMyA0Ny4wMTQ5TDQ0LjYyOTIgNDEuOTYzMUM0NS4wMDI5IDQxLjcxNjkgNDUuMjI5OSA0MS4yODUyIDQ1LjIyOTkgNDAuODIwN1YyNi43NDg5QzQ1LjIyOTkgMjYuNzQ4OSA0Ny43ODYyIDI2Ljc0ODkgNTQuMTc3IDIyLjcwNzRDNTcuMzAxMiAyMC43MzE4IDU4Ljg5OCAxOC43NTYxIDU5LjcxNDMgMTcuMjUyNUM2MC4yNDY1IDE2LjI3MjEgNTkuMjUxMSAxNS41NTg4IDU4LjI5NjUgMTYuMDcyM0M1NS4zMjkyIDE3LjY2ODQgNDkuMzkxOCAyMC4wMTMyIDQwLjExNzMgMjAuMDEzMkMzMC44NDI4IDIwLjAxMzIgMjQuOTA1NCAxNy42Njg0IDIxLjkzODEgMTYuMDcyM0MyMC45ODM1IDE1LjU1ODggMTkuOTg4MSAxNi4yNzIxIDIwLjUyMDMgMTcuMjUyNUMyMS4zMzY2IDE4Ljc1NjEgMjIuOTMzNCAyMC43MzE4IDI2LjA1NzcgMjIuNzA3NEMzMi40NDg0IDI2Ljc0ODkgMzUuMDA0NyAyNi43NDg5IDM1LjAwNDcgMjYuNzQ4OVY0NS44NzI1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxlbGxpcHNlIGN4PSIyNi45NTM1IiBjeT0iMzUuNTUyNiIgcng9IjMuOTQ5MTQiIHJ5PSI0LjA4NTMyIiBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxlbGxpcHNlIGN4PSI1My4yODEyIiBjeT0iMzUuNTUyNiIgcng9IjMuOTQ5MTQiIHJ5PSI0LjA4NTMyIiBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTI1LjYzNzIgNTkuNjM1QzI1LjYzNzIgNTguNzk2MSAyNi4yMjY2IDU4LjExNiAyNi45NTM2IDU4LjExNkMyNy42ODA2IDU4LjExNiAyOC4yNyA1OC43OTYxIDI4LjI3IDU5LjYzNVY2Mi4wNDRDMjguMjcgNjIuNDQ2OSAyOC4xMzEzIDYyLjgzMzMgMjcuODg0NCA2My4xMTgxQzI3LjM3MDMgNjMuNzExNCAyNi41MzY4IDYzLjcxMTQgMjYuMDIyOCA2My4xMTgxQzI1Ljc3NTkgNjIuODMzMyAyNS42MzcyIDYyLjQ0NjkgMjUuNjM3MiA2Mi4wNDRWNTkuNjM1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTQ0LjA2NjUgNTkuNjM1QzQ0LjA2NjUgNTguNzk2MSA0NC42NTU5IDU4LjExNiA0NS4zODI5IDU4LjExNkM0Ni4xMDk5IDU4LjExNiA0Ni42OTkzIDU4Ljc5NjEgNDYuNjk5MyA1OS42MzVWNjIuMDQ0QzQ2LjY5OTMgNjIuNDQ2OSA0Ni41NjA2IDYyLjgzMzMgNDYuMzEzOCA2My4xMTgxQzQ1Ljc5OTcgNjMuNzExNCA0NC45NjYyIDYzLjcxMTQgNDQuNDUyMSA2My4xMTgxQzQ0LjIwNTIgNjIuODMzMyA0NC4wNjY1IDYyLjQ0NjkgNDQuMDY2NSA2Mi4wNDRWNTkuNjM1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTI5LjU4NjQgNTkuNjM1QzI5LjU4NjQgNTguNzk2MSAzMC4xNzU3IDU4LjExNiAzMC45MDI3IDU4LjExNkMzMS42Mjk4IDU4LjExNiAzMi4yMTkxIDU4Ljc5NjEgMzIuMjE5MSA1OS42MzVWNjIuMDQ0QzMyLjIxOTEgNjIuNDQ2OSAzMi4wODA0IDYyLjgzMzMgMzEuODMzNiA2My4xMTgxQzMxLjMxOTUgNjMuNzExNCAzMC40ODYgNjMuNzExNCAyOS45NzE5IDYzLjExODFDMjkuNzI1IDYyLjgzMzMgMjkuNTg2NCA2Mi40NDY5IDI5LjU4NjQgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik00OC4wMTU3IDU5LjYzNUM0OC4wMTU3IDU4Ljc5NjEgNDguNjA1MSA1OC4xMTYgNDkuMzMyMSA1OC4xMTZDNTAuMDU5MSA1OC4xMTYgNTAuNjQ4NSA1OC43OTYxIDUwLjY0ODUgNTkuNjM1VjYyLjA0NEM1MC42NDg1IDYyLjQ0NjkgNTAuNTA5OCA2Mi44MzMzIDUwLjI2MjkgNjMuMTE4MUM0OS43NDg4IDYzLjcxMTQgNDguOTE1MyA2My43MTE0IDQ4LjQwMTIgNjMuMTE4MUM0OC4xNTQ0IDYyLjgzMzMgNDguMDE1NyA2Mi40NDY5IDQ4LjAxNTcgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik0zMy41MzU1IDU5LjYzNUMzMy41MzU1IDU4Ljc5NjEgMzQuMTI0OSA1OC4xMTYgMzQuODUxOSA1OC4xMTZDMzUuNTc4OSA1OC4xMTYgMzYuMTY4MyA1OC43OTYxIDM2LjE2ODMgNTkuNjM1VjYyLjA0NEMzNi4xNjgzIDYyLjQ0NjkgMzYuMDI5NiA2Mi44MzMzIDM1Ljc4MjcgNjMuMTE4MUMzNS4yNjg2IDYzLjcxMTQgMzQuNDM1MSA2My43MTE0IDMzLjkyMTEgNjMuMTE4MUMzMy42NzQyIDYyLjgzMzMgMzMuNTM1NSA2Mi40NDY5IDMzLjUzNTUgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik01MS45NjQ4IDU5LjYzNUM1MS45NjQ4IDU4Ljc5NjEgNTIuNTU0MiA1OC4xMTYgNTMuMjgxMiA1OC4xMTZDNTQuMDA4MiA1OC4xMTYgNTQuNTk3NiA1OC43OTYxIDU0LjU5NzYgNTkuNjM1VjYyLjA0NEM1NC41OTc2IDYyLjQ0NjkgNTQuNDU4OSA2Mi44MzMzIDU0LjIxMiA2My4xMTgxQzUzLjY5OCA2My43MTE0IDUyLjg2NDUgNjMuNzExNCA1Mi4zNTA0IDYzLjExODFDNTIuMTAzNSA2Mi44MzMzIDUxLjk2NDggNjIuNDQ2OSA1MS45NjQ4IDYyLjA0NFY1OS42MzVaIgogICAgICAgIGZpbGw9IndoaXRlIiAvPgo8L3N2Zz4="
class="header3__card-recommendation-header-photo header3__card-recommendation-header-photo_default"
/>
<img
src="data:image/svg+xml;base64,<svg width="80" height="80" viewBox="0 0 80 80" fill="none" xmlns="http://www.w3.org/2000/svg">
    <mask id="a" fill="#fff">
        <path fill-rule="evenodd" clip-rule="evenodd"
            d="M0 12C0 5.373 5.373 0 12 0h56c6.627 0 12 5.373 12 12v56c0 6.627-5.373 12-12 12H12C5.373 80 0 74.627 0 68V12Zm35.005 33.873c0 1.058 1.104 1.703 1.955 1.142l7.67-5.052c.373-.246.6-.678.6-1.142V26.749s2.556 0 8.947-4.042c3.124-1.975 4.721-3.951 5.537-5.454.533-.981-.463-1.694-1.417-1.18-2.968 1.595-8.905 3.94-18.18 3.94-9.274 0-15.212-2.345-18.179-3.94-.954-.514-1.95.199-1.418 1.18.817 1.503 2.413 3.479 5.538 5.454 6.39 4.042 8.947 4.042 8.947 4.042v19.124Zm-8.051-6.235c2.18 0 3.949-1.83 3.949-4.085 0-2.257-1.768-4.086-3.95-4.086-2.18 0-3.949 1.83-3.949 4.086 0 2.256 1.768 4.085 3.95 4.085Zm30.276-4.085c0 2.256-1.768 4.085-3.949 4.085-2.18 0-3.949-1.83-3.949-4.085 0-2.257 1.768-4.086 3.95-4.086 2.18 0 3.948 1.83 3.948 4.086ZM26.954 58.116c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.139.79.386 1.074.514.593 1.347.593 1.861 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Zm18.429 0c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.14.79.386 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.59-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.386 1.074-.514.593-1.347.593-1.861 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.589-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.385 1.074-.514.593-1.348.593-1.862 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Z" />
    </mask>
    <path fill-rule="evenodd" clip-rule="evenodd"
        d="M0 12C0 5.373 5.373 0 12 0h56c6.627 0 12 5.373 12 12v56c0 6.627-5.373 12-12 12H12C5.373 80 0 74.627 0 68V12Zm35.005 33.873c0 1.058 1.104 1.703 1.955 1.142l7.67-5.052c.373-.246.6-.678.6-1.142V26.749s2.556 0 8.947-4.042c3.124-1.975 4.721-3.951 5.537-5.454.533-.981-.463-1.694-1.417-1.18-2.968 1.595-8.905 3.94-18.18 3.94-9.274 0-15.212-2.345-18.179-3.94-.954-.514-1.95.199-1.418 1.18.817 1.503 2.413 3.479 5.538 5.454 6.39 4.042 8.947 4.042 8.947 4.042v19.124Zm-8.051-6.235c2.18 0 3.949-1.83 3.949-4.085 0-2.257-1.768-4.086-3.95-4.086-2.18 0-3.949 1.83-3.949 4.086 0 2.256 1.768 4.085 3.95 4.085Zm30.276-4.085c0 2.256-1.768 4.085-3.949 4.085-2.18 0-3.949-1.83-3.949-4.085 0-2.257 1.768-4.086 3.95-4.086 2.18 0 3.948 1.83 3.948 4.086ZM26.954 58.116c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.139.79.386 1.074.514.593 1.347.593 1.861 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Zm18.429 0c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.14.79.386 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.59-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.386 1.074-.514.593-1.347.593-1.861 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.589-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.385 1.074-.514.593-1.348.593-1.862 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Z"
        fill="#fff" />
    <path
        d="m36.96 47.015 1.1 1.67-1.1-1.67Zm7.67-5.052 1.1 1.67-1.1-1.67Zm.6-15.214v-2h-2v2h2Zm8.947-4.042 1.069 1.69-1.069-1.69Zm5.537-5.454 1.758.954-1.758-.954Zm-1.417-1.18.947 1.76-.947-1.76Zm-36.359 0-.947 1.76.947-1.76Zm-1.418 1.18 1.758-.955-1.758.955Zm5.538 5.454-1.07 1.69 1.07-1.69Zm8.947 4.042h2v-2h-2v2Zm-8.982 36.37 1.511-1.31-1.511 1.31Zm1.861 0 1.512 1.309-1.512-1.31Zm16.568 0-1.511 1.309 1.511-1.31Zm1.862 0 1.511 1.309-1.511-1.31Zm-14.48 0-1.512-1.31 1.512 1.31Zm-1.862 0 1.511-1.31-1.511 1.31Zm18.43 0-1.512 1.309 1.511-1.31Zm1.86 0 1.512 1.309-1.511-1.31Zm-14.48 0 1.512 1.309-1.511-1.31Zm-1.861 0-1.511 1.309 1.511-1.31Zm18.43 0 1.51-1.31-1.51 1.31Zm1.861 0 1.511 1.309-1.511-1.31ZM12-2C4.268-2-2 4.268-2 12h4C2 6.477 6.477 2 12 2v-4Zm56 0H12v4h56v-4Zm14 14c0-7.732-6.268-14-14-14v4c5.523 0 10 4.477 10 10h4Zm0 56V12h-4v56h4ZM68 82c7.732 0 14-6.268 14-14h-4c0 5.523-4.477 10-10 10v4Zm-56 0h56v-4H12v4ZM-2 68c0 7.732 6.268 14 14 14v-4C6.477 78 2 73.523 2 68h-4Zm0-56v56h4V12h-4Zm37.86 33.345a.791.791 0 0 1 .812-.021.656.656 0 0 1 .333.549h-4c0 2.454 2.714 4.355 5.055 2.812l-2.2-3.34Zm7.669-5.052-7.669 5.052 2.2 3.34 7.67-5.052-2.201-3.34Zm-.299.528c0-.178.086-.388.299-.528l2.2 3.34a3.365 3.365 0 0 0 1.501-2.812h-4Zm0-14.072V40.82h4V26.749h-4Zm9.878-5.732c-3.097 1.959-5.197 2.895-6.464 3.34a8.845 8.845 0 0 1-1.28.362 2.42 2.42 0 0 1-.203.031H45.19l.018-.001h.02c.001 0 .003 0 .003 2s.002 2 .003 2h.054a3.196 3.196 0 0 0 .256-.017c.144-.014.327-.038.552-.08.45-.084 1.069-.238 1.875-.521 1.61-.566 3.982-1.65 7.276-3.733l-2.138-3.381Zm4.849-4.719c-.637 1.173-1.983 2.906-4.849 4.719l2.138 3.38c3.382-2.138 5.23-4.356 6.226-6.19l-3.515-1.909Zm1.287 1.536c-.106.056-.494.173-.916-.158a1.233 1.233 0 0 1-.448-.76.985.985 0 0 1 .077-.618l3.515 1.909c.308-.568.472-1.248.351-1.96a2.768 2.768 0 0 0-1.025-1.718c-1.026-.805-2.388-.788-3.449-.218l1.895 3.523Zm-19.127 4.18c9.628 0 15.889-2.438 19.127-4.18l-1.895-3.523c-2.696 1.45-8.31 3.702-17.232 3.702v4Zm-19.126-4.18c3.238 1.742 9.499 4.18 19.126 4.18v-4c-8.921 0-14.535-2.253-17.231-3.703l-1.895 3.523Zm1.287-1.536c.042.078.131.3.077.619a1.232 1.232 0 0 1-.449.759c-.422.33-.81.215-.915.158l1.895-3.523c-1.06-.57-2.424-.587-3.449.218a2.768 2.768 0 0 0-1.026 1.719c-.12.711.043 1.391.352 1.959l3.515-1.909Zm4.849 4.719c-2.867-1.813-4.212-3.546-4.849-4.719l-3.515 1.909c.995 1.834 2.844 4.052 6.226 6.19l2.138-3.38Zm7.878 5.732c0-2 .001-2 .003-2H35.047l.025.001h.001a2.42 2.42 0 0 1-.202-.03 8.852 8.852 0 0 1-1.28-.363c-1.267-.445-3.367-1.381-6.464-3.34l-2.138 3.38c3.293 2.084 5.666 3.168 7.276 3.734.806.283 1.425.437 1.875.52a6.356 6.356 0 0 0 .737.095l.071.002h.054c.001 0 .003 0 .003-2Zm2 19.124V26.749h-4v19.124h4Zm-8.102-10.32c0 1.216-.936 2.085-1.95 2.085v4c3.35 0 5.95-2.789 5.95-6.085h-4Zm-1.95-2.086c1.014 0 1.95.87 1.95 2.086h4c0-3.297-2.6-6.086-5.95-6.086v4Zm-1.949 2.086c0-1.216.936-2.086 1.95-2.086v-4c-3.35 0-5.95 2.79-5.95 6.086h4Zm1.95 2.085c-1.014 0-1.95-.87-1.95-2.085h-4c0 3.296 2.6 6.085 5.95 6.085v-4Zm26.327 4c3.35 0 5.95-2.789 5.95-6.085h-4c0 1.216-.937 2.085-1.95 2.085v4Zm-5.949-6.085c0 3.296 2.6 6.085 5.95 6.085v-4c-1.014 0-1.95-.87-1.95-2.085h-4Zm5.95-6.086c-3.35 0-5.95 2.79-5.95 6.086h4c0-1.216.936-2.086 1.95-2.086v-4Zm5.948 6.086c0-3.297-2.6-6.086-5.949-6.086v4c1.013 0 1.95.87 1.95 2.086h4ZM27.637 59.635a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4c-2.091 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.45.45 0 0 1 .086.142.277.277 0 0 1 .017.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.161 0a.776.776 0 0 1 .58-.245c.27 0 .476.124.581.245l-3.023 2.62c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62Zm-.103.236c0-.027.004-.058.017-.094a.45.45 0 0 1 .086-.142l3.023 2.62a3.644 3.644 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.683.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.103-.236h4c0-1.665-1.226-3.519-3.317-3.519v4Zm19.113-.481a.37.37 0 0 1-.102.236.774.774 0 0 1-.581.245v-4c-2.091 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.102-.236a.45.45 0 0 1 .085.142.277.277 0 0 1 .017.094h-4c0 .848.29 1.71.875 2.384l3.023-2.62Zm-1.162 0a.775.775 0 0 1 .58-.245c.271 0 .477.124.582.245l-3.023 2.62c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62Zm-.103.236c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62a3.645 3.645 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.684.481c-.29 0-.49-.14-.58-.245a.369.369 0 0 1-.104-.236h4c0-1.665-1.225-3.519-3.316-3.519v4Zm-14.48-4c-2.091 0-3.317 1.854-3.317 3.519h4a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4Zm3.316 3.519c0-1.665-1.225-3.519-3.316-3.519v4c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4Zm0 2.409v-2.409h-4v2.409h4Zm-.874 2.384a3.644 3.644 0 0 0 .874-2.384h-4c0-.027.005-.058.017-.094a.45.45 0 0 1 .086-.142l3.023 2.62Zm-4.885 0c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62a.775.775 0 0 1 .58-.245c.27 0 .476.124.581.245l-3.023 2.62Zm-.874-2.384c0 .848.29 1.71.874 2.384l3.023-2.62a.45.45 0 0 1 .086.142.277.277 0 0 1 .017.094h-4Zm0-2.409v2.409h4v-2.409h-4Zm22.43 0a.37.37 0 0 1-.103.236.774.774 0 0 1-.581.245v-4c-2.09 0-3.316 1.854-3.316 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.453.453 0 0 1 .085.142.277.277 0 0 1 .018.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.162 0a.776.776 0 0 1 .581-.245c.27 0 .476.124.58.245l-3.022 2.62c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62Zm-.103.236c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62a3.644 3.644 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.684.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4c0-1.665-1.225-3.519-3.316-3.519v4Zm-14.48-4c-2.091 0-3.316 1.854-3.316 3.519h4a.37.37 0 0 1-.104.236.774.774 0 0 1-.58.245v-4Zm3.316 3.519c0-1.665-1.225-3.519-3.316-3.519v4c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4Zm0 2.409v-2.409h-4v2.409h4Zm-.874 2.384a3.644 3.644 0 0 0 .874-2.384h-4c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62Zm-4.884 0c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62a.776.776 0 0 1 .58-.245c.271 0 .477.124.581.245l-3.022 2.62Zm-.874-2.384c0 .848.289 1.71.874 2.384l3.023-2.62a.453.453 0 0 1 .085.142.277.277 0 0 1 .017.094h-4Zm0-2.409v2.409h4v-2.409h-4Zm22.429 0a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4c-2.092 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.45.45 0 0 1 .085.142.277.277 0 0 1 .018.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.162 0a.776.776 0 0 1 .581-.245c.27 0 .476.124.58.245l-3.022 2.62c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62Zm-.102.236c0-.027.004-.058.017-.094a.455.455 0 0 1 .085-.142l3.023 2.62a3.645 3.645 0 0 0 .875-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.683.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.103-.236h4c0-1.665-1.226-3.519-3.317-3.519v4Z"
        fill="#fff" mask="url(#a)" />
</svg>"
class="header3__card-recommendation-header-photo header3__card-recommendation-header-photo_default_inverted"
/>
</div>
<div class="header3__card-recommendation-header-content">
<div class="header3__card-recommendation-header-content-tag">
Курс
</div>
<div class="header3__card-recommendation-header-sub">
<span
class="header3__card-recommendation-header-sub-chunk header3__card-recommendation-header-sub-chunk_green"
>
Скидка 20000 ₽
</span>
</div>
</div>
</div>
<h6 class="header3__card-recommendation-title">
ИИ для аналитики и работы с данными
</h6>
<div class="header3__card-recommendation-footer">
12 марта
, 2026
· 2 месяца
</div>
</a>
</div>
</div>
</div>
</div>
<svg
class="header3__nav-item-popup-figure"
viewBox="0 0 600 600"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M125.573 140.619C131.659 71.6017 210.245 34.9562 267.027 74.6573L553.942 275.262C610.723 314.962 603.117 401.233 540.247 430.55L222.58 578.681C159.71 607.997 88.7344 558.37 94.8204 489.355L125.573 140.619Z"
stroke="#eaeaea"
class="header3__nav-item-popup-figure-spinner"
></path>
<path
d="M148.472 246.647C133.624 191.005 184.615 140.013 240.257 154.862L519.856 229.476C575.498 244.325 594.059 313.877 553.266 354.67L348.281 559.656C307.488 600.449 237.935 581.888 223.087 526.246L148.472 246.647Z"
fill="url(#paint0_linear-learning)"
></path>
<defs>
<linearGradient
id="paint0_linear-learning"
x1="128.696"
y1="395.739"
x2="443.538"
y2="180.173"
gradientUnits="userSpaceOnUse"
>
<stop stop-color="#f9f9f9"></stop>
<stop offset="1" stop-color="#EBEBEB"></stop>
</linearGradient>
</defs>
</svg>
</div>
</div>
<div data-name="info" class="header3__nav-item js-header3-popup-trigger header3__nav-item_only-desktop header3__nav-item_with-hover " >
<span title="Информация" class="header3__nav-item-arrow-title">Информация</span>
<div class="header3__nav-item-arrow-container js-header3-popup-arrow">
<svg
width="10"
height="5"
viewBox="0 0 10 5"
fill="none"
xmlns="http://www.w3.org/2000/svg"
class="header3__nav-item-arrow"
>
<path
fill-rule="evenodd"
clip-rule="evenodd"
d="M1.10067 0.378818C1.29593 0.183556 1.61251 0.183555 1.80778 0.378818L5.00023 3.57127L8.19272 0.378777C8.38798 0.183515 8.70457 0.183515 8.89983 0.378777C9.09509 0.574039 9.09509 0.890622 8.89983 1.08588L5.3643 4.62142C5.26426 4.72146 5.13237 4.77024 5.00127 4.76777C4.8695 4.77079 4.73676 4.72202 4.6362 4.62146L1.10067 1.08592C0.905408 0.890663 0.905408 0.57408 1.10067 0.378818Z"
fill="currentColor"
></path>
</svg>
</div>
</div>
<div class="header3__nav-item-popup-wrapper js-header3-popup" data-name="info" style="display: none;">
<div class="header3__nav-item-popup-container js-header3-popup-container">
<div class="header3__nav-item-popup-content">
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title ">OTUS</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item"
href="/about"
>
О компании
</a>
<a
class="header3__nav-section-item"
href="/smi/"
>
СМИ о нас
</a>
<a
class="header3__nav-section-item js-stats"
href="/journal/"
target="_blank"
rel="noreferrer nofollow"
data-event="header;click_otus_journal"
data-goal="click_otus_journal"
>
OTUS Журнал
</a>
<a
class="header3__nav-section-item"
href="https://direct.otus.ru/"
target="_blank"
rel="noreferrer nofollow"
>
OTUS Директ
</a>
<a
class="header3__nav-section-item"
href="/legal/common/"
>
Сведения об образовательной организации
</a>
<a
class="header3__nav-section-item"
href="/contacts/"
>
Контактная информация
</a>
</div>
</div>
</div>
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title ">Студентам</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item"
href="/reviews"
>
Отзывы
</a>
<a
class="header3__nav-section-item"
target="_blank"
href="https://landing.otus.ru/about-otus"
>
Как выбрать курс
</a>
<a
class="header3__nav-section-item"
target="_blank"
href="https://landing.otus.ru/gallery"
>
Истории выпускников
</a>
<a
class="header3__nav-section-item"
href="/instructors/"
>
Наши преподаватели
</a>
<a
class="header3__nav-section-item"
href="/employers/all/"
>
Наши партнеры
</a>
<a
class="header3__nav-section-item"
href="/about/loyalty/"
>
Программа лояльности
</a>
<a
class="header3__nav-section-item"
href="/faq/"
>
Вопросы и ответы
</a>
</div>
</div>
</div>
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title ">Преподавателям</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item"
href="/teach/"
>
Стать преподавателем
</a>
<a
class="header3__nav-section-item"
href="/instructors/"
>
Наши преподаватели
</a>
<a
class="header3__nav-section-item"
href="/nest/dlja-prepodavatelej/"
>
База знаний
</a>
</div>
</div>
</div>
<div class="header3__nav-column header3__nav-column_space-between">
<div>
<p class="header3__nav-section-title header3__nav-section-items-recommendation-title">OTUS рекомендует</p>
<div class="header3__nav-section-items header3__nav-section-items_not-items header3__nav-section-items-recommendation">
<a
href="/lessons/ai-dlya-analitiki-i-raboty-s-dannymi/"
class="header3__card-recommendation"
>
<div
class="header3__card-recommendation-background"
style="background: linear-gradient( 90deg,#0A4489, #00316B);"
></div>
<div class="header3__card-recommendation-header">
<div class="header3__card-recommendation-header-photo-wrapper" style="background: linear-gradient( 90deg,#0A4489, #00316B);">
<img
src="data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iODAiIGhlaWdodD0iODAiIHZpZXdCb3g9IjAgMCA4MCA4MCIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KICAgIDxwYXRoCiAgICAgICAgZD0iTTM1LjAwNDcgNDUuODcyNUMzNS4wMDQ3IDQ2LjkzMDYgMzYuMTA5IDQ3LjU3NTcgMzYuOTYwMyA0Ny4wMTQ5TDQ0LjYyOTIgNDEuOTYzMUM0NS4wMDI5IDQxLjcxNjkgNDUuMjI5OSA0MS4yODUyIDQ1LjIyOTkgNDAuODIwN1YyNi43NDg5QzQ1LjIyOTkgMjYuNzQ4OSA0Ny43ODYyIDI2Ljc0ODkgNTQuMTc3IDIyLjcwNzRDNTcuMzAxMiAyMC43MzE4IDU4Ljg5OCAxOC43NTYxIDU5LjcxNDMgMTcuMjUyNUM2MC4yNDY1IDE2LjI3MjEgNTkuMjUxMSAxNS41NTg4IDU4LjI5NjUgMTYuMDcyM0M1NS4zMjkyIDE3LjY2ODQgNDkuMzkxOCAyMC4wMTMyIDQwLjExNzMgMjAuMDEzMkMzMC44NDI4IDIwLjAxMzIgMjQuOTA1NCAxNy42Njg0IDIxLjkzODEgMTYuMDcyM0MyMC45ODM1IDE1LjU1ODggMTkuOTg4MSAxNi4yNzIxIDIwLjUyMDMgMTcuMjUyNUMyMS4zMzY2IDE4Ljc1NjEgMjIuOTMzNCAyMC43MzE4IDI2LjA1NzcgMjIuNzA3NEMzMi40NDg0IDI2Ljc0ODkgMzUuMDA0NyAyNi43NDg5IDM1LjAwNDcgMjYuNzQ4OVY0NS44NzI1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxlbGxpcHNlIGN4PSIyNi45NTM1IiBjeT0iMzUuNTUyNiIgcng9IjMuOTQ5MTQiIHJ5PSI0LjA4NTMyIiBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxlbGxpcHNlIGN4PSI1My4yODEyIiBjeT0iMzUuNTUyNiIgcng9IjMuOTQ5MTQiIHJ5PSI0LjA4NTMyIiBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTI1LjYzNzIgNTkuNjM1QzI1LjYzNzIgNTguNzk2MSAyNi4yMjY2IDU4LjExNiAyNi45NTM2IDU4LjExNkMyNy42ODA2IDU4LjExNiAyOC4yNyA1OC43OTYxIDI4LjI3IDU5LjYzNVY2Mi4wNDRDMjguMjcgNjIuNDQ2OSAyOC4xMzEzIDYyLjgzMzMgMjcuODg0NCA2My4xMTgxQzI3LjM3MDMgNjMuNzExNCAyNi41MzY4IDYzLjcxMTQgMjYuMDIyOCA2My4xMTgxQzI1Ljc3NTkgNjIuODMzMyAyNS42MzcyIDYyLjQ0NjkgMjUuNjM3MiA2Mi4wNDRWNTkuNjM1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTQ0LjA2NjUgNTkuNjM1QzQ0LjA2NjUgNTguNzk2MSA0NC42NTU5IDU4LjExNiA0NS4zODI5IDU4LjExNkM0Ni4xMDk5IDU4LjExNiA0Ni42OTkzIDU4Ljc5NjEgNDYuNjk5MyA1OS42MzVWNjIuMDQ0QzQ2LjY5OTMgNjIuNDQ2OSA0Ni41NjA2IDYyLjgzMzMgNDYuMzEzOCA2My4xMTgxQzQ1Ljc5OTcgNjMuNzExNCA0NC45NjYyIDYzLjcxMTQgNDQuNDUyMSA2My4xMTgxQzQ0LjIwNTIgNjIuODMzMyA0NC4wNjY1IDYyLjQ0NjkgNDQuMDY2NSA2Mi4wNDRWNTkuNjM1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTI5LjU4NjQgNTkuNjM1QzI5LjU4NjQgNTguNzk2MSAzMC4xNzU3IDU4LjExNiAzMC45MDI3IDU4LjExNkMzMS42Mjk4IDU4LjExNiAzMi4yMTkxIDU4Ljc5NjEgMzIuMjE5MSA1OS42MzVWNjIuMDQ0QzMyLjIxOTEgNjIuNDQ2OSAzMi4wODA0IDYyLjgzMzMgMzEuODMzNiA2My4xMTgxQzMxLjMxOTUgNjMuNzExNCAzMC40ODYgNjMuNzExNCAyOS45NzE5IDYzLjExODFDMjkuNzI1IDYyLjgzMzMgMjkuNTg2NCA2Mi40NDY5IDI5LjU4NjQgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik00OC4wMTU3IDU5LjYzNUM0OC4wMTU3IDU4Ljc5NjEgNDguNjA1MSA1OC4xMTYgNDkuMzMyMSA1OC4xMTZDNTAuMDU5MSA1OC4xMTYgNTAuNjQ4NSA1OC43OTYxIDUwLjY0ODUgNTkuNjM1VjYyLjA0NEM1MC42NDg1IDYyLjQ0NjkgNTAuNTA5OCA2Mi44MzMzIDUwLjI2MjkgNjMuMTE4MUM0OS43NDg4IDYzLjcxMTQgNDguOTE1MyA2My43MTE0IDQ4LjQwMTIgNjMuMTE4MUM0OC4xNTQ0IDYyLjgzMzMgNDguMDE1NyA2Mi40NDY5IDQ4LjAxNTcgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik0zMy41MzU1IDU5LjYzNUMzMy41MzU1IDU4Ljc5NjEgMzQuMTI0OSA1OC4xMTYgMzQuODUxOSA1OC4xMTZDMzUuNTc4OSA1OC4xMTYgMzYuMTY4MyA1OC43OTYxIDM2LjE2ODMgNTkuNjM1VjYyLjA0NEMzNi4xNjgzIDYyLjQ0NjkgMzYuMDI5NiA2Mi44MzMzIDM1Ljc4MjcgNjMuMTE4MUMzNS4yNjg2IDYzLjcxMTQgMzQuNDM1MSA2My43MTE0IDMzLjkyMTEgNjMuMTE4MUMzMy42NzQyIDYyLjgzMzMgMzMuNTM1NSA2Mi40NDY5IDMzLjUzNTUgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik01MS45NjQ4IDU5LjYzNUM1MS45NjQ4IDU4Ljc5NjEgNTIuNTU0MiA1OC4xMTYgNTMuMjgxMiA1OC4xMTZDNTQuMDA4MiA1OC4xMTYgNTQuNTk3NiA1OC43OTYxIDU0LjU5NzYgNTkuNjM1VjYyLjA0NEM1NC41OTc2IDYyLjQ0NjkgNTQuNDU4OSA2Mi44MzMzIDU0LjIxMiA2My4xMTgxQzUzLjY5OCA2My43MTE0IDUyLjg2NDUgNjMuNzExNCA1Mi4zNTA0IDYzLjExODFDNTIuMTAzNSA2Mi44MzMzIDUxLjk2NDggNjIuNDQ2OSA1MS45NjQ4IDYyLjA0NFY1OS42MzVaIgogICAgICAgIGZpbGw9IndoaXRlIiAvPgo8L3N2Zz4="
class="header3__card-recommendation-header-photo header3__card-recommendation-header-photo_default"
/>
<img
src="data:image/svg+xml;base64,<svg width="80" height="80" viewBox="0 0 80 80" fill="none" xmlns="http://www.w3.org/2000/svg">
    <mask id="a" fill="#fff">
        <path fill-rule="evenodd" clip-rule="evenodd"
            d="M0 12C0 5.373 5.373 0 12 0h56c6.627 0 12 5.373 12 12v56c0 6.627-5.373 12-12 12H12C5.373 80 0 74.627 0 68V12Zm35.005 33.873c0 1.058 1.104 1.703 1.955 1.142l7.67-5.052c.373-.246.6-.678.6-1.142V26.749s2.556 0 8.947-4.042c3.124-1.975 4.721-3.951 5.537-5.454.533-.981-.463-1.694-1.417-1.18-2.968 1.595-8.905 3.94-18.18 3.94-9.274 0-15.212-2.345-18.179-3.94-.954-.514-1.95.199-1.418 1.18.817 1.503 2.413 3.479 5.538 5.454 6.39 4.042 8.947 4.042 8.947 4.042v19.124Zm-8.051-6.235c2.18 0 3.949-1.83 3.949-4.085 0-2.257-1.768-4.086-3.95-4.086-2.18 0-3.949 1.83-3.949 4.086 0 2.256 1.768 4.085 3.95 4.085Zm30.276-4.085c0 2.256-1.768 4.085-3.949 4.085-2.18 0-3.949-1.83-3.949-4.085 0-2.257 1.768-4.086 3.95-4.086 2.18 0 3.948 1.83 3.948 4.086ZM26.954 58.116c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.139.79.386 1.074.514.593 1.347.593 1.861 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Zm18.429 0c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.14.79.386 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.59-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.386 1.074-.514.593-1.347.593-1.861 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.589-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.385 1.074-.514.593-1.348.593-1.862 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Z" />
    </mask>
    <path fill-rule="evenodd" clip-rule="evenodd"
        d="M0 12C0 5.373 5.373 0 12 0h56c6.627 0 12 5.373 12 12v56c0 6.627-5.373 12-12 12H12C5.373 80 0 74.627 0 68V12Zm35.005 33.873c0 1.058 1.104 1.703 1.955 1.142l7.67-5.052c.373-.246.6-.678.6-1.142V26.749s2.556 0 8.947-4.042c3.124-1.975 4.721-3.951 5.537-5.454.533-.981-.463-1.694-1.417-1.18-2.968 1.595-8.905 3.94-18.18 3.94-9.274 0-15.212-2.345-18.179-3.94-.954-.514-1.95.199-1.418 1.18.817 1.503 2.413 3.479 5.538 5.454 6.39 4.042 8.947 4.042 8.947 4.042v19.124Zm-8.051-6.235c2.18 0 3.949-1.83 3.949-4.085 0-2.257-1.768-4.086-3.95-4.086-2.18 0-3.949 1.83-3.949 4.086 0 2.256 1.768 4.085 3.95 4.085Zm30.276-4.085c0 2.256-1.768 4.085-3.949 4.085-2.18 0-3.949-1.83-3.949-4.085 0-2.257 1.768-4.086 3.95-4.086 2.18 0 3.948 1.83 3.948 4.086ZM26.954 58.116c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.139.79.386 1.074.514.593 1.347.593 1.861 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Zm18.429 0c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.14.79.386 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.59-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.386 1.074-.514.593-1.347.593-1.861 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.589-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.385 1.074-.514.593-1.348.593-1.862 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Z"
        fill="#fff" />
    <path
        d="m36.96 47.015 1.1 1.67-1.1-1.67Zm7.67-5.052 1.1 1.67-1.1-1.67Zm.6-15.214v-2h-2v2h2Zm8.947-4.042 1.069 1.69-1.069-1.69Zm5.537-5.454 1.758.954-1.758-.954Zm-1.417-1.18.947 1.76-.947-1.76Zm-36.359 0-.947 1.76.947-1.76Zm-1.418 1.18 1.758-.955-1.758.955Zm5.538 5.454-1.07 1.69 1.07-1.69Zm8.947 4.042h2v-2h-2v2Zm-8.982 36.37 1.511-1.31-1.511 1.31Zm1.861 0 1.512 1.309-1.512-1.31Zm16.568 0-1.511 1.309 1.511-1.31Zm1.862 0 1.511 1.309-1.511-1.31Zm-14.48 0-1.512-1.31 1.512 1.31Zm-1.862 0 1.511-1.31-1.511 1.31Zm18.43 0-1.512 1.309 1.511-1.31Zm1.86 0 1.512 1.309-1.511-1.31Zm-14.48 0 1.512 1.309-1.511-1.31Zm-1.861 0-1.511 1.309 1.511-1.31Zm18.43 0 1.51-1.31-1.51 1.31Zm1.861 0 1.511 1.309-1.511-1.31ZM12-2C4.268-2-2 4.268-2 12h4C2 6.477 6.477 2 12 2v-4Zm56 0H12v4h56v-4Zm14 14c0-7.732-6.268-14-14-14v4c5.523 0 10 4.477 10 10h4Zm0 56V12h-4v56h4ZM68 82c7.732 0 14-6.268 14-14h-4c0 5.523-4.477 10-10 10v4Zm-56 0h56v-4H12v4ZM-2 68c0 7.732 6.268 14 14 14v-4C6.477 78 2 73.523 2 68h-4Zm0-56v56h4V12h-4Zm37.86 33.345a.791.791 0 0 1 .812-.021.656.656 0 0 1 .333.549h-4c0 2.454 2.714 4.355 5.055 2.812l-2.2-3.34Zm7.669-5.052-7.669 5.052 2.2 3.34 7.67-5.052-2.201-3.34Zm-.299.528c0-.178.086-.388.299-.528l2.2 3.34a3.365 3.365 0 0 0 1.501-2.812h-4Zm0-14.072V40.82h4V26.749h-4Zm9.878-5.732c-3.097 1.959-5.197 2.895-6.464 3.34a8.845 8.845 0 0 1-1.28.362 2.42 2.42 0 0 1-.203.031H45.19l.018-.001h.02c.001 0 .003 0 .003 2s.002 2 .003 2h.054a3.196 3.196 0 0 0 .256-.017c.144-.014.327-.038.552-.08.45-.084 1.069-.238 1.875-.521 1.61-.566 3.982-1.65 7.276-3.733l-2.138-3.381Zm4.849-4.719c-.637 1.173-1.983 2.906-4.849 4.719l2.138 3.38c3.382-2.138 5.23-4.356 6.226-6.19l-3.515-1.909Zm1.287 1.536c-.106.056-.494.173-.916-.158a1.233 1.233 0 0 1-.448-.76.985.985 0 0 1 .077-.618l3.515 1.909c.308-.568.472-1.248.351-1.96a2.768 2.768 0 0 0-1.025-1.718c-1.026-.805-2.388-.788-3.449-.218l1.895 3.523Zm-19.127 4.18c9.628 0 15.889-2.438 19.127-4.18l-1.895-3.523c-2.696 1.45-8.31 3.702-17.232 3.702v4Zm-19.126-4.18c3.238 1.742 9.499 4.18 19.126 4.18v-4c-8.921 0-14.535-2.253-17.231-3.703l-1.895 3.523Zm1.287-1.536c.042.078.131.3.077.619a1.232 1.232 0 0 1-.449.759c-.422.33-.81.215-.915.158l1.895-3.523c-1.06-.57-2.424-.587-3.449.218a2.768 2.768 0 0 0-1.026 1.719c-.12.711.043 1.391.352 1.959l3.515-1.909Zm4.849 4.719c-2.867-1.813-4.212-3.546-4.849-4.719l-3.515 1.909c.995 1.834 2.844 4.052 6.226 6.19l2.138-3.38Zm7.878 5.732c0-2 .001-2 .003-2H35.047l.025.001h.001a2.42 2.42 0 0 1-.202-.03 8.852 8.852 0 0 1-1.28-.363c-1.267-.445-3.367-1.381-6.464-3.34l-2.138 3.38c3.293 2.084 5.666 3.168 7.276 3.734.806.283 1.425.437 1.875.52a6.356 6.356 0 0 0 .737.095l.071.002h.054c.001 0 .003 0 .003-2Zm2 19.124V26.749h-4v19.124h4Zm-8.102-10.32c0 1.216-.936 2.085-1.95 2.085v4c3.35 0 5.95-2.789 5.95-6.085h-4Zm-1.95-2.086c1.014 0 1.95.87 1.95 2.086h4c0-3.297-2.6-6.086-5.95-6.086v4Zm-1.949 2.086c0-1.216.936-2.086 1.95-2.086v-4c-3.35 0-5.95 2.79-5.95 6.086h4Zm1.95 2.085c-1.014 0-1.95-.87-1.95-2.085h-4c0 3.296 2.6 6.085 5.95 6.085v-4Zm26.327 4c3.35 0 5.95-2.789 5.95-6.085h-4c0 1.216-.937 2.085-1.95 2.085v4Zm-5.949-6.085c0 3.296 2.6 6.085 5.95 6.085v-4c-1.014 0-1.95-.87-1.95-2.085h-4Zm5.95-6.086c-3.35 0-5.95 2.79-5.95 6.086h4c0-1.216.936-2.086 1.95-2.086v-4Zm5.948 6.086c0-3.297-2.6-6.086-5.949-6.086v4c1.013 0 1.95.87 1.95 2.086h4ZM27.637 59.635a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4c-2.091 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.45.45 0 0 1 .086.142.277.277 0 0 1 .017.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.161 0a.776.776 0 0 1 .58-.245c.27 0 .476.124.581.245l-3.023 2.62c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62Zm-.103.236c0-.027.004-.058.017-.094a.45.45 0 0 1 .086-.142l3.023 2.62a3.644 3.644 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.683.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.103-.236h4c0-1.665-1.226-3.519-3.317-3.519v4Zm19.113-.481a.37.37 0 0 1-.102.236.774.774 0 0 1-.581.245v-4c-2.091 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.102-.236a.45.45 0 0 1 .085.142.277.277 0 0 1 .017.094h-4c0 .848.29 1.71.875 2.384l3.023-2.62Zm-1.162 0a.775.775 0 0 1 .58-.245c.271 0 .477.124.582.245l-3.023 2.62c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62Zm-.103.236c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62a3.645 3.645 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.684.481c-.29 0-.49-.14-.58-.245a.369.369 0 0 1-.104-.236h4c0-1.665-1.225-3.519-3.316-3.519v4Zm-14.48-4c-2.091 0-3.317 1.854-3.317 3.519h4a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4Zm3.316 3.519c0-1.665-1.225-3.519-3.316-3.519v4c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4Zm0 2.409v-2.409h-4v2.409h4Zm-.874 2.384a3.644 3.644 0 0 0 .874-2.384h-4c0-.027.005-.058.017-.094a.45.45 0 0 1 .086-.142l3.023 2.62Zm-4.885 0c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62a.775.775 0 0 1 .58-.245c.27 0 .476.124.581.245l-3.023 2.62Zm-.874-2.384c0 .848.29 1.71.874 2.384l3.023-2.62a.45.45 0 0 1 .086.142.277.277 0 0 1 .017.094h-4Zm0-2.409v2.409h4v-2.409h-4Zm22.43 0a.37.37 0 0 1-.103.236.774.774 0 0 1-.581.245v-4c-2.09 0-3.316 1.854-3.316 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.453.453 0 0 1 .085.142.277.277 0 0 1 .018.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.162 0a.776.776 0 0 1 .581-.245c.27 0 .476.124.58.245l-3.022 2.62c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62Zm-.103.236c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62a3.644 3.644 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.684.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4c0-1.665-1.225-3.519-3.316-3.519v4Zm-14.48-4c-2.091 0-3.316 1.854-3.316 3.519h4a.37.37 0 0 1-.104.236.774.774 0 0 1-.58.245v-4Zm3.316 3.519c0-1.665-1.225-3.519-3.316-3.519v4c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4Zm0 2.409v-2.409h-4v2.409h4Zm-.874 2.384a3.644 3.644 0 0 0 .874-2.384h-4c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62Zm-4.884 0c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62a.776.776 0 0 1 .58-.245c.271 0 .477.124.581.245l-3.022 2.62Zm-.874-2.384c0 .848.289 1.71.874 2.384l3.023-2.62a.453.453 0 0 1 .085.142.277.277 0 0 1 .017.094h-4Zm0-2.409v2.409h4v-2.409h-4Zm22.429 0a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4c-2.092 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.45.45 0 0 1 .085.142.277.277 0 0 1 .018.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.162 0a.776.776 0 0 1 .581-.245c.27 0 .476.124.58.245l-3.022 2.62c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62Zm-.102.236c0-.027.004-.058.017-.094a.455.455 0 0 1 .085-.142l3.023 2.62a3.645 3.645 0 0 0 .875-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.683.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.103-.236h4c0-1.665-1.226-3.519-3.317-3.519v4Z"
        fill="#fff" mask="url(#a)" />
</svg>"
class="header3__card-recommendation-header-photo header3__card-recommendation-header-photo_default_inverted"
/>
</div>
<div class="header3__card-recommendation-header-content">
<div class="header3__card-recommendation-header-content-tag">
Курс
</div>
<div class="header3__card-recommendation-header-sub">
<span
class="header3__card-recommendation-header-sub-chunk header3__card-recommendation-header-sub-chunk_green"
>
Скидка 20000 ₽
</span>
</div>
</div>
</div>
<h6 class="header3__card-recommendation-title">
ИИ для аналитики и работы с данными
</h6>
<div class="header3__card-recommendation-footer">
12 марта
, 2026
· 2 месяца
</div>
</a>
</div>
</div>
<div>
<p class="header3__nav-section-title ">Ответим на ваши вопросы</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item header3__nav-section-item_phone"
rel="noopener noreferrer"
href="tel:+7 499 938-92-02"
>
<svg
class="header3__phone-icon"
width="24"
height="24"
viewBox="0 0 24 24"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M14.352 2.75a7.971 7.971 0 0 1 7.041 7.032M14.352 6.293a4.426 4.426 0 0 1 3.5 3.5"
stroke="currentColor"
stroke-width="2"
stroke-linecap="round"
stroke-linejoin="round"
></path>
<path
clip-rule="evenodd"
d="M7.7 16.299C.803 9.4 1.783 6.241 2.51 5.223c.094-.164 2.396-3.611 4.865-1.589 6.126 5.045-1.63 4.332 3.514 9.477 5.146 5.144 4.431-2.611 9.477 3.514 2.022 2.469-1.425 4.771-1.588 4.864-1.018.728-4.178 1.709-11.078-5.19Z"
stroke="currentColor"
stroke-width="2"
stroke-linecap="round"
stroke-linejoin="round"
></path>
</svg>
+7 499 938-92-02
</a>
</div>
</div>
</div>
</div>
<svg
class="header3__nav-item-popup-figure"
viewBox="0 0 600 600"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M125.573 140.619C131.659 71.6017 210.245 34.9562 267.027 74.6573L553.942 275.262C610.723 314.962 603.117 401.233 540.247 430.55L222.58 578.681C159.71 607.997 88.7344 558.37 94.8204 489.355L125.573 140.619Z"
stroke="#eaeaea"
class="header3__nav-item-popup-figure-spinner"
></path>
<path
d="M148.472 246.647C133.624 191.005 184.615 140.013 240.257 154.862L519.856 229.476C575.498 244.325 594.059 313.877 553.266 354.67L348.281 559.656C307.488 600.449 237.935 581.888 223.087 526.246L148.472 246.647Z"
fill="url(#paint0_linear-info)"
></path>
<defs>
<linearGradient
id="paint0_linear-info"
x1="128.696"
y1="395.739"
x2="443.538"
y2="180.173"
gradientUnits="userSpaceOnUse"
>
<stop stop-color="#f9f9f9"></stop>
<stop offset="1" stop-color="#EBEBEB"></stop>
</linearGradient>
</defs>
</svg>
</div>
</div>
<a class="header3__nav-item header3__nav-item-b2b header3__nav-item_with-hover" href="/b2b">
Компаниям
</a>
</nav>
<div class="header3__nav header3__nav_right">
<div data-name="" class="header3__nav-item js-header3-popup-trigger js-open-modal-reg header3__button-sign-up-container" >
<button class="header3__button-sign-up">
<svg class="header3__button-sign-up-icon" width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
<path
clip-rule="evenodd"
d="M9.922 21.808c-3.814 0-7.072-.577-7.072-2.887s3.237-4.41 7.072-4.41c3.814 0 7.072 2.08 7.072 4.39 0 2.308-3.237 2.907-7.072 2.907ZM9.922 11.216A4.534 4.534 0 1 0 5.39 6.683a4.518 4.518 0 0 0 4.501 4.533h.032Z"
stroke="currentColor"
stroke-width="2"
stroke-linecap="round"
stroke-linejoin="round"
></path>
<path
d="M19.131 8.13v4.01M21.178 10.134h-4.09"
stroke="currentColor"
stroke-width="2"
stroke-linecap="round"
stroke-linejoin="round"
></path>
</svg>
Зарегистрироваться
</button>
</div>
<div data-name="" class="header3__nav-item js-header3-popup-trigger js-open-modal-login header3__button-sign-in-container" data-modal-id="new-log-reg">
<button class="header3__button-sign-in">
Войти
</button>
</div>
<div class="header3__hamburger">
<button data-name="hamburger" class="header3__hamburger-button js-header3-popup-trigger">
<svg
class="header3__hamburger-icon js-header3-popup-trigger-icon-default"
width="28"
height="22"
viewBox="0 0 28 22"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M2.15999 0.119995C1.09961 0.119995 0.23999 0.979576 0.23999 2.04C0.23999 3.10035 1.09961 3.96 2.15999 3.96H25.84C26.9004 3.96 27.76 3.10035 27.76 2.04C27.76 0.979576 26.9004 0.119995 25.84 0.119995H2.15999ZM2.15999 9.08C1.09961 9.08 0.23999 9.93957 0.23999 11C0.23999 12.0604 1.09961 12.92 2.15999 12.92H25.84C26.9004 12.92 27.76 12.0604 27.76 11C27.76 9.93957 26.9004 9.08 25.84 9.08H2.15999ZM2.15999 18.04C1.09961 18.04 0.23999 18.8996 0.23999 19.96C0.23999 21.0204 1.09961 21.88 2.15999 21.88H25.84C26.9004 21.88 27.76 21.0204 27.76 19.96C27.76 18.8996 26.9004 18.04 25.84 18.04H2.15999Z"
fill="currentColor"
></path>
</svg>
<svg
width="24"
height="24"
viewBox="0 0 24 24"
fill="none"
xmlns="http://www.w3.org/2000/svg"
class="header3__hamburger-icon-close js-header3-popup-trigger-icon-close"
>
<path
fill-rule="evenodd"
clip-rule="evenodd"
d="M12 0.75C11.3096 0.75 10.75 1.30964 10.75 2V10.75L2 10.75C1.30964 10.75 0.75 11.3096 0.75 12C0.75 12.6904 1.30964 13.25 2 13.25L10.75 13.25L10.75 22C10.75 22.6904 11.3096 23.25 12 23.25C12.6904 23.25 13.25 22.6904 13.25 22L13.25 13.25L22 13.25C22.6904 13.25 23.25 12.6904 23.25 12C23.25 11.3096 22.6904 10.75 22 10.75L13.25 10.75V2C13.25 1.30964 12.6904 0.75 12 0.75Z"
fill="currentColor"
></path>
</svg>
</button>
<div
data-name="hamburger"
data-only-click="true"
class="header3__nav-item-popup-wrapper js-header3-popup"
style="display: none;"
>
<div
class="header3__nav-item-popup-container js-header3-popup-container"
>
<svg
class="header3__nav-item-popup-figure"
viewBox="0 0 600 600"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M125.573 140.619C131.659 71.6017 210.245 34.9562 267.027 74.6573L553.942 275.262C610.723 314.962 603.117 401.233 540.247 430.55L222.58 578.681C159.71 607.997 88.7344 558.37 94.8204 489.355L125.573 140.619Z"
stroke="#eaeaea"
class="header3__nav-item-popup-figure-spinner"
></path>
<path
d="M148.472 246.647C133.624 191.005 184.615 140.013 240.257 154.862L519.856 229.476C575.498 244.325 594.059 313.877 553.266 354.67L348.281 559.656C307.488 600.449 237.935 581.888 223.087 526.246L148.472 246.647Z"
fill="url(#paint0_linear-hamburger)"
></path>
<defs>
<linearGradient
id="paint0_linear-hamburger"
x1="128.696"
y1="395.739"
x2="443.538"
y2="180.173"
gradientUnits="userSpaceOnUse"
>
<stop stop-color="#f9f9f9"></stop>
<stop offset="1" stop-color="#EBEBEB"></stop>
</linearGradient>
</defs>
</svg>
<div class="header3__nav-item-popup-content header3__hamburger-tabs">
<ul>
<li
class="header3__hamburger-tabs-tab header3__hamburger-tabs-tab_active"
>
Обучение
</li>
<li class="header3__hamburger-tabs-tab">
Информация
</li>
<li data-url="/b2b" class="header3__hamburger-tabs-tab header3__hamburger-tabs-tab_link">
Компаниям
</li>
</ul>
<div class="header3__hamburger-tabs-tab-content-wrapper">
<div
class="header3__hamburger-tabs-tab-content header3__hamburger-tabs-tab-learning header3__hamburger-tabs-tab-content_active"
>
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title ">Направления</p>
<div class="header3__nav-section-items header3__nav-section-items_learning header3__nav-section-items_learning_rows-8">
<a
class="header3__nav-section-item"
href="/categories/programming/"
>
Программирование (117)
</a>
<a
class="header3__nav-section-item"
href="/categories/architecture/"
>
Архитектура (17)
</a>
<a
class="header3__nav-section-item"
href="/categories/data-science/"
>
Data Science (27)
</a>
<a
class="header3__nav-section-item"
href="/categories/operations/"
>
Инфраструктура (58)
</a>
<a
class="header3__nav-section-item"
href="/categories/gamedev/"
>
GameDev (10)
</a>
<a
class="header3__nav-section-item"
href="/categories/information-security-courses/"
>
Безопасность (15)
</a>
<a
class="header3__nav-section-item"
href="/categories/marketing-business/"
>
Управление (46)
</a>
<a
class="header3__nav-section-item"
href="/categories/analytics/"
>
Аналитика и анализ (25)
</a>
<a
class="header3__nav-section-item"
href="/categories/business-product/"
>
Бизнес и продукт в IT (26)
</a>
<a
class="header3__nav-section-item"
href="/categories/import-substitution/"
>
Импортозамещение (15)
</a>
<a
class="header3__nav-section-item"
href="/categories/testing/"
>
Тестирование (12)
</a>
<a
class="header3__nav-section-item"
href="/categories/neural_networks/"
>
Нейросети (9)
</a>
<a
class="header3__nav-section-item"
href="/categories/it-bez-programmirovanija/"
>
IT без программирования (19)
</a>
<a
class="header3__nav-section-item"
href="/categories/corporate/"
>
Корпоративные курсы (27)
</a>
</div>
</div>
</div>
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title ">События</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item"
href="/lessons/calendar/2026/"
>
Календарь запуска курсов
</a>
<a
class="header3__nav-section-item"
href="/events/near/"
>
Календарь мероприятий
</a>
</div>
</div>
<div>
<p class="header3__nav-section-title ">Другое</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item"
href="/categories/spec/"
>
Специализации (13)
</a>
<a
class="header3__nav-section-item"
href="/categories/online/"
>
Подготовительные курсы (14)
</a>
<a
class="header3__nav-section-item header3__nav-section-item_bold"
href="/subscription"
>
Подписка на курсы
</a>
<a
class="header3__nav-section-item"
href="/tests"
>
Проверьте свои знания
</a>
</div>
</div>
</div>
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title header3__nav-section-items-recommendation-title">OTUS рекомендует</p>
<div class="header3__nav-section-items header3__nav-section-items_not-items header3__nav-section-items-recommendation">
<a
href="/lessons/ai-dlya-analitiki-i-raboty-s-dannymi/"
class="header3__card-recommendation"
>
<div
class="header3__card-recommendation-background"
style="background: linear-gradient( 90deg,#0A4489, #00316B);"
></div>
<div class="header3__card-recommendation-header">
<div class="header3__card-recommendation-header-photo-wrapper" style="background: linear-gradient( 90deg,#0A4489, #00316B);">
<img
src="data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iODAiIGhlaWdodD0iODAiIHZpZXdCb3g9IjAgMCA4MCA4MCIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KICAgIDxwYXRoCiAgICAgICAgZD0iTTM1LjAwNDcgNDUuODcyNUMzNS4wMDQ3IDQ2LjkzMDYgMzYuMTA5IDQ3LjU3NTcgMzYuOTYwMyA0Ny4wMTQ5TDQ0LjYyOTIgNDEuOTYzMUM0NS4wMDI5IDQxLjcxNjkgNDUuMjI5OSA0MS4yODUyIDQ1LjIyOTkgNDAuODIwN1YyNi43NDg5QzQ1LjIyOTkgMjYuNzQ4OSA0Ny43ODYyIDI2Ljc0ODkgNTQuMTc3IDIyLjcwNzRDNTcuMzAxMiAyMC43MzE4IDU4Ljg5OCAxOC43NTYxIDU5LjcxNDMgMTcuMjUyNUM2MC4yNDY1IDE2LjI3MjEgNTkuMjUxMSAxNS41NTg4IDU4LjI5NjUgMTYuMDcyM0M1NS4zMjkyIDE3LjY2ODQgNDkuMzkxOCAyMC4wMTMyIDQwLjExNzMgMjAuMDEzMkMzMC44NDI4IDIwLjAxMzIgMjQuOTA1NCAxNy42Njg0IDIxLjkzODEgMTYuMDcyM0MyMC45ODM1IDE1LjU1ODggMTkuOTg4MSAxNi4yNzIxIDIwLjUyMDMgMTcuMjUyNUMyMS4zMzY2IDE4Ljc1NjEgMjIuOTMzNCAyMC43MzE4IDI2LjA1NzcgMjIuNzA3NEMzMi40NDg0IDI2Ljc0ODkgMzUuMDA0NyAyNi43NDg5IDM1LjAwNDcgMjYuNzQ4OVY0NS44NzI1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxlbGxpcHNlIGN4PSIyNi45NTM1IiBjeT0iMzUuNTUyNiIgcng9IjMuOTQ5MTQiIHJ5PSI0LjA4NTMyIiBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxlbGxpcHNlIGN4PSI1My4yODEyIiBjeT0iMzUuNTUyNiIgcng9IjMuOTQ5MTQiIHJ5PSI0LjA4NTMyIiBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTI1LjYzNzIgNTkuNjM1QzI1LjYzNzIgNTguNzk2MSAyNi4yMjY2IDU4LjExNiAyNi45NTM2IDU4LjExNkMyNy42ODA2IDU4LjExNiAyOC4yNyA1OC43OTYxIDI4LjI3IDU5LjYzNVY2Mi4wNDRDMjguMjcgNjIuNDQ2OSAyOC4xMzEzIDYyLjgzMzMgMjcuODg0NCA2My4xMTgxQzI3LjM3MDMgNjMuNzExNCAyNi41MzY4IDYzLjcxMTQgMjYuMDIyOCA2My4xMTgxQzI1Ljc3NTkgNjIuODMzMyAyNS42MzcyIDYyLjQ0NjkgMjUuNjM3MiA2Mi4wNDRWNTkuNjM1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTQ0LjA2NjUgNTkuNjM1QzQ0LjA2NjUgNTguNzk2MSA0NC42NTU5IDU4LjExNiA0NS4zODI5IDU4LjExNkM0Ni4xMDk5IDU4LjExNiA0Ni42OTkzIDU4Ljc5NjEgNDYuNjk5MyA1OS42MzVWNjIuMDQ0QzQ2LjY5OTMgNjIuNDQ2OSA0Ni41NjA2IDYyLjgzMzMgNDYuMzEzOCA2My4xMTgxQzQ1Ljc5OTcgNjMuNzExNCA0NC45NjYyIDYzLjcxMTQgNDQuNDUyMSA2My4xMTgxQzQ0LjIwNTIgNjIuODMzMyA0NC4wNjY1IDYyLjQ0NjkgNDQuMDY2NSA2Mi4wNDRWNTkuNjM1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTI5LjU4NjQgNTkuNjM1QzI5LjU4NjQgNTguNzk2MSAzMC4xNzU3IDU4LjExNiAzMC45MDI3IDU4LjExNkMzMS42Mjk4IDU4LjExNiAzMi4yMTkxIDU4Ljc5NjEgMzIuMjE5MSA1OS42MzVWNjIuMDQ0QzMyLjIxOTEgNjIuNDQ2OSAzMi4wODA0IDYyLjgzMzMgMzEuODMzNiA2My4xMTgxQzMxLjMxOTUgNjMuNzExNCAzMC40ODYgNjMuNzExNCAyOS45NzE5IDYzLjExODFDMjkuNzI1IDYyLjgzMzMgMjkuNTg2NCA2Mi40NDY5IDI5LjU4NjQgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik00OC4wMTU3IDU5LjYzNUM0OC4wMTU3IDU4Ljc5NjEgNDguNjA1MSA1OC4xMTYgNDkuMzMyMSA1OC4xMTZDNTAuMDU5MSA1OC4xMTYgNTAuNjQ4NSA1OC43OTYxIDUwLjY0ODUgNTkuNjM1VjYyLjA0NEM1MC42NDg1IDYyLjQ0NjkgNTAuNTA5OCA2Mi44MzMzIDUwLjI2MjkgNjMuMTE4MUM0OS43NDg4IDYzLjcxMTQgNDguOTE1MyA2My43MTE0IDQ4LjQwMTIgNjMuMTE4MUM0OC4xNTQ0IDYyLjgzMzMgNDguMDE1NyA2Mi40NDY5IDQ4LjAxNTcgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik0zMy41MzU1IDU5LjYzNUMzMy41MzU1IDU4Ljc5NjEgMzQuMTI0OSA1OC4xMTYgMzQuODUxOSA1OC4xMTZDMzUuNTc4OSA1OC4xMTYgMzYuMTY4MyA1OC43OTYxIDM2LjE2ODMgNTkuNjM1VjYyLjA0NEMzNi4xNjgzIDYyLjQ0NjkgMzYuMDI5NiA2Mi44MzMzIDM1Ljc4MjcgNjMuMTE4MUMzNS4yNjg2IDYzLjcxMTQgMzQuNDM1MSA2My43MTE0IDMzLjkyMTEgNjMuMTE4MUMzMy42NzQyIDYyLjgzMzMgMzMuNTM1NSA2Mi40NDY5IDMzLjUzNTUgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik01MS45NjQ4IDU5LjYzNUM1MS45NjQ4IDU4Ljc5NjEgNTIuNTU0MiA1OC4xMTYgNTMuMjgxMiA1OC4xMTZDNTQuMDA4MiA1OC4xMTYgNTQuNTk3NiA1OC43OTYxIDU0LjU5NzYgNTkuNjM1VjYyLjA0NEM1NC41OTc2IDYyLjQ0NjkgNTQuNDU4OSA2Mi44MzMzIDU0LjIxMiA2My4xMTgxQzUzLjY5OCA2My43MTE0IDUyLjg2NDUgNjMuNzExNCA1Mi4zNTA0IDYzLjExODFDNTIuMTAzNSA2Mi44MzMzIDUxLjk2NDggNjIuNDQ2OSA1MS45NjQ4IDYyLjA0NFY1OS42MzVaIgogICAgICAgIGZpbGw9IndoaXRlIiAvPgo8L3N2Zz4="
class="header3__card-recommendation-header-photo header3__card-recommendation-header-photo_default"
/>
<img
src="data:image/svg+xml;base64,<svg width="80" height="80" viewBox="0 0 80 80" fill="none" xmlns="http://www.w3.org/2000/svg">
    <mask id="a" fill="#fff">
        <path fill-rule="evenodd" clip-rule="evenodd"
            d="M0 12C0 5.373 5.373 0 12 0h56c6.627 0 12 5.373 12 12v56c0 6.627-5.373 12-12 12H12C5.373 80 0 74.627 0 68V12Zm35.005 33.873c0 1.058 1.104 1.703 1.955 1.142l7.67-5.052c.373-.246.6-.678.6-1.142V26.749s2.556 0 8.947-4.042c3.124-1.975 4.721-3.951 5.537-5.454.533-.981-.463-1.694-1.417-1.18-2.968 1.595-8.905 3.94-18.18 3.94-9.274 0-15.212-2.345-18.179-3.94-.954-.514-1.95.199-1.418 1.18.817 1.503 2.413 3.479 5.538 5.454 6.39 4.042 8.947 4.042 8.947 4.042v19.124Zm-8.051-6.235c2.18 0 3.949-1.83 3.949-4.085 0-2.257-1.768-4.086-3.95-4.086-2.18 0-3.949 1.83-3.949 4.086 0 2.256 1.768 4.085 3.95 4.085Zm30.276-4.085c0 2.256-1.768 4.085-3.949 4.085-2.18 0-3.949-1.83-3.949-4.085 0-2.257 1.768-4.086 3.95-4.086 2.18 0 3.948 1.83 3.948 4.086ZM26.954 58.116c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.139.79.386 1.074.514.593 1.347.593 1.861 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Zm18.429 0c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.14.79.386 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.59-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.386 1.074-.514.593-1.347.593-1.861 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.589-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.385 1.074-.514.593-1.348.593-1.862 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Z" />
    </mask>
    <path fill-rule="evenodd" clip-rule="evenodd"
        d="M0 12C0 5.373 5.373 0 12 0h56c6.627 0 12 5.373 12 12v56c0 6.627-5.373 12-12 12H12C5.373 80 0 74.627 0 68V12Zm35.005 33.873c0 1.058 1.104 1.703 1.955 1.142l7.67-5.052c.373-.246.6-.678.6-1.142V26.749s2.556 0 8.947-4.042c3.124-1.975 4.721-3.951 5.537-5.454.533-.981-.463-1.694-1.417-1.18-2.968 1.595-8.905 3.94-18.18 3.94-9.274 0-15.212-2.345-18.179-3.94-.954-.514-1.95.199-1.418 1.18.817 1.503 2.413 3.479 5.538 5.454 6.39 4.042 8.947 4.042 8.947 4.042v19.124Zm-8.051-6.235c2.18 0 3.949-1.83 3.949-4.085 0-2.257-1.768-4.086-3.95-4.086-2.18 0-3.949 1.83-3.949 4.086 0 2.256 1.768 4.085 3.95 4.085Zm30.276-4.085c0 2.256-1.768 4.085-3.949 4.085-2.18 0-3.949-1.83-3.949-4.085 0-2.257 1.768-4.086 3.95-4.086 2.18 0 3.948 1.83 3.948 4.086ZM26.954 58.116c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.139.79.386 1.074.514.593 1.347.593 1.861 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Zm18.429 0c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.14.79.386 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.59-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.386 1.074-.514.593-1.347.593-1.861 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.589-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.385 1.074-.514.593-1.348.593-1.862 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Z"
        fill="#fff" />
    <path
        d="m36.96 47.015 1.1 1.67-1.1-1.67Zm7.67-5.052 1.1 1.67-1.1-1.67Zm.6-15.214v-2h-2v2h2Zm8.947-4.042 1.069 1.69-1.069-1.69Zm5.537-5.454 1.758.954-1.758-.954Zm-1.417-1.18.947 1.76-.947-1.76Zm-36.359 0-.947 1.76.947-1.76Zm-1.418 1.18 1.758-.955-1.758.955Zm5.538 5.454-1.07 1.69 1.07-1.69Zm8.947 4.042h2v-2h-2v2Zm-8.982 36.37 1.511-1.31-1.511 1.31Zm1.861 0 1.512 1.309-1.512-1.31Zm16.568 0-1.511 1.309 1.511-1.31Zm1.862 0 1.511 1.309-1.511-1.31Zm-14.48 0-1.512-1.31 1.512 1.31Zm-1.862 0 1.511-1.31-1.511 1.31Zm18.43 0-1.512 1.309 1.511-1.31Zm1.86 0 1.512 1.309-1.511-1.31Zm-14.48 0 1.512 1.309-1.511-1.31Zm-1.861 0-1.511 1.309 1.511-1.31Zm18.43 0 1.51-1.31-1.51 1.31Zm1.861 0 1.511 1.309-1.511-1.31ZM12-2C4.268-2-2 4.268-2 12h4C2 6.477 6.477 2 12 2v-4Zm56 0H12v4h56v-4Zm14 14c0-7.732-6.268-14-14-14v4c5.523 0 10 4.477 10 10h4Zm0 56V12h-4v56h4ZM68 82c7.732 0 14-6.268 14-14h-4c0 5.523-4.477 10-10 10v4Zm-56 0h56v-4H12v4ZM-2 68c0 7.732 6.268 14 14 14v-4C6.477 78 2 73.523 2 68h-4Zm0-56v56h4V12h-4Zm37.86 33.345a.791.791 0 0 1 .812-.021.656.656 0 0 1 .333.549h-4c0 2.454 2.714 4.355 5.055 2.812l-2.2-3.34Zm7.669-5.052-7.669 5.052 2.2 3.34 7.67-5.052-2.201-3.34Zm-.299.528c0-.178.086-.388.299-.528l2.2 3.34a3.365 3.365 0 0 0 1.501-2.812h-4Zm0-14.072V40.82h4V26.749h-4Zm9.878-5.732c-3.097 1.959-5.197 2.895-6.464 3.34a8.845 8.845 0 0 1-1.28.362 2.42 2.42 0 0 1-.203.031H45.19l.018-.001h.02c.001 0 .003 0 .003 2s.002 2 .003 2h.054a3.196 3.196 0 0 0 .256-.017c.144-.014.327-.038.552-.08.45-.084 1.069-.238 1.875-.521 1.61-.566 3.982-1.65 7.276-3.733l-2.138-3.381Zm4.849-4.719c-.637 1.173-1.983 2.906-4.849 4.719l2.138 3.38c3.382-2.138 5.23-4.356 6.226-6.19l-3.515-1.909Zm1.287 1.536c-.106.056-.494.173-.916-.158a1.233 1.233 0 0 1-.448-.76.985.985 0 0 1 .077-.618l3.515 1.909c.308-.568.472-1.248.351-1.96a2.768 2.768 0 0 0-1.025-1.718c-1.026-.805-2.388-.788-3.449-.218l1.895 3.523Zm-19.127 4.18c9.628 0 15.889-2.438 19.127-4.18l-1.895-3.523c-2.696 1.45-8.31 3.702-17.232 3.702v4Zm-19.126-4.18c3.238 1.742 9.499 4.18 19.126 4.18v-4c-8.921 0-14.535-2.253-17.231-3.703l-1.895 3.523Zm1.287-1.536c.042.078.131.3.077.619a1.232 1.232 0 0 1-.449.759c-.422.33-.81.215-.915.158l1.895-3.523c-1.06-.57-2.424-.587-3.449.218a2.768 2.768 0 0 0-1.026 1.719c-.12.711.043 1.391.352 1.959l3.515-1.909Zm4.849 4.719c-2.867-1.813-4.212-3.546-4.849-4.719l-3.515 1.909c.995 1.834 2.844 4.052 6.226 6.19l2.138-3.38Zm7.878 5.732c0-2 .001-2 .003-2H35.047l.025.001h.001a2.42 2.42 0 0 1-.202-.03 8.852 8.852 0 0 1-1.28-.363c-1.267-.445-3.367-1.381-6.464-3.34l-2.138 3.38c3.293 2.084 5.666 3.168 7.276 3.734.806.283 1.425.437 1.875.52a6.356 6.356 0 0 0 .737.095l.071.002h.054c.001 0 .003 0 .003-2Zm2 19.124V26.749h-4v19.124h4Zm-8.102-10.32c0 1.216-.936 2.085-1.95 2.085v4c3.35 0 5.95-2.789 5.95-6.085h-4Zm-1.95-2.086c1.014 0 1.95.87 1.95 2.086h4c0-3.297-2.6-6.086-5.95-6.086v4Zm-1.949 2.086c0-1.216.936-2.086 1.95-2.086v-4c-3.35 0-5.95 2.79-5.95 6.086h4Zm1.95 2.085c-1.014 0-1.95-.87-1.95-2.085h-4c0 3.296 2.6 6.085 5.95 6.085v-4Zm26.327 4c3.35 0 5.95-2.789 5.95-6.085h-4c0 1.216-.937 2.085-1.95 2.085v4Zm-5.949-6.085c0 3.296 2.6 6.085 5.95 6.085v-4c-1.014 0-1.95-.87-1.95-2.085h-4Zm5.95-6.086c-3.35 0-5.95 2.79-5.95 6.086h4c0-1.216.936-2.086 1.95-2.086v-4Zm5.948 6.086c0-3.297-2.6-6.086-5.949-6.086v4c1.013 0 1.95.87 1.95 2.086h4ZM27.637 59.635a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4c-2.091 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.45.45 0 0 1 .086.142.277.277 0 0 1 .017.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.161 0a.776.776 0 0 1 .58-.245c.27 0 .476.124.581.245l-3.023 2.62c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62Zm-.103.236c0-.027.004-.058.017-.094a.45.45 0 0 1 .086-.142l3.023 2.62a3.644 3.644 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.683.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.103-.236h4c0-1.665-1.226-3.519-3.317-3.519v4Zm19.113-.481a.37.37 0 0 1-.102.236.774.774 0 0 1-.581.245v-4c-2.091 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.102-.236a.45.45 0 0 1 .085.142.277.277 0 0 1 .017.094h-4c0 .848.29 1.71.875 2.384l3.023-2.62Zm-1.162 0a.775.775 0 0 1 .58-.245c.271 0 .477.124.582.245l-3.023 2.62c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62Zm-.103.236c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62a3.645 3.645 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.684.481c-.29 0-.49-.14-.58-.245a.369.369 0 0 1-.104-.236h4c0-1.665-1.225-3.519-3.316-3.519v4Zm-14.48-4c-2.091 0-3.317 1.854-3.317 3.519h4a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4Zm3.316 3.519c0-1.665-1.225-3.519-3.316-3.519v4c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4Zm0 2.409v-2.409h-4v2.409h4Zm-.874 2.384a3.644 3.644 0 0 0 .874-2.384h-4c0-.027.005-.058.017-.094a.45.45 0 0 1 .086-.142l3.023 2.62Zm-4.885 0c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62a.775.775 0 0 1 .58-.245c.27 0 .476.124.581.245l-3.023 2.62Zm-.874-2.384c0 .848.29 1.71.874 2.384l3.023-2.62a.45.45 0 0 1 .086.142.277.277 0 0 1 .017.094h-4Zm0-2.409v2.409h4v-2.409h-4Zm22.43 0a.37.37 0 0 1-.103.236.774.774 0 0 1-.581.245v-4c-2.09 0-3.316 1.854-3.316 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.453.453 0 0 1 .085.142.277.277 0 0 1 .018.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.162 0a.776.776 0 0 1 .581-.245c.27 0 .476.124.58.245l-3.022 2.62c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62Zm-.103.236c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62a3.644 3.644 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.684.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4c0-1.665-1.225-3.519-3.316-3.519v4Zm-14.48-4c-2.091 0-3.316 1.854-3.316 3.519h4a.37.37 0 0 1-.104.236.774.774 0 0 1-.58.245v-4Zm3.316 3.519c0-1.665-1.225-3.519-3.316-3.519v4c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4Zm0 2.409v-2.409h-4v2.409h4Zm-.874 2.384a3.644 3.644 0 0 0 .874-2.384h-4c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62Zm-4.884 0c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62a.776.776 0 0 1 .58-.245c.271 0 .477.124.581.245l-3.022 2.62Zm-.874-2.384c0 .848.289 1.71.874 2.384l3.023-2.62a.453.453 0 0 1 .085.142.277.277 0 0 1 .017.094h-4Zm0-2.409v2.409h4v-2.409h-4Zm22.429 0a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4c-2.092 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.45.45 0 0 1 .085.142.277.277 0 0 1 .018.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.162 0a.776.776 0 0 1 .581-.245c.27 0 .476.124.58.245l-3.022 2.62c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62Zm-.102.236c0-.027.004-.058.017-.094a.455.455 0 0 1 .085-.142l3.023 2.62a3.645 3.645 0 0 0 .875-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.683.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.103-.236h4c0-1.665-1.226-3.519-3.317-3.519v4Z"
        fill="#fff" mask="url(#a)" />
</svg>"
class="header3__card-recommendation-header-photo header3__card-recommendation-header-photo_default_inverted"
/>
</div>
<div class="header3__card-recommendation-header-content">
<div class="header3__card-recommendation-header-content-tag">
Курс
</div>
<div class="header3__card-recommendation-header-sub">
<span
class="header3__card-recommendation-header-sub-chunk header3__card-recommendation-header-sub-chunk_green"
>
Скидка 20000 ₽
</span>
</div>
</div>
</div>
<h6 class="header3__card-recommendation-title">
ИИ для аналитики и работы с данными
</h6>
<div class="header3__card-recommendation-footer">
12 марта
, 2026
· 2 месяца
</div>
</a>
</div>
</div>
</div>
</div>
<div class="header3__hamburger-tabs-tab-content">
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title ">OTUS</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item"
href="/about"
>
О компании
</a>
<a
class="header3__nav-section-item"
href="/smi/"
>
СМИ о нас
</a>
<a
class="header3__nav-section-item js-stats"
href="/journal/"
target="_blank"
rel="noreferrer nofollow"
data-event="header;click_otus_journal"
data-goal="click_otus_journal"
>
OTUS Журнал
</a>
<a
class="header3__nav-section-item"
href="https://direct.otus.ru/"
target="_blank"
rel="noreferrer nofollow"
>
OTUS Директ
</a>
<a
class="header3__nav-section-item"
href="/legal/common/"
>
Сведения об образовательной организации
</a>
<a
class="header3__nav-section-item"
href="/contacts/"
>
Контактная информация
</a>
</div>
</div>
</div>
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title ">Студентам</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item"
href="/reviews"
>
Отзывы
</a>
<a
class="header3__nav-section-item"
target="_blank"
href="https://landing.otus.ru/about-otus"
>
Как выбрать курс
</a>
<a
class="header3__nav-section-item"
target="_blank"
href="https://landing.otus.ru/gallery"
>
Истории выпускников
</a>
<a
class="header3__nav-section-item"
href="/instructors/"
>
Наши преподаватели
</a>
<a
class="header3__nav-section-item"
href="/employers/all/"
>
Наши партнеры
</a>
<a
class="header3__nav-section-item"
href="/about/loyalty/"
>
Программа лояльности
</a>
<a
class="header3__nav-section-item"
href="/faq/"
>
Вопросы и ответы
</a>
</div>
</div>
</div>
<div class="header3__nav-column">
<div>
<p class="header3__nav-section-title ">Преподавателям</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item"
href="/teach/"
>
Стать преподавателем
</a>
<a
class="header3__nav-section-item"
href="/instructors/"
>
Наши преподаватели
</a>
<a
class="header3__nav-section-item"
href="/nest/dlja-prepodavatelej/"
>
База знаний
</a>
</div>
</div>
</div>
<div class="header3__nav-column header3__nav-column_space-between">
<div>
<p class="header3__nav-section-title header3__nav-section-items-recommendation-title">OTUS рекомендует</p>
<div class="header3__nav-section-items header3__nav-section-items_not-items header3__nav-section-items-recommendation">
<a
href="/lessons/ai-dlya-analitiki-i-raboty-s-dannymi/"
class="header3__card-recommendation"
>
<div
class="header3__card-recommendation-background"
style="background: linear-gradient( 90deg,#0A4489, #00316B);"
></div>
<div class="header3__card-recommendation-header">
<div class="header3__card-recommendation-header-photo-wrapper" style="background: linear-gradient( 90deg,#0A4489, #00316B);">
<img
src="data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iODAiIGhlaWdodD0iODAiIHZpZXdCb3g9IjAgMCA4MCA4MCIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KICAgIDxwYXRoCiAgICAgICAgZD0iTTM1LjAwNDcgNDUuODcyNUMzNS4wMDQ3IDQ2LjkzMDYgMzYuMTA5IDQ3LjU3NTcgMzYuOTYwMyA0Ny4wMTQ5TDQ0LjYyOTIgNDEuOTYzMUM0NS4wMDI5IDQxLjcxNjkgNDUuMjI5OSA0MS4yODUyIDQ1LjIyOTkgNDAuODIwN1YyNi43NDg5QzQ1LjIyOTkgMjYuNzQ4OSA0Ny43ODYyIDI2Ljc0ODkgNTQuMTc3IDIyLjcwNzRDNTcuMzAxMiAyMC43MzE4IDU4Ljg5OCAxOC43NTYxIDU5LjcxNDMgMTcuMjUyNUM2MC4yNDY1IDE2LjI3MjEgNTkuMjUxMSAxNS41NTg4IDU4LjI5NjUgMTYuMDcyM0M1NS4zMjkyIDE3LjY2ODQgNDkuMzkxOCAyMC4wMTMyIDQwLjExNzMgMjAuMDEzMkMzMC44NDI4IDIwLjAxMzIgMjQuOTA1NCAxNy42Njg0IDIxLjkzODEgMTYuMDcyM0MyMC45ODM1IDE1LjU1ODggMTkuOTg4MSAxNi4yNzIxIDIwLjUyMDMgMTcuMjUyNUMyMS4zMzY2IDE4Ljc1NjEgMjIuOTMzNCAyMC43MzE4IDI2LjA1NzcgMjIuNzA3NEMzMi40NDg0IDI2Ljc0ODkgMzUuMDA0NyAyNi43NDg5IDM1LjAwNDcgMjYuNzQ4OVY0NS44NzI1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxlbGxpcHNlIGN4PSIyNi45NTM1IiBjeT0iMzUuNTUyNiIgcng9IjMuOTQ5MTQiIHJ5PSI0LjA4NTMyIiBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxlbGxpcHNlIGN4PSI1My4yODEyIiBjeT0iMzUuNTUyNiIgcng9IjMuOTQ5MTQiIHJ5PSI0LjA4NTMyIiBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTI1LjYzNzIgNTkuNjM1QzI1LjYzNzIgNTguNzk2MSAyNi4yMjY2IDU4LjExNiAyNi45NTM2IDU4LjExNkMyNy42ODA2IDU4LjExNiAyOC4yNyA1OC43OTYxIDI4LjI3IDU5LjYzNVY2Mi4wNDRDMjguMjcgNjIuNDQ2OSAyOC4xMzEzIDYyLjgzMzMgMjcuODg0NCA2My4xMTgxQzI3LjM3MDMgNjMuNzExNCAyNi41MzY4IDYzLjcxMTQgMjYuMDIyOCA2My4xMTgxQzI1Ljc3NTkgNjIuODMzMyAyNS42MzcyIDYyLjQ0NjkgMjUuNjM3MiA2Mi4wNDRWNTkuNjM1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTQ0LjA2NjUgNTkuNjM1QzQ0LjA2NjUgNTguNzk2MSA0NC42NTU5IDU4LjExNiA0NS4zODI5IDU4LjExNkM0Ni4xMDk5IDU4LjExNiA0Ni42OTkzIDU4Ljc5NjEgNDYuNjk5MyA1OS42MzVWNjIuMDQ0QzQ2LjY5OTMgNjIuNDQ2OSA0Ni41NjA2IDYyLjgzMzMgNDYuMzEzOCA2My4xMTgxQzQ1Ljc5OTcgNjMuNzExNCA0NC45NjYyIDYzLjcxMTQgNDQuNDUyMSA2My4xMTgxQzQ0LjIwNTIgNjIuODMzMyA0NC4wNjY1IDYyLjQ0NjkgNDQuMDY2NSA2Mi4wNDRWNTkuNjM1WiIKICAgICAgICBmaWxsPSJ3aGl0ZSIgLz4KICAgIDxwYXRoCiAgICAgICAgZD0iTTI5LjU4NjQgNTkuNjM1QzI5LjU4NjQgNTguNzk2MSAzMC4xNzU3IDU4LjExNiAzMC45MDI3IDU4LjExNkMzMS42Mjk4IDU4LjExNiAzMi4yMTkxIDU4Ljc5NjEgMzIuMjE5MSA1OS42MzVWNjIuMDQ0QzMyLjIxOTEgNjIuNDQ2OSAzMi4wODA0IDYyLjgzMzMgMzEuODMzNiA2My4xMTgxQzMxLjMxOTUgNjMuNzExNCAzMC40ODYgNjMuNzExNCAyOS45NzE5IDYzLjExODFDMjkuNzI1IDYyLjgzMzMgMjkuNTg2NCA2Mi40NDY5IDI5LjU4NjQgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik00OC4wMTU3IDU5LjYzNUM0OC4wMTU3IDU4Ljc5NjEgNDguNjA1MSA1OC4xMTYgNDkuMzMyMSA1OC4xMTZDNTAuMDU5MSA1OC4xMTYgNTAuNjQ4NSA1OC43OTYxIDUwLjY0ODUgNTkuNjM1VjYyLjA0NEM1MC42NDg1IDYyLjQ0NjkgNTAuNTA5OCA2Mi44MzMzIDUwLjI2MjkgNjMuMTE4MUM0OS43NDg4IDYzLjcxMTQgNDguOTE1MyA2My43MTE0IDQ4LjQwMTIgNjMuMTE4MUM0OC4xNTQ0IDYyLjgzMzMgNDguMDE1NyA2Mi40NDY5IDQ4LjAxNTcgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik0zMy41MzU1IDU5LjYzNUMzMy41MzU1IDU4Ljc5NjEgMzQuMTI0OSA1OC4xMTYgMzQuODUxOSA1OC4xMTZDMzUuNTc4OSA1OC4xMTYgMzYuMTY4MyA1OC43OTYxIDM2LjE2ODMgNTkuNjM1VjYyLjA0NEMzNi4xNjgzIDYyLjQ0NjkgMzYuMDI5NiA2Mi44MzMzIDM1Ljc4MjcgNjMuMTE4MUMzNS4yNjg2IDYzLjcxMTQgMzQuNDM1MSA2My43MTE0IDMzLjkyMTEgNjMuMTE4MUMzMy42NzQyIDYyLjgzMzMgMzMuNTM1NSA2Mi40NDY5IDMzLjUzNTUgNjIuMDQ0VjU5LjYzNVoiCiAgICAgICAgZmlsbD0id2hpdGUiIC8+CiAgICA8cGF0aAogICAgICAgIGQ9Ik01MS45NjQ4IDU5LjYzNUM1MS45NjQ4IDU4Ljc5NjEgNTIuNTU0MiA1OC4xMTYgNTMuMjgxMiA1OC4xMTZDNTQuMDA4MiA1OC4xMTYgNTQuNTk3NiA1OC43OTYxIDU0LjU5NzYgNTkuNjM1VjYyLjA0NEM1NC41OTc2IDYyLjQ0NjkgNTQuNDU4OSA2Mi44MzMzIDU0LjIxMiA2My4xMTgxQzUzLjY5OCA2My43MTE0IDUyLjg2NDUgNjMuNzExNCA1Mi4zNTA0IDYzLjExODFDNTIuMTAzNSA2Mi44MzMzIDUxLjk2NDggNjIuNDQ2OSA1MS45NjQ4IDYyLjA0NFY1OS42MzVaIgogICAgICAgIGZpbGw9IndoaXRlIiAvPgo8L3N2Zz4="
class="header3__card-recommendation-header-photo header3__card-recommendation-header-photo_default"
/>
<img
src="data:image/svg+xml;base64,<svg width="80" height="80" viewBox="0 0 80 80" fill="none" xmlns="http://www.w3.org/2000/svg">
    <mask id="a" fill="#fff">
        <path fill-rule="evenodd" clip-rule="evenodd"
            d="M0 12C0 5.373 5.373 0 12 0h56c6.627 0 12 5.373 12 12v56c0 6.627-5.373 12-12 12H12C5.373 80 0 74.627 0 68V12Zm35.005 33.873c0 1.058 1.104 1.703 1.955 1.142l7.67-5.052c.373-.246.6-.678.6-1.142V26.749s2.556 0 8.947-4.042c3.124-1.975 4.721-3.951 5.537-5.454.533-.981-.463-1.694-1.417-1.18-2.968 1.595-8.905 3.94-18.18 3.94-9.274 0-15.212-2.345-18.179-3.94-.954-.514-1.95.199-1.418 1.18.817 1.503 2.413 3.479 5.538 5.454 6.39 4.042 8.947 4.042 8.947 4.042v19.124Zm-8.051-6.235c2.18 0 3.949-1.83 3.949-4.085 0-2.257-1.768-4.086-3.95-4.086-2.18 0-3.949 1.83-3.949 4.086 0 2.256 1.768 4.085 3.95 4.085Zm30.276-4.085c0 2.256-1.768 4.085-3.949 4.085-2.18 0-3.949-1.83-3.949-4.085 0-2.257 1.768-4.086 3.95-4.086 2.18 0 3.948 1.83 3.948 4.086ZM26.954 58.116c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.139.79.386 1.074.514.593 1.347.593 1.861 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Zm18.429 0c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.14.79.386 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.59-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.386 1.074-.514.593-1.347.593-1.861 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.589-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.385 1.074-.514.593-1.348.593-1.862 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Z" />
    </mask>
    <path fill-rule="evenodd" clip-rule="evenodd"
        d="M0 12C0 5.373 5.373 0 12 0h56c6.627 0 12 5.373 12 12v56c0 6.627-5.373 12-12 12H12C5.373 80 0 74.627 0 68V12Zm35.005 33.873c0 1.058 1.104 1.703 1.955 1.142l7.67-5.052c.373-.246.6-.678.6-1.142V26.749s2.556 0 8.947-4.042c3.124-1.975 4.721-3.951 5.537-5.454.533-.981-.463-1.694-1.417-1.18-2.968 1.595-8.905 3.94-18.18 3.94-9.274 0-15.212-2.345-18.179-3.94-.954-.514-1.95.199-1.418 1.18.817 1.503 2.413 3.479 5.538 5.454 6.39 4.042 8.947 4.042 8.947 4.042v19.124Zm-8.051-6.235c2.18 0 3.949-1.83 3.949-4.085 0-2.257-1.768-4.086-3.95-4.086-2.18 0-3.949 1.83-3.949 4.086 0 2.256 1.768 4.085 3.95 4.085Zm30.276-4.085c0 2.256-1.768 4.085-3.949 4.085-2.18 0-3.949-1.83-3.949-4.085 0-2.257 1.768-4.086 3.95-4.086 2.18 0 3.948 1.83 3.948 4.086ZM26.954 58.116c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.139.79.386 1.074.514.593 1.347.593 1.861 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Zm18.429 0c-.727 0-1.317.68-1.317 1.519v2.409c0 .403.14.79.386 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.59-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.386 1.074-.514.593-1.347.593-1.861 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.385-.671.385-1.074v-2.409c0-.839-.589-1.519-1.316-1.519Zm-15.797 1.519c0-.839.59-1.519 1.317-1.519s1.316.68 1.316 1.519v2.409c0 .403-.139.79-.385 1.074-.514.593-1.348.593-1.862 0a1.645 1.645 0 0 1-.386-1.074v-2.409Zm19.746-1.519c-.727 0-1.316.68-1.316 1.519v2.409c0 .403.138.79.385 1.074.514.593 1.348.593 1.862 0 .247-.285.386-.671.386-1.074v-2.409c0-.839-.59-1.519-1.317-1.519Z"
        fill="#fff" />
    <path
        d="m36.96 47.015 1.1 1.67-1.1-1.67Zm7.67-5.052 1.1 1.67-1.1-1.67Zm.6-15.214v-2h-2v2h2Zm8.947-4.042 1.069 1.69-1.069-1.69Zm5.537-5.454 1.758.954-1.758-.954Zm-1.417-1.18.947 1.76-.947-1.76Zm-36.359 0-.947 1.76.947-1.76Zm-1.418 1.18 1.758-.955-1.758.955Zm5.538 5.454-1.07 1.69 1.07-1.69Zm8.947 4.042h2v-2h-2v2Zm-8.982 36.37 1.511-1.31-1.511 1.31Zm1.861 0 1.512 1.309-1.512-1.31Zm16.568 0-1.511 1.309 1.511-1.31Zm1.862 0 1.511 1.309-1.511-1.31Zm-14.48 0-1.512-1.31 1.512 1.31Zm-1.862 0 1.511-1.31-1.511 1.31Zm18.43 0-1.512 1.309 1.511-1.31Zm1.86 0 1.512 1.309-1.511-1.31Zm-14.48 0 1.512 1.309-1.511-1.31Zm-1.861 0-1.511 1.309 1.511-1.31Zm18.43 0 1.51-1.31-1.51 1.31Zm1.861 0 1.511 1.309-1.511-1.31ZM12-2C4.268-2-2 4.268-2 12h4C2 6.477 6.477 2 12 2v-4Zm56 0H12v4h56v-4Zm14 14c0-7.732-6.268-14-14-14v4c5.523 0 10 4.477 10 10h4Zm0 56V12h-4v56h4ZM68 82c7.732 0 14-6.268 14-14h-4c0 5.523-4.477 10-10 10v4Zm-56 0h56v-4H12v4ZM-2 68c0 7.732 6.268 14 14 14v-4C6.477 78 2 73.523 2 68h-4Zm0-56v56h4V12h-4Zm37.86 33.345a.791.791 0 0 1 .812-.021.656.656 0 0 1 .333.549h-4c0 2.454 2.714 4.355 5.055 2.812l-2.2-3.34Zm7.669-5.052-7.669 5.052 2.2 3.34 7.67-5.052-2.201-3.34Zm-.299.528c0-.178.086-.388.299-.528l2.2 3.34a3.365 3.365 0 0 0 1.501-2.812h-4Zm0-14.072V40.82h4V26.749h-4Zm9.878-5.732c-3.097 1.959-5.197 2.895-6.464 3.34a8.845 8.845 0 0 1-1.28.362 2.42 2.42 0 0 1-.203.031H45.19l.018-.001h.02c.001 0 .003 0 .003 2s.002 2 .003 2h.054a3.196 3.196 0 0 0 .256-.017c.144-.014.327-.038.552-.08.45-.084 1.069-.238 1.875-.521 1.61-.566 3.982-1.65 7.276-3.733l-2.138-3.381Zm4.849-4.719c-.637 1.173-1.983 2.906-4.849 4.719l2.138 3.38c3.382-2.138 5.23-4.356 6.226-6.19l-3.515-1.909Zm1.287 1.536c-.106.056-.494.173-.916-.158a1.233 1.233 0 0 1-.448-.76.985.985 0 0 1 .077-.618l3.515 1.909c.308-.568.472-1.248.351-1.96a2.768 2.768 0 0 0-1.025-1.718c-1.026-.805-2.388-.788-3.449-.218l1.895 3.523Zm-19.127 4.18c9.628 0 15.889-2.438 19.127-4.18l-1.895-3.523c-2.696 1.45-8.31 3.702-17.232 3.702v4Zm-19.126-4.18c3.238 1.742 9.499 4.18 19.126 4.18v-4c-8.921 0-14.535-2.253-17.231-3.703l-1.895 3.523Zm1.287-1.536c.042.078.131.3.077.619a1.232 1.232 0 0 1-.449.759c-.422.33-.81.215-.915.158l1.895-3.523c-1.06-.57-2.424-.587-3.449.218a2.768 2.768 0 0 0-1.026 1.719c-.12.711.043 1.391.352 1.959l3.515-1.909Zm4.849 4.719c-2.867-1.813-4.212-3.546-4.849-4.719l-3.515 1.909c.995 1.834 2.844 4.052 6.226 6.19l2.138-3.38Zm7.878 5.732c0-2 .001-2 .003-2H35.047l.025.001h.001a2.42 2.42 0 0 1-.202-.03 8.852 8.852 0 0 1-1.28-.363c-1.267-.445-3.367-1.381-6.464-3.34l-2.138 3.38c3.293 2.084 5.666 3.168 7.276 3.734.806.283 1.425.437 1.875.52a6.356 6.356 0 0 0 .737.095l.071.002h.054c.001 0 .003 0 .003-2Zm2 19.124V26.749h-4v19.124h4Zm-8.102-10.32c0 1.216-.936 2.085-1.95 2.085v4c3.35 0 5.95-2.789 5.95-6.085h-4Zm-1.95-2.086c1.014 0 1.95.87 1.95 2.086h4c0-3.297-2.6-6.086-5.95-6.086v4Zm-1.949 2.086c0-1.216.936-2.086 1.95-2.086v-4c-3.35 0-5.95 2.79-5.95 6.086h4Zm1.95 2.085c-1.014 0-1.95-.87-1.95-2.085h-4c0 3.296 2.6 6.085 5.95 6.085v-4Zm26.327 4c3.35 0 5.95-2.789 5.95-6.085h-4c0 1.216-.937 2.085-1.95 2.085v4Zm-5.949-6.085c0 3.296 2.6 6.085 5.95 6.085v-4c-1.014 0-1.95-.87-1.95-2.085h-4Zm5.95-6.086c-3.35 0-5.95 2.79-5.95 6.086h4c0-1.216.936-2.086 1.95-2.086v-4Zm5.948 6.086c0-3.297-2.6-6.086-5.949-6.086v4c1.013 0 1.95.87 1.95 2.086h4ZM27.637 59.635a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4c-2.091 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.45.45 0 0 1 .086.142.277.277 0 0 1 .017.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.161 0a.776.776 0 0 1 .58-.245c.27 0 .476.124.581.245l-3.023 2.62c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62Zm-.103.236c0-.027.004-.058.017-.094a.45.45 0 0 1 .086-.142l3.023 2.62a3.644 3.644 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.683.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.103-.236h4c0-1.665-1.226-3.519-3.317-3.519v4Zm19.113-.481a.37.37 0 0 1-.102.236.774.774 0 0 1-.581.245v-4c-2.091 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.102-.236a.45.45 0 0 1 .085.142.277.277 0 0 1 .017.094h-4c0 .848.29 1.71.875 2.384l3.023-2.62Zm-1.162 0a.775.775 0 0 1 .58-.245c.271 0 .477.124.582.245l-3.023 2.62c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62Zm-.103.236c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62a3.645 3.645 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.684.481c-.29 0-.49-.14-.58-.245a.369.369 0 0 1-.104-.236h4c0-1.665-1.225-3.519-3.316-3.519v4Zm-14.48-4c-2.091 0-3.317 1.854-3.317 3.519h4a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4Zm3.316 3.519c0-1.665-1.225-3.519-3.316-3.519v4c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4Zm0 2.409v-2.409h-4v2.409h4Zm-.874 2.384a3.644 3.644 0 0 0 .874-2.384h-4c0-.027.005-.058.017-.094a.45.45 0 0 1 .086-.142l3.023 2.62Zm-4.885 0c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62a.775.775 0 0 1 .58-.245c.27 0 .476.124.581.245l-3.023 2.62Zm-.874-2.384c0 .848.29 1.71.874 2.384l3.023-2.62a.45.45 0 0 1 .086.142.277.277 0 0 1 .017.094h-4Zm0-2.409v2.409h4v-2.409h-4Zm22.43 0a.37.37 0 0 1-.103.236.774.774 0 0 1-.581.245v-4c-2.09 0-3.316 1.854-3.316 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.453.453 0 0 1 .085.142.277.277 0 0 1 .018.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.162 0a.776.776 0 0 1 .581-.245c.27 0 .476.124.58.245l-3.022 2.62c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62Zm-.103.236c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62a3.644 3.644 0 0 0 .874-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.684.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4c0-1.665-1.225-3.519-3.316-3.519v4Zm-14.48-4c-2.091 0-3.316 1.854-3.316 3.519h4a.37.37 0 0 1-.104.236.774.774 0 0 1-.58.245v-4Zm3.316 3.519c0-1.665-1.225-3.519-3.316-3.519v4c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.104-.236h4Zm0 2.409v-2.409h-4v2.409h4Zm-.874 2.384a3.644 3.644 0 0 0 .874-2.384h-4c0-.027.005-.058.018-.094a.45.45 0 0 1 .085-.142l3.023 2.62Zm-4.884 0c1.311 1.513 3.573 1.513 4.884 0l-3.023-2.62a.776.776 0 0 1 .58-.245c.271 0 .477.124.581.245l-3.022 2.62Zm-.874-2.384c0 .848.289 1.71.874 2.384l3.023-2.62a.453.453 0 0 1 .085.142.277.277 0 0 1 .017.094h-4Zm0-2.409v2.409h4v-2.409h-4Zm22.429 0a.37.37 0 0 1-.103.236.774.774 0 0 1-.58.245v-4c-2.092 0-3.317 1.854-3.317 3.519h4Zm0 2.409v-2.409h-4v2.409h4Zm-.103-.236a.45.45 0 0 1 .085.142.277.277 0 0 1 .018.094h-4c0 .848.29 1.71.874 2.384l3.023-2.62Zm-1.162 0a.776.776 0 0 1 .581-.245c.27 0 .476.124.58.245l-3.022 2.62c1.312 1.513 3.573 1.513 4.885 0l-3.023-2.62Zm-.102.236c0-.027.004-.058.017-.094a.455.455 0 0 1 .085-.142l3.023 2.62a3.645 3.645 0 0 0 .875-2.384h-4Zm0-2.409v2.409h4v-2.409h-4Zm.683.481c-.29 0-.49-.14-.58-.245a.37.37 0 0 1-.103-.236h4c0-1.665-1.226-3.519-3.317-3.519v4Z"
        fill="#fff" mask="url(#a)" />
</svg>"
class="header3__card-recommendation-header-photo header3__card-recommendation-header-photo_default_inverted"
/>
</div>
<div class="header3__card-recommendation-header-content">
<div class="header3__card-recommendation-header-content-tag">
Курс
</div>
<div class="header3__card-recommendation-header-sub">
<span
class="header3__card-recommendation-header-sub-chunk header3__card-recommendation-header-sub-chunk_green"
>
Скидка 20000 ₽
</span>
</div>
</div>
</div>
<h6 class="header3__card-recommendation-title">
ИИ для аналитики и работы с данными
</h6>
<div class="header3__card-recommendation-footer">
12 марта
, 2026
· 2 месяца
</div>
</a>
</div>
</div>
<div>
<p class="header3__nav-section-title ">Ответим на ваши вопросы</p>
<div class="header3__nav-section-items ">
<a
class="header3__nav-section-item header3__nav-section-item_phone"
rel="noopener noreferrer"
href="tel:+7 499 938-92-02"
>
<svg
class="header3__phone-icon"
width="24"
height="24"
viewBox="0 0 24 24"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M14.352 2.75a7.971 7.971 0 0 1 7.041 7.032M14.352 6.293a4.426 4.426 0 0 1 3.5 3.5"
stroke="currentColor"
stroke-width="2"
stroke-linecap="round"
stroke-linejoin="round"
></path>
<path
clip-rule="evenodd"
d="M7.7 16.299C.803 9.4 1.783 6.241 2.51 5.223c.094-.164 2.396-3.611 4.865-1.589 6.126 5.045-1.63 4.332 3.514 9.477 5.146 5.144 4.431-2.611 9.477 3.514 2.022 2.469-1.425 4.771-1.588 4.864-1.018.728-4.178 1.709-11.078-5.19Z"
stroke="currentColor"
stroke-width="2"
stroke-linecap="round"
stroke-linejoin="round"
></path>
</svg>
+7 499 938-92-02
</a>
</div>
</div>
</div>
</div>
</div>
<div class="header3__hamburger-pagination">
<div
class="header3__hamburger-pagination-button header3__hamburger-pagination-button_prev"
>
<svg
width="24"
height="24"
viewBox="0 0 24 24"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M19.75 11.726h-15M13.7 5.701l6.05 6.024-6.05 6.025"
stroke="currentColor"
stroke-width="1.5"
stroke-linecap="round"
stroke-linejoin="round"
></path>
</svg>
</div>
<div class="header3__hamburger-pagination-items"></div>
<div
class="header3__hamburger-pagination-button header3__hamburger-pagination-button_next"
>
<svg
width="24"
height="24"
viewBox="0 0 24 24"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M19.75 11.726h-15M13.7 5.701l6.05 6.024-6.05 6.025"
stroke="currentColor"
stroke-width="1.5"
stroke-linecap="round"
stroke-linejoin="round"
></path>
</svg>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</header>
<div class="blog js-blog">
<div class="blog__nav-wrapper">
<div class="nav nav_blue nav_mobile-fix">
<div class="nav__scroll">
<div class="container container-relative container-overflow-auto">
<div class="nav__items nav__float-left">
<a href="/nest/" class="nav__item" title="Блоги">Блоги</a>
<a href="/nest/posts/" class="nav__item nav__item_divider nav__item_divider-pad" title="Посты">
Посты
</a>
<a href="/nest/best/" class="nav__item nav__item_divider"
title="Лучшие">
Лучшие
</a>
<a href="/nest/users/" class="nav__item" title="Участники">Участники</a>
<div class="nav__item show-md">
<form action="/nest/search/" method="get" class="inline-block search js-search">
<div class="search__box js-search-box search__box_close">
<input autocomplete="off"
name="q"
placeholder="Поиск блогов, постов, текста в постах"
class="search__input input"
/>
<i class="ic ic-close search__box-close js-cancel"></i>
<button class="button button_gray2 search__box-button">
<i class="ic ic-search-black search__icon"></i>Найти
</button>
</div>
<div class="search__button js-open-search">
<i class="ic ic-search search__icon"></i>Поиск
</div>
</form>
<form action="/nest/post/add/" method="get" class="js-form-need-auth inline-block">
<input type="hidden" name="blog" value="6"/>
</form>
</div>
</div>
<div class="nav__items nav__float-right hide-md">
<div class="nav__item vertical-middle">
<form action="/nest/search/" method="get" class="inline-block search js-search">
<div class="search__box js-search-box search__box_close">
<input autocomplete="off"
name="q"
placeholder="Поиск блогов, постов, текста в постах"
class="search__input input"
/>
<i class="ic ic-close search__box-close js-cancel"></i>
<button class="button button_gray2 search__box-button">
<i class="ic ic-search-black search__icon"></i>Найти
</button>
</div>
<div class="search__button js-open-search">
<i class="ic ic-search search__icon"></i>Поиск
</div>
</form>
<form action="/nest/post/add/" method="get" class="js-form-need-auth inline-block">
<input type="hidden" name="blog" value="6"/>
</form>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="blog__content">
<div class="container">
<div class="container__row container__row_gutter-24">
<div class="container__col container__col_8 container__col_md-12">
<div class="blog-post">
<div class="blog__tile blog-post__tile">
<article class="blog-post-content">
<div class="post-info">
<a href="/profile/4868/">
<div class="post-info__avatar ic ic-blog-default-avatar" style="background-image: url(https://cdn.otus.ru/media/public/a9/67/avatar-4868-a967e9.png);"></div>
<div class="post-info__author">Дмитрий</div>
</a>
<div class="post-info__time">16.05.18 в 20:30</div>
</div>
<h1 class="blog__h1">20 типичных ошибок многопоточности в C++</h1>
<div class="post-info">
<div class="post-info__blogs">
<a href="/nest/cpp/" class="post-info__blog" title="C++" >C++</a> →
<a href="/nest/cpp-art/" class="post-info__blog" title="Полезные материалы по С++" >Полезные материалы по С++</a>
</div>
</div>
<div class="blog-post-text blog-post-text_markdown markdown">
<p><img alt="С___Deep_25-5020-1175ed.09_site.png" src="https://cdn.otus.ru/media/public/11/75/С___Deep_25-5020-1175ed.09_site.png"></p>
<p>Многопоточное программирование одна из самых сложных тем в программировании, особенно в C++. Трудно избежать при этом ошибок. К счастью большую часть удаётся отловить на этапе проверки кода или тестирования. Но особо коварные проникают в рабочие системы и исправлять их достаточно затруднительно.</p>
<p>В этой статье собраны и переведены самые значимые по мнению автора <a href="http://www.acodersjourney.com/2017/08/top-20-cplusplus-multithreading-mistakes/">заметки</a> ошибочные ситуации. Если у вас есть свои любимые ошибки или варианты их решения, оставьте, пожалуйста, их в комментариях.<cut></cut></p>
<p>Все примеры успешно компилируются и исполняются в Ubuntu 16.04 LTS:</p>
<pre><div class="codehilite"><pre><span></span>g++ -std<span class="o">=</span>c++14 -O2 -Wall -pedantic -pthread main.cpp <span class="o">&&</span> ./a.out
</pre></div>
</pre>
<h2>#1 Отсутствие join() или detach() перед завершением</h2>
<p>Если забыть вызвать <codeinline>join()</codeinline> или <codeinline>detach()</codeinline> перед завершением программы, это может привести к её аварийному завершению.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><thread></span><span class="cp"></span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"foo"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t</span><span class="p">(</span><span class="n">foo</span><span class="p">);</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>В конце функции <codeinline>main()</codeinline> объект <codeinline>t</codeinline> выходит из области видимости и вызывается деструктор. Внутри деструктора выполняется проверка на подключаемость потока. Подключаемый поток - это поток который может или уже выполняется. В данном случае это именно так поэтому будет вызвана функция <codeinline>std::terminate()</codeinline>.</p>
<p>В зависимости желаемого поведения следует либо подождать завершения потока:</p>
<pre><div class="codehilite"><pre><span></span><span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t</span><span class="p">(</span><span class="n">foo</span><span class="p">);</span>
<span class="n">t</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>либо разорвать с ним связь</p>
<pre><div class="codehilite"><pre><span></span><span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t</span><span class="p">(</span><span class="n">foo</span><span class="p">);</span>
<span class="n">t</span><span class="p">.</span><span class="n">detach</span><span class="p">();</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<h2>#2 Попытка дождаться завершения неподключаемого потока</h2>
<p>Для объектов <codeinline>std::thread</codeinline> которые были перемещены, завершены <codeinline>join()</codeinline> или брошены <codeinline>detach()</codeinline> нельзя дождаться завершения.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><thread></span><span class="cp"></span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"foo"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t</span><span class="p">(</span><span class="n">foo</span><span class="p">);</span>
<span class="n">t</span><span class="p">.</span><span class="n">detach</span><span class="p">();</span>
<span class="c1">// ... какая-то логика ...</span>
<span class="n">t</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>В таких случаях следует проверять, а можно ли в принципе подключить поток, и только потом уже вызывать <codeinline>join()</codeinline>.</p>
<pre><div class="codehilite"><pre><span></span> <span class="k">if</span> <span class="p">(</span><span class="n">t</span><span class="p">.</span><span class="n">joinable</span><span class="p">())</span>
<span class="p">{</span>
<span class="n">t</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="p">}</span>
</pre></div>
</pre>
<h2>#3 Вызов join() блокирует вызывающий поток</h2>
<p>В реальных приложениях потоки могут обслуживать достаточно длительные операции связанные с сетевым вводом/выводом или реакцией пользователя в пользовательском интерфейсе. Попытка дождаться завершения таких потоков в основном потоке или в потоке отвечающим за интерфейс может привести к "заморозке". Лучше найти другое решение.</p>
<p>Например, для десктопного приложения вспомогательный поток перед своим завершением может послать сообщение в поток отвечающий за интерфейс. Последний как правило организован как обработчик очереди сообщений. Чаще всего это сообщения от элементов интерфейса, но могут быть и нажатия клавиш и даже перемещения мыши. В этом потоке точно так же можно получить сообщение от вспомогательного потока и среагировать на завершение, не дожидаясь его буквально, и не блокируя ожиданием обработку событий.</p>
<h2>#4 Не учитывать особенности передачи аргументов в поток</h2>
<p>Аргументы в функцию потока перемещаются или копируются по значению. Пример ниже даже не скомпилируется.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><thread></span><span class="cp"></span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">(</span><span class="kt">int</span> <span class="o">&</span><span class="n">s</span><span class="p">)</span>
<span class="p">{</span>
<span class="n">s</span> <span class="o">=</span> <span class="mi">42</span><span class="p">;</span>
<span class="p">}</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="kt">int</span> <span class="n">answer</span> <span class="o">=</span> <span class="mi">0</span><span class="p">;</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t</span><span class="p">(</span><span class="n">foo</span><span class="p">,</span> <span class="n">answer</span><span class="p">);</span>
<span class="n">t</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Для успешной компиляции ссылка должна быть передана через <codeinline>std::ref</codeinline></p>
<pre><div class="codehilite"><pre><span></span> <span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t</span><span class="p">(</span><span class="n">foo</span><span class="p">,</span> <span class="n">std</span><span class="o">::</span><span class="n">ref</span><span class="p">(</span><span class="n">answer</span><span class="p">));</span>
</pre></div>
</pre>
<h2>#5 Игнорировать общий доступ к ресурсам</h2>
<p>В условиях многопоточности несколько потоков могут одновременно использовать общие ресурсы или данные. Это может приводить не только к труднопрогнозируемому поведению но и к краху программы. В таких случаях необходимо упорядочивать и контролировать доступ.</p>
<p>В качестве примера рассмотрим вывод в консоль в несколько потоков - основной и шесть дополнительных.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><thread></span><span class="cp"></span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">(</span><span class="k">const</span> <span class="n">std</span><span class="o">::</span><span class="n">string</span> <span class="o">&</span><span class="n">message</span><span class="p">)</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"thread "</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">this_thread</span><span class="o">::</span><span class="n">get_id</span><span class="p">()</span> <span class="o"><<</span> <span class="s">", message "</span> <span class="o"><<</span> <span class="n">message</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t1</span><span class="p">(</span><span class="n">foo</span><span class="p">,</span> <span class="s">"каждый"</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t2</span><span class="p">(</span><span class="n">foo</span><span class="p">,</span> <span class="s">"охотник"</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t3</span><span class="p">(</span><span class="n">foo</span><span class="p">,</span> <span class="s">"желает"</span><span class="p">);</span>
<span class="n">foo</span><span class="p">(</span><span class="s">"знать"</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t4</span><span class="p">(</span><span class="n">foo</span><span class="p">,</span> <span class="s">"где"</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t5</span><span class="p">(</span><span class="n">foo</span><span class="p">,</span> <span class="s">"сидит"</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t6</span><span class="p">(</span><span class="n">foo</span><span class="p">,</span> <span class="s">"фазан"</span><span class="p">);</span>
<span class="n">t1</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="n">t2</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="n">t3</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="n">t4</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="n">t5</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="n">t6</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>В результате получим мешанину из слов:</p>
<pre><div class="codehilite"><pre><span></span>thread thread thread 140597982512960, message знать140597965162240
thread 140597939984128, message где
thread 140597931591424, message 140597956769536, message thread каждый140597948376832, message желает, message
сидит
охотник
thread 140597923198720, message фазан
</pre></div>
</pre>
<p>Дело в том, что консоль одна, а семь потоков пытаются выводить на неё одновременно. Чтобы сделать вывод более предсказуемым необходимо ограничить одновременный доступ. Сделаем это через <codeinline>std::mutex</codeinline> - заблокируем до вывода и освободим после.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><mutex></span><span class="cp"></span>
<span class="n">std</span><span class="o">::</span><span class="n">mutex</span> <span class="n">cout_guard</span><span class="p">;</span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">(</span><span class="k">const</span> <span class="n">std</span><span class="o">::</span><span class="n">string</span> <span class="o">&</span><span class="n">message</span><span class="p">)</span>
<span class="p">{</span>
<span class="n">cout_guard</span><span class="p">.</span><span class="n">lock</span><span class="p">();</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"thread "</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">this_thread</span><span class="o">::</span><span class="n">get_id</span><span class="p">()</span> <span class="o"><<</span> <span class="s">", message "</span> <span class="o"><<</span> <span class="n">message</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="n">cout_guard</span><span class="p">.</span><span class="n">unlock</span><span class="p">();</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Вывод получится читаемый:</p>
<pre><div class="codehilite"><pre><span></span>thread 140710040659776, message знать
thread 140710023309056, message каждый
thread 140710006523648, message желает
thread 140709989738240, message сидит
thread 140709981345536, message фазан
thread 140710014916352, message охотник
thread 140709998130944, message где
</pre></div>
</pre>
<p>Мешанины из фрагментов строк уже нет. Порядок захвата нами не управляется, поэтому сами сообщения появляются в произвольном порядке.</p>
<h2>#6 Забыть вызвать unlock()</h2>
<p>В предыдущем примере для разделения доступа к ресурсу использовался <codeinline>std::mutex</codeinline>. Это не самый удачный способ, поскольку вызова <codeinline>unlock()</codeinline> мы можем и не достичь, если вообще не забыли его вызвать.</p>
<pre><div class="codehilite"><pre><span></span><span class="kt">void</span> <span class="nf">foo</span><span class="p">(</span><span class="k">const</span> <span class="n">std</span><span class="o">::</span><span class="n">string</span> <span class="o">&</span><span class="n">message</span><span class="p">)</span>
<span class="p">{</span>
<span class="n">cout_guard</span><span class="p">.</span><span class="n">lock</span><span class="p">();</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"thread "</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">this_thread</span><span class="o">::</span><span class="n">get_id</span><span class="p">()</span> <span class="o"><<</span> <span class="s">", message "</span> <span class="o"><<</span> <span class="n">message</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="c1">// cout_guard.unlock();</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>После появления первого же сообщения программа зависнет:</p>
<pre><div class="codehilite"><pre><span></span>thread 140569688782656, message знать
</pre></div>
</pre>
<p>Чтобы защититься от ошибок такого рода воспользуемся <codeinline>std::lock_guard</codeinline>, который манипулирует временем жизни блокировки в стиле RAII.</p>
<p>В конструкторе захватывает, в деструкторе освобождает. По какой бы причине мы не покинули область видимости - блокировка будет снята.</p>
<pre><div class="codehilite"><pre><span></span><span class="kt">void</span> <span class="nf">foo</span><span class="p">(</span><span class="k">const</span> <span class="n">std</span><span class="o">::</span><span class="n">string</span> <span class="o">&</span><span class="n">message</span><span class="p">)</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">lock_guard</span><span class="o"><</span><span class="n">std</span><span class="o">::</span><span class="n">mutex</span><span class="o">></span> <span class="n">lock</span><span class="p">(</span><span class="n">cout_guard</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"thread "</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">this_thread</span><span class="o">::</span><span class="n">get_id</span><span class="p">()</span> <span class="o"><<</span> <span class="s">", message "</span> <span class="o"><<</span> <span class="n">message</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<h2>#7 Пренебрежение размером защищённой секции</h2>
<p>Пока мы находимся внутри защищённой секции все остальные потоки рвущиеся её выполнить заблокированы. Старайтесь делать заблокированный участок как можно меньше.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><chrono></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><mutex></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><thread></span><span class="cp"></span>
<span class="k">using</span> <span class="k">namespace</span> <span class="n">std</span><span class="o">::</span><span class="n">chrono_literals</span><span class="p">;</span>
<span class="n">std</span><span class="o">::</span><span class="n">mutex</span> <span class="n">cout_mutex</span><span class="p">;</span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">lock_guard</span><span class="o"><</span><span class="n">std</span><span class="o">::</span><span class="n">mutex</span><span class="o">></span> <span class="n">lock</span><span class="p">(</span><span class="n">cout_mutex</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="n">this_thread</span><span class="o">::</span><span class="n">sleep_for</span><span class="p">(</span><span class="mi">1</span><span class="n">s</span><span class="p">);</span> <span class="c1">// на самом деле что-то очень полезное и безопасное</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"foo"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t1</span><span class="p">(</span><span class="n">foo</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t2</span><span class="p">(</span><span class="n">foo</span><span class="p">);</span>
<span class="n">t1</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="n">t2</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Безопасный вариант программы будет исполняться почти две секунды.
Если нам необходимо защитить только вывод в консоль, то нет необходимости в секции такого размера. Мы можем переместить блокировки ближе к выводу.</p>
<pre><div class="codehilite"><pre><span></span><span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">this_thread</span><span class="o">::</span><span class="n">sleep_for</span><span class="p">(</span><span class="mi">1</span><span class="n">s</span><span class="p">);</span> <span class="c1">// на самом деле что-то очень полезное и безопасное</span>
<span class="n">std</span><span class="o">::</span><span class="n">lock_guard</span><span class="o"><</span><span class="n">std</span><span class="o">::</span><span class="n">mutex</span><span class="o">></span> <span class="n">lock</span><span class="p">(</span><span class="n">cout_mutex</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"foo"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Такой вариант будет выполняться уже около одной секунды и при этом останется безопасным.</p>
<h2>#8 Взаимные блокировки</h2>
<p>Как правило такие блокировки уже навсегда из-за чего получили название deadlock. Типичная ситуация такой блокировки представлена ниже. Функция <codeinline>sleep_for</codeinline> даёт нам 100% шанс попасть в вечную блокировку, без неё скорее всего на любой машине этот код выполнился бы без зависания.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><chrono></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><mutex></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><thread></span><span class="cp"></span>
<span class="k">using</span> <span class="k">namespace</span> <span class="n">std</span><span class="o">::</span><span class="n">chrono_literals</span><span class="p">;</span>
<span class="n">std</span><span class="o">::</span><span class="n">mutex</span> <span class="n">cerr_mutex</span><span class="p">;</span>
<span class="n">std</span><span class="o">::</span><span class="n">mutex</span> <span class="n">cout_mutex</span><span class="p">;</span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">cerr_mutex</span><span class="p">.</span><span class="n">lock</span><span class="p">();</span>
<span class="n">std</span><span class="o">::</span><span class="n">cerr</span> <span class="o"><<</span> <span class="s">"use cerr in foo"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="n">std</span><span class="o">::</span><span class="n">this_thread</span><span class="o">::</span><span class="n">sleep_for</span><span class="p">(</span><span class="mi">1</span><span class="n">s</span><span class="p">);</span>
<span class="n">cout_mutex</span><span class="p">.</span><span class="n">lock</span><span class="p">();</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"use cout in foo"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="n">cout_mutex</span><span class="p">.</span><span class="n">unlock</span><span class="p">();</span>
<span class="n">cerr_mutex</span><span class="p">.</span><span class="n">unlock</span><span class="p">();</span>
<span class="p">}</span>
<span class="kt">void</span> <span class="nf">bar</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">cout_mutex</span><span class="p">.</span><span class="n">lock</span><span class="p">();</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"use cout in bar"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="n">std</span><span class="o">::</span><span class="n">this_thread</span><span class="o">::</span><span class="n">sleep_for</span><span class="p">(</span><span class="mi">1</span><span class="n">s</span><span class="p">);</span>
<span class="n">cerr_mutex</span><span class="p">.</span><span class="n">lock</span><span class="p">();</span>
<span class="n">std</span><span class="o">::</span><span class="n">cerr</span> <span class="o"><<</span> <span class="s">"use cerr in bar"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="n">cerr_mutex</span><span class="p">.</span><span class="n">unlock</span><span class="p">();</span>
<span class="n">cout_mutex</span><span class="p">.</span><span class="n">unlock</span><span class="p">();</span>
<span class="p">}</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t1</span><span class="p">(</span><span class="n">foo</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t2</span><span class="p">(</span><span class="n">bar</span><span class="p">);</span>
<span class="n">t1</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="n">t2</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Причина зависания кроется в перекрёстной блокировке. Когда оба потока начинают работать каждый из них блокирует свой mutex. То есть t1 захватил cerr_mutex, а t2 - cout_mutex. После вызова <codeinline>sleep_for</codeinline> потоки пытаются захватить их наоборот, еще не освободив занятые. Для того чтобы освободить свой mutex потоку приходится ждать пока это сделает второй, а у второго ситуация ровно такая же.</p>
<p>Самое простое решение использовать <codeinline>std::lock</codeinline> для захвата обоих mutex.</p>
<pre><div class="codehilite"><pre><span></span><span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">lock</span><span class="p">(</span><span class="n">cerr_mutex</span><span class="p">,</span> <span class="n">cout_mutex</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="n">cerr</span> <span class="o"><<</span> <span class="s">"use cerr in foo"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="n">std</span><span class="o">::</span><span class="n">this_thread</span><span class="o">::</span><span class="n">sleep_for</span><span class="p">(</span><span class="mi">1</span><span class="n">s</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"use cout in foo"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="n">cout_mutex</span><span class="p">.</span><span class="n">unlock</span><span class="p">();</span>
<span class="n">cerr_mutex</span><span class="p">.</span><span class="n">unlock</span><span class="p">();</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Если условия позволяют, можно использовать <codeinline>std::timed_mutex</codeinline>. Для выхода из перекрёстной блокировки достаточно, чтобы одну из них можно было нарушить.</p>
<pre><div class="codehilite"><pre><span></span><span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">cerr_mutex</span><span class="p">.</span><span class="n">lock</span><span class="p">();</span>
<span class="n">std</span><span class="o">::</span><span class="n">cerr</span> <span class="o"><<</span> <span class="s">"use cerr in foo"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="n">std</span><span class="o">::</span><span class="n">this_thread</span><span class="o">::</span><span class="n">sleep_for</span><span class="p">(</span><span class="mi">1</span><span class="n">s</span><span class="p">);</span>
<span class="k">if</span> <span class="p">(</span><span class="n">cout_mutex</span><span class="p">.</span><span class="n">try_lock_for</span><span class="p">(</span><span class="mi">1</span><span class="n">s</span><span class="p">))</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"use cout in foo"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="n">cout_mutex</span><span class="p">.</span><span class="n">unlock</span><span class="p">();</span>
<span class="p">}</span>
<span class="n">cerr_mutex</span><span class="p">.</span><span class="n">unlock</span><span class="p">();</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>С одной стороны мы успешно избежали блокировки, но с другой стороны нам пришлось взять на себя обработку этой ситуации.</p>
<h2>#9 Повторный захват std::mutex</h2>
<p>Эта некоторая вариация на тему перекрёстного захвата с тем лишь отличием, что для такой блокировки достаточно одного std::mutex. Даже дополнительные потоки не нужны. Тут можно было бы привести в качестве примера страшную банальщину типа:</p>
<pre><div class="codehilite"><pre><span></span><span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">lock_guard</span><span class="o"><</span><span class="n">std</span><span class="o">::</span><span class="n">mutex</span><span class="o">></span> <span class="n">lock1</span><span class="p">(</span><span class="n">cerr_mutex</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="n">lock_guard</span><span class="o"><</span><span class="n">std</span><span class="o">::</span><span class="n">mutex</span><span class="o">></span> <span class="n">lock2</span><span class="p">(</span><span class="n">cerr_mutex</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="n">cerr</span> <span class="o"><<</span> <span class="s">"foo"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Да, это именно такого рода ошибка, вот только в жизни она встречается в несколько более изощрённой форме. В примере ниже нет ни потоков, ни рекурсии и всего один mutex.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><mutex></span><span class="cp"></span>
<span class="n">std</span><span class="o">::</span><span class="n">mutex</span> <span class="n">cerr_mutex</span><span class="p">;</span>
<span class="kt">int</span> <span class="nf">bar</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">lock_guard</span><span class="o"><</span><span class="n">std</span><span class="o">::</span><span class="n">mutex</span><span class="o">></span> <span class="n">lock</span><span class="p">(</span><span class="n">cerr_mutex</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="n">cerr</span> <span class="o"><<</span> <span class="s">"bar"</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="k">return</span> <span class="mi">42</span><span class="p">;</span>
<span class="p">}</span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">lock_guard</span><span class="o"><</span><span class="n">std</span><span class="o">::</span><span class="n">mutex</span><span class="o">></span> <span class="n">lock</span><span class="p">(</span><span class="n">cerr_mutex</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="n">cerr</span> <span class="o"><<</span> <span class="s">"foo, bar = "</span> <span class="o"><<</span> <span class="n">bar</span><span class="p">()</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="n">foo</span><span class="p">();</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Вызов foo приводит к захвату mutex, но в процессе вызова bar возникает необходимость блокировки ресурса повторно. Это такой же deadlock, только теперь основной поток ждёт сам себя.</p>
<p>Существует очевидный способ решить проблему - заменить обычный <codeinline>std::mutex</codeinline> на рекурсивный <codeinline>std::recursive_mutex</codeinline> и это решит нашу проблему, но решит её ой каким опасным способом. Тысячу раз подумайте всё ли будет в порядке при таком подходе, не удастся ли найти решение более элегантное</p>
<pre><div class="codehilite"><pre><span></span><span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="k">auto</span> <span class="n">b</span> <span class="o">=</span> <span class="n">bar</span><span class="p">();</span>
<span class="n">std</span><span class="o">::</span><span class="n">lock_guard</span><span class="o"><</span><span class="n">std</span><span class="o">::</span><span class="n">mutex</span><span class="o">></span> <span class="n">lock</span><span class="p">(</span><span class="n">cerr_mutex</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="n">cerr</span> <span class="o"><<</span> <span class="s">"foo, bar = "</span> <span class="o"><<</span> <span class="n">b</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<h2>#10 Излишняя предосторожность</h2>
<p>Когда возникает необходимость модифицировать простые типы наподобие <codeinline>bool</codeinline> или <codeinline>int</codeinline> использование 'std::atomic' почти всегда более эффективно в сравнении с использованием mutex.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><mutex></span><span class="cp"></span>
<span class="n">std</span><span class="o">::</span><span class="n">mutex</span> <span class="n">counter_mutex</span><span class="p">;</span>
<span class="kt">int</span> <span class="n">counter</span><span class="p">;</span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">lock_guard</span><span class="o"><</span><span class="n">std</span><span class="o">::</span><span class="n">mutex</span><span class="o">></span> <span class="n">lock</span><span class="p">(</span><span class="n">counter_mutex</span><span class="p">);</span>
<span class="o">++</span><span class="n">counter</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Та же самая логика без использования mutex и использованием atomic.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><atomic></span><span class="cp"></span>
<span class="n">std</span><span class="o">::</span><span class="n">atomic</span><span class="o"><</span><span class="kt">int</span><span class="o">></span> <span class="n">counter</span><span class="p">;</span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="o">++</span><span class="n">counter</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<h2>#11 Частое создание потоков без использования пулов</h2>
<p>Создание и удаление потоков дорогое удовольствия с точки зрения затрат CPU. Часто создавать и удалять потоки в приложении, которое само по активно занимается вычислениями значит мешать ему. Вместо того, чтобы часто создавать и удалять потоки лучше создать пул предварительно запущенных потоков и распределять между ними задания.</p>
<p>Использование пула потоков позволяет сократить объём кода и количество потенциальных ошибок связанных с запуском и остановкой потоков. Позволит избежать избыточного количества потоков, которое может негативно сказаться на производительности. </p>
<p>Существуют готовые реализации такого рода пулов. Например, <a href="https://www.threadingbuildingblocks.org/">TBB</a></p>
<h2>#12 Не обработанные исключения в потоке</h2>
<p>Исключения брошенные в одном потоке не могут быть перехвачены другим. Представим себе функцию, которая может бросить исключение. Если мы выполним эту функцию в отдельном потоке, то попытка поймать исключение в основном не сработает.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><stdexcept></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><thread></span><span class="cp"></span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="k">throw</span> <span class="n">std</span><span class="o">::</span><span class="n">runtime_error</span><span class="p">(</span><span class="s">"foo"</span><span class="p">);</span>
<span class="p">}</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="k">try</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t</span><span class="p">(</span><span class="n">foo</span><span class="p">);</span>
<span class="n">t</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="p">}</span>
<span class="k">catch</span> <span class="p">(</span><span class="k">const</span> <span class="n">std</span><span class="o">::</span><span class="n">exception</span> <span class="o">&</span><span class="n">e</span><span class="p">)</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"error"</span> <span class="o"><<</span> <span class="n">e</span><span class="p">.</span><span class="n">what</span><span class="p">()</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Программа аварийно завершится так и не вызвав обработчик исключения. Решением может быть перехват исключения в потоке и передача информации о нём через экземпляр <codeinline>std::exception_ptr</codeinline> в родительский поток и повторного бросания уже за пределами породившего исключения потока.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><stdexcept></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><thread></span><span class="cp"></span>
<span class="k">static</span> <span class="n">std</span><span class="o">::</span><span class="n">exception_ptr</span> <span class="n">eptr</span> <span class="o">=</span> <span class="k">nullptr</span><span class="p">;</span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="k">try</span>
<span class="p">{</span>
<span class="k">throw</span> <span class="n">std</span><span class="o">::</span><span class="n">runtime_error</span><span class="p">(</span><span class="s">"foo"</span><span class="p">);</span>
<span class="p">}</span>
<span class="k">catch</span> <span class="p">(...)</span>
<span class="p">{</span>
<span class="n">eptr</span> <span class="o">=</span> <span class="n">std</span><span class="o">::</span><span class="n">current_exception</span><span class="p">();</span>
<span class="p">}</span>
<span class="p">}</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="kr">thread</span> <span class="n">t</span><span class="p">(</span><span class="n">foo</span><span class="p">);</span>
<span class="n">t</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="k">if</span> <span class="p">(</span><span class="n">eptr</span><span class="p">)</span>
<span class="p">{</span>
<span class="k">try</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">rethrow_exception</span><span class="p">(</span><span class="n">eptr</span><span class="p">);</span>
<span class="p">}</span>
<span class="k">catch</span> <span class="p">(</span><span class="k">const</span> <span class="n">std</span><span class="o">::</span><span class="n">exception</span> <span class="o">&</span><span class="n">e</span><span class="p">)</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"error"</span> <span class="o"><<</span> <span class="n">e</span><span class="p">.</span><span class="n">what</span><span class="p">()</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
<span class="p">}</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<h2>#13 Имитация асинхронной работы без std::async</h2>
<p>Когда нужно выполнить часть кода независимо от основного потока отличным выбором будет использование <codeinline>std::async</codeinline> для запуска. Это тоже самое, что создать ещё один поток и передать ему на выполнение функцию или лямбду. Правда при этом за жизненным циклом потока и исключений в нём тоже придётся следить самостоятельно. В случае использования <codeinline>std::async</codeinline> можно не заботиться об этом да ещё и существенно сократить вероятность блокировки.</p>
<p>Еще одно важно преимущество заключается в возможности получить результат работы функции через <codeinline>std::future</codeinline>. Функция <codeinline>int foo()</codeinline>, будучи выполнена как асинхронная задача, заранее установит результат своей работы. А получим мы его тогда, когда нам это будет удобно.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><cmath></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><future></span><span class="cp"></span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="k">auto</span> <span class="n">f</span> <span class="o">=</span> <span class="n">std</span><span class="o">::</span><span class="n">async</span><span class="p">(</span><span class="n">sqrt</span><span class="p">,</span> <span class="mf">9.0</span><span class="p">);</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="n">f</span><span class="p">.</span><span class="n">get</span><span class="p">()</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Использование потоков напрямую делает получение результатов чуть более громоздким. Это может быть:</p>
<p>Передача ссылки на результирующую переменную в поток, в котором необходимо сохранить результат.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><cmath></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><thread></span><span class="cp"></span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">(</span><span class="kt">double</span> <span class="n">i</span><span class="p">,</span> <span class="kt">double</span> <span class="o">&</span><span class="n">r</span><span class="p">)</span>
<span class="p">{</span>
<span class="n">r</span> <span class="o">=</span> <span class="n">std</span><span class="o">::</span><span class="n">sqrt</span><span class="p">(</span><span class="n">i</span><span class="p">);</span>
<span class="p">}</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="kt">double</span> <span class="n">result</span> <span class="o">=</span> <span class="mf">0.0</span><span class="p">;</span>
<span class="k">auto</span> <span class="n">t</span> <span class="o">=</span> <span class="n">std</span><span class="o">::</span><span class="kr">thread</span><span class="p">(</span><span class="n">foo</span><span class="p">,</span> <span class="mi">9</span><span class="p">,</span> <span class="n">std</span><span class="o">::</span><span class="n">ref</span><span class="p">(</span><span class="n">result</span><span class="p">));</span>
<span class="n">t</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="n">result</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Сохранение результата в переменной класса функционального объекта и чтение его после завершения работы.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><cmath></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><thread></span><span class="cp"></span>
<span class="k">template</span><span class="o"><</span><span class="k">typename</span> <span class="n">T</span><span class="o">></span>
<span class="k">class</span> <span class="nc">foo</span>
<span class="p">{</span>
<span class="n">T</span> <span class="n">r</span><span class="p">;</span>
<span class="k">public</span><span class="o">:</span>
<span class="n">T</span> <span class="n">get</span><span class="p">()</span>
<span class="p">{</span>
<span class="k">return</span> <span class="n">r</span><span class="p">;</span>
<span class="p">}</span>
<span class="kt">void</span> <span class="k">operator</span><span class="p">()(</span><span class="n">T</span> <span class="n">i</span><span class="p">)</span>
<span class="p">{</span>
<span class="n">r</span> <span class="o">=</span> <span class="n">std</span><span class="o">::</span><span class="n">sqrt</span><span class="p">(</span><span class="n">i</span><span class="p">);</span>
<span class="p">}</span>
<span class="p">};</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="k">auto</span> <span class="n">f</span> <span class="o">=</span> <span class="n">foo</span><span class="o"><</span><span class="kt">double</span><span class="o">></span><span class="p">();</span>
<span class="k">auto</span> <span class="n">t</span> <span class="o">=</span> <span class="n">std</span><span class="o">::</span><span class="kr">thread</span><span class="p">(</span><span class="n">std</span><span class="o">::</span><span class="n">ref</span><span class="p">(</span><span class="n">f</span><span class="p">),</span> <span class="mi">9</span><span class="p">);</span>
<span class="n">t</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="n">f</span><span class="p">.</span><span class="n">get</span><span class="p">()</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Курт Гантерот в своей книге утверждает, что создание потоков в 14 раз дороже использования <codeinline>std::async</codeinline>.</p>
<table><tbody><tr><td><a href="http://www.ozon.ru/context/detail/id/140145932/?partner=shebdim&from=bar" title="Курт Гантерот, «Оптимизация программ на C++» "><img alt="Курт Гантерот, «Оптимизация программ на C++» " src="//ozon-st.cdn.ngenix.net/multimedia/c300/1017974039.jpg" style=""></a></td><td style=""><a href="http://www.ozon.ru/context/detail/id/140145932/?partner=shebdim&from=bar" style="" title="Курт Гантерот, «Оптимизация программ на C++» ">Курт Гантерот, «Оптимизация программ на C++» </a></td></tr></tbody></table>
<p>Короче говоря, пока не доказано обратное использовать следует <codeinline>std::async</codeinline>.</p>
<h2>#14 Опускание std::launch::async когда это действительно необходимо</h2>
<p>Название <codeinline>std::async</codeinline> может ввести в заблуждение, потому что функция, которая будет передана для запуска по умолчанию может и не запуститься отдельно от вызываемого потока!</p>
<p>Существует два способа запуска:</p>
<ol>
<li><codeinline>std::launch::async</codeinline>. Задача будет немедленно запущена в отдельном потоке.</li>
<li><codeinline>std::launch::deferred</codeinline>. Выполнение задачи будет отложено до вызова <codeinline>.get()</codeinline> или <codeinline>.wait()</codeinline> возвращаемого объекта <codeinline>std::future</codeinline>. При этом выполнение осуществляется синхронно!</li>
</ol>
<p>Без явного указания способа запуска предполагается комбинация этих вариантов и фактически предсказать как именно будет запущена задача невозможно. Существуют связанные с этим сложности, например невозможно предсказать корректно ли будет обращение к переменным потока, невозможно предсказать будет ли выполнена функция вообще, если до выполнения функций <codeinline>.get()</codeinline> или <codeinline>.wait()</codeinline> дело так и не дошло ну и цикл ожидания готовности future никогда не закончится для отложенного сценария, ждать бесполезно.</p>
<p>Чтобы избежать недоразумений явно указывайте <codeinline>std::launch::async</codeinline> при запуске <codeinline>std::async</codeinline>.</p>
<p>Непредсказуемо:</p>
<pre><div class="codehilite"><pre><span></span> <span class="k">auto</span> <span class="n">f</span> <span class="o">=</span> <span class="n">std</span><span class="o">::</span><span class="n">async</span><span class="p">(</span><span class="n">sqrt</span><span class="p">,</span> <span class="mf">9.0</span><span class="p">);</span>
</pre></div>
</pre>
<p>Гарантировано в другом потоке:</p>
<pre><div class="codehilite"><pre><span></span> <span class="k">auto</span> <span class="n">f</span> <span class="o">=</span> <span class="n">std</span><span class="o">::</span><span class="n">async</span><span class="p">(</span><span class="n">std</span><span class="o">::</span><span class="n">launch</span><span class="o">::</span><span class="n">async</span><span class="p">,</span> <span class="n">sqrt</span><span class="p">,</span> <span class="mf">9.0</span><span class="p">);</span>
</pre></div>
</pre>
<table><tbody><tr><td><a href="http://www.ozon.ru/context/detail/id/34747131/?partner=shebdim&from=bar" title="Скотт Мейерс, «Эффективный и современный С++» "><img alt="Скотт Мейерс, «Эффективный и современный С++» " src="//ozon-st.cdn.ngenix.net/multimedia/c300/1013639927.jpg" style=""></a></td><td style=""><a href="http://www.ozon.ru/context/detail/id/34747131/?partner=shebdim&from=bar" style="" title="Скотт Мейерс, «Эффективный и современный С++» ">Скотт Мейерс, «Эффективный и современный С++» </a></td></tr></tbody></table>
<h2>#15 Использование .get() может привести к ожиданию</h2>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><chrono></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><future></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">future</span><span class="o"><</span><span class="kt">int</span><span class="o">></span> <span class="n">f</span> <span class="o">=</span> <span class="n">std</span><span class="o">::</span><span class="n">async</span><span class="p">(</span><span class="n">std</span><span class="o">::</span><span class="n">launch</span><span class="o">::</span><span class="n">async</span><span class="p">,</span> <span class="p">[](){</span>
<span class="n">std</span><span class="o">::</span><span class="n">this_thread</span><span class="o">::</span><span class="n">sleep_for</span><span class="p">(</span><span class="n">std</span><span class="o">::</span><span class="n">chrono</span><span class="o">::</span><span class="n">seconds</span><span class="p">(</span><span class="mi">1</span><span class="p">));</span>
<span class="k">return</span> <span class="mi">42</span><span class="p">;</span>
<span class="p">});</span>
<span class="k">while</span> <span class="p">(</span><span class="nb">true</span><span class="p">)</span>
<span class="p">{</span>
<span class="c1">// ...</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="n">f</span><span class="p">.</span><span class="n">get</span><span class="p">()</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="c1">// ...</span>
<span class="p">}</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Несмотря на то, что лямбда явно будет запущена в отдельном потоке вызов метода <codeinline>.get()</codeinline> может привести к нежелательному ожиданию. Более того, на следующей итерации код вообще аварийно завершится, потому что первый результат уже был выведен на консоль а никакого другого во future нет.</p>
<p>Обе проблемы можно решить проверив future на готовность.</p>
<pre><div class="codehilite"><pre><span></span> <span class="k">if</span> <span class="p">(</span><span class="n">f</span><span class="p">.</span><span class="n">valid</span><span class="p">())</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="n">f</span><span class="p">.</span><span class="n">get</span><span class="p">()</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<h2>#16 Исключение из задачи перетечёт во future</h2>
<p>Исключение брошенное в асинхронной задаче должно быть обработано так, будто оно возникло в вызываемом потоке. В данном случае пример себя поведёт так, будто исключение никто не обработал.</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><future></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">future</span><span class="o"><</span><span class="kt">int</span><span class="o">></span> <span class="n">f</span> <span class="o">=</span> <span class="n">std</span><span class="o">::</span><span class="n">async</span><span class="p">(</span><span class="n">std</span><span class="o">::</span><span class="n">launch</span><span class="o">::</span><span class="n">async</span><span class="p">,</span> <span class="p">[](){</span>
<span class="k">throw</span> <span class="n">std</span><span class="o">::</span><span class="n">runtime_error</span><span class="p">(</span><span class="s">"error"</span><span class="p">);</span>
<span class="k">return</span> <span class="mi">42</span><span class="p">;</span>
<span class="p">});</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="n">f</span><span class="p">.</span><span class="n">get</span><span class="p">()</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Программа аварийно завершится. Если в задаче было брошено исключение, оно распространится и на вызов .get(). Если до конца жизни future .get() так и не будет вызван исключение просто проигнорируется.</p>
<p>Для таких задач имеет смысл использование обычной конструкции try/catch.</p>
<pre><div class="codehilite"><pre><span></span> <span class="k">try</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="n">f</span><span class="p">.</span><span class="n">get</span><span class="p">()</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
<span class="k">catch</span> <span class="p">(</span><span class="k">const</span> <span class="n">std</span><span class="o">::</span><span class="n">exception</span> <span class="o">&</span><span class="n">e</span><span class="p">)</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="n">e</span><span class="p">.</span><span class="n">what</span><span class="p">()</span> <span class="o"><<</span> <span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<h2>#17 Использование std::async там, где нужен тонкий контроль потоков</h2>
<p>В большинстве случаев достаточно использования <codeinline>std::async</codeinline>, кроме ситуаций когда возникает необходимость в полном контроле над исполняемым потоком.</p>
<p>Например изменить параметры для планировщика:</p>
<pre><div class="codehilite"><pre><span></span><span class="cp">#include</span> <span class="cpf"><iostream></span><span class="cp"></span>
<span class="cp">#include</span> <span class="cpf"><thread></span><span class="cp"></span>
<span class="kt">void</span> <span class="nf">foo</span><span class="p">()</span>
<span class="p">{</span>
<span class="n">std</span><span class="o">::</span><span class="n">cout</span> <span class="o"><<</span> <span class="s">"foo"</span> <span class="o"><<</span><span class="n">std</span><span class="o">::</span><span class="n">endl</span><span class="p">;</span>
<span class="p">}</span>
<span class="kt">int</span> <span class="nf">main</span><span class="p">(</span><span class="kt">int</span> <span class="n">argc</span><span class="p">,</span> <span class="kt">char</span> <span class="o">*</span><span class="n">argv</span><span class="p">[])</span>
<span class="p">{</span>
<span class="k">auto</span> <span class="n">t</span> <span class="o">=</span> <span class="n">std</span><span class="o">::</span><span class="kr">thread</span><span class="p">(</span><span class="n">foo</span><span class="p">);</span>
<span class="n">sched_param</span> <span class="n">sch</span><span class="p">;</span>
<span class="kt">int</span> <span class="n">policy</span><span class="p">;</span>
<span class="n">pthread_getschedparam</span><span class="p">(</span><span class="n">t</span><span class="p">.</span><span class="n">native_handle</span><span class="p">(),</span> <span class="o">&</span><span class="n">policy</span><span class="p">,</span> <span class="o">&</span><span class="n">sch</span><span class="p">);</span>
<span class="n">sch</span><span class="p">.</span><span class="n">sched_priority</span> <span class="o">=</span> <span class="mi">20</span><span class="p">;</span>
<span class="n">pthread_setschedparam</span><span class="p">(</span><span class="n">t</span><span class="p">.</span><span class="n">native_handle</span><span class="p">(),</span> <span class="n">SCHED_FIFO</span><span class="p">,</span> <span class="o">&</span><span class="n">sch</span><span class="p">);</span>
<span class="n">t</span><span class="p">.</span><span class="n">join</span><span class="p">();</span>
<span class="k">return</span> <span class="mi">0</span><span class="p">;</span>
<span class="p">}</span>
</pre></div>
</pre>
<p>Это возможно благодаря наличию метода <codeinline>.native_handle()</codeinline> у <codeinline>std::thread</codeinline>, значение которого можно использовать в POSIX системах. Использование этого метода полезно всегда, когда не хватает функциональности ни <codeinline>std::async</codeinline> ни <codeinline>std::thread</codeinline>. Использование <codeinline>std::async</codeinline> скрывает детали реализации и непригодно для такой тонкой работы.</p>
<h2>#18 Пренебрежение анализом нагрузки на CPU</h2>
<p>В любой момент времени потоки можно разделить на две группы - те которые что-то делают и те, который спят.</p>
<p>Потоки которые что-то делают занимают ядра процессора на которые их отправил планировщик. Для потоков которые занимаются активными вычислениями важно иметь в своём распоряжении свободные ядра, в противном случае большое количество потоков не даст никакого прироста в производительности. Даже скорее наоборот снизит производительность за счёт дополнительных переключений контекста потока.</p>
<p>Потоки которые преимущественно находятся в режиме ожидания в таком внимании процессора не нуждаются и могут находится в системе в гораздо большем количестве. И в случае с вводом/выводом даже помогают увеличить пропускную способность.</p>
<p>Я рассмотрел два крайних варианта, но наш конечно же будет посередине. Да, есть метод <codeinline>std::thread::hardware_concurrency()</codeinline>, которая сообщит нам сколько ядер доступно планировщику с учётом физических и логических.</p>
<p>Но это не помогает ответить правильно на вопрос - сколько же потоков можно запустить одновременно? Число ядер помогает понять сколько одновременно потоков, которые непрерывно длительное время активно потребляют процессор.</p>
<p>Если сценарий использования вычислений именно такой,то количество потоков должно быть как можно ближе к количеству ядер, а то и меньше, чтобы исключить распределение на логических ядрах.</p>
<p>Если потоки преимущественно заблокированы мьютексами или вводом/выводом, то ограничивать количество потоков ради экономии процессора не имеет большого смысла.</p>
<p>В остальных случаях необходимо нагрузочное тестирование и мониторинг с регулировкой количества потоков. Иными словами подбирается экспериментально.</p>
<h2>#19 Использование квалификатора volatile для синхронизации</h2>
<p>Использование этого квалификатора указание компилятору того, что изменения объекта могут происходить без контроля на этапе компиляции. Не в том смысле, что они происходят из разных потоков, а в том, что вообще за пределами кода, грубо говоря самопроизвольно. Это очень низкоуровневое указание и это никак не помогает в одновременном доступе внутри процесса.</p>
<p>Для синхронизации следует использовать <codeinline>atomic</codeinline>, <codeinline>mutex</codeinline>, и <codeinline>condition_variable</codeinline>.</p>
<h2>#20 Неоправданное использование lock-free алгоритмов</h2>
<p>Программирование без необходимости блокировок звучит очень привлекательно в сравнении с обычными механизмами синхронизации. Возможно, в случае жёсткого ограничения вычислительных ресурсов применение подобных алгоритмов может быть оправдано. В остальных случаях выглядит скорее преждевременной оптимизацией, которая, к тому же, может обернуться сложными ошибками в самое неподходящее время (и без coredump тут не обойтись).</p>
<p>Прежде чем приступить к использованию свободных от блокировок алгоритмов следует подумать над тремя вопросами:</p>
<ol>
<li>Пробовали ли спроектировать код без необходимости синхронизации?</li>
<li>Выполняли ли анализ производительности, поиск и оптимизацию узких мест?</li>
<li>Можно ли ограничиться горизонтальным масштабированием?</li>
</ol>
<p>Использование lock-free алгоритмов оправдано, когда никаких других решений просто не осталось.</p>
<hr>
<p>Надеюсь, чтение этого материала было так же полезно, как его перевод и публикация.</p>
</div>
</article>
</div>
<div class="blog__tile blog__tile_slim">
<div class="blog-post__footer">
<div class="blog-post__footer-item blog-post__footer-item_right">
<div class="inline-block">
<div class="blog-post__footer-text blog-post__footer-text_share">Поделиться</div>
<div class="blog-post__footer-icons">
<div class="blog-post__footer-icon-box hover-ic js-share" data-href="https://otus.ru/nest/post/145/" data-type="tg">
<div class="blog-post__footer-icon ic ic-tg-square ic-tg-square-hover"></div>
</div>
<div class="blog-post__footer-icon-box hover-ic js-share" data-href="https://otus.ru/nest/post/145/" data-type="tw">
<div class="blog-post__footer-icon ic ic-twitter-square ic-twitter-square-hover"></div>
</div>
<div class="blog-post__footer-icon-box hover-ic js-share" data-href="https://otus.ru/nest/post/145/" data-type="vk">
<div class="blog-post__footer-icon ic ic-vk-square ic-vk-square-hover"></div>
</div>
</div>
</div>
<div class="inline-block float-right_ssm">
<div class="blog__counters-item blog__counters-item_bookmark">
<a href="#"
class="js-blog-post-mark blog-post__footer-icon blog-post__footer-icon_star"
post_id=145></a>
</div>
</div>
</div>
<div class="blog-post__footer-item">
<div class="blog-vote">
<div class="blog-vote__item blog-vote__item_border blog-vote__item_left js-blog-post-dislike"
data-url="/nest/vote/post/" data-id=145></div>
<div class="blog-vote__item blog-vote__item_border blog-vote__item_center blog-vote__item_pos js-post-votes"
data-url="/nest/vote/post/" data-id=145>
7
</div>
<div class="blog-vote__item blog-vote__item_border blog-vote__item_right js-blog-post-like"
data-url="/nest/vote/post/" data-id=145></div>
</div>
</div>
<div class="inline-block float-right_ssm">
<a class="blog-post__footer-item" href="/nest/post/145/#comments">
<div class="blog-post__footer-icon blog-post__footer-icon_comments"></div>
<div class="blog-post__footer-text js-blog-post-comments-counter">0</div>
</a>
<div class="blog-post__footer-item">
<div class="blog-post__footer-icon blog-post__footer-icon_views"></div>
<div class="blog-post__footer-text">304</div>
</div>
</div>
</div>
</div>
</div>
<div
id="formSubscribe"
class="blog-subscribe__container js-subscribe-container"
>
<form
method="post"
novalidate
action="/nest/blog/subscribe/"
class="js-ajax-form js-new-validation blog-subscribe blog__external"
data-before-send='[
{"action": "addClass", "el": "#formSubscribe .js-content", "class": "hidden"},
{"action": "show", "el": "#formSubscribe .js-loader"}
]'
data-after-complete='[
{"action": "hide", "el": "#formSubscribe .js-loader"}
]'
data-after-error='[{"action": "removeClass", "el": "#formSubscribe .js-content", "class": "hidden"}]'
data-after-success='[
{"action": "addClass", "el": "#formSubscribe .js-block-common", "class": "hidden"},
{"action": "show", "el": "#formSubscribe .js-block-success"},
{"action": "addClass", "el": "#formSubscribe.js-subscribe-container", "class": "blog-subscribe__container_finish"}
]'
data-success-msg="false"
>
<div class="js-block-common">
<p class="blog-subscribe__title">Не пропустите новые полезные статьи!</p>
<div class="blog-subscribe__fields">
<input type="hidden" name="csrfmiddlewaretoken" value="lXYhBGOxOepaErVDt9EvPCNH9v8oLtyRgmCExQIGsHTWoKQEjexlmkk5IShqMan6">
<input type="hidden" name="object_id" value="6"/>
<div class="js-content">
<div class="new-input-line blog-subscribe__input-line">
<div class="new-input-group new-input-group_right blog-subscribe__group">
<div class="new-input new-input_fake new-input_full">
<input
type="email"
class="new-input new-input_full js-placeholder new-input_border-no blog-subscribe__input"
data-title="email"
autocomplete="email"
name="email"
required
placeholder="Введите ваш email"
/>
<div class="new-input__error-sign new-ic new-ic-warning js-new-input-error">
<div class="new-input__error-text-container">
<p class="new-input__error-text js-new-input-error-text"></p>
</div>
</div>
</div>
<button
type="submit"
class="new-input-group__addon new-input-group__addon_button new-button new-button_blue blog-subscribe__button"
>
Подписаться
</button>
</div>
</div>
<div class="new-input-line new-input-line_last">
<label class="checkbox checkbox_new">
<input required type="checkbox" checked name="subscription_agree" value="true">
<div class="checkbox__label blog-subscribe__checkbox-text">
Соглашаюсь получать полезные новости, статьи,
приглашения на мастер-классы и специальные предложения OTUS
</div>
</label>
<div class="new-input__error-sign new-ic new-ic-warning js-new-input-error">
<div class="new-input__error-text-container">
<p class="new-input__error-text js-new-input-error-text"></p>
</div>
</div>
</div>
</div>
<div class="hide js-loader">
<i class="ic loader loader_md loader_absolute-center ic-loader"></i>
</div>
</div>
</div>
<div class="js-block-success blog-subscribe__success hide">
<p class="blog-subscribe__title">Спасибо за подписку!</p>
<p class="blog-subscribe__text text-center">
Мы отправили вам письмо для подтверждения вашего email.
<br/>
С уважением, OTUS!
</p>
</div>
</form>
</div>
<div class="blog__tile" id="author_block">
<div class="blog__h2">Автор</div>
<div class="post-info">
<a href="/profile/4868/">
<div class="post-info__avatar post-info__avatar_big ic ic-blog-default-avatar" style="background-image: url(https://cdn.otus.ru/media/public/a9/67/avatar-4868-a967e9.png);"></div>
<div class="post-info__author">Дмитрий</div>
</a>
<div class="post-info__text">
Рейтинг:
<div class="post-info__text-ratingprofile__rating_pos">
+102
</div>
</div>
<div class="post-info__text">
3095 дней
</div>
<div class="post-info__button">
</div>
</div>
</div>
<div class="blog__comments blog__comments-blog-default">
<div data-comments>
<div class="blog__h2">
<div class="text text_inline text_default js-blog-comments-counter">
0 комментариев
</div>
</div>
<div class="js-comments">
</div>
<div class="blog-comment-deny">
Для комментирования необходимо <a href="/login/?next=https%3A%2F%2Fotus.ru%2Fnest%2Fpost%2F145%2F">авторизоваться</a>
</div>
</div>
</div>
</div>
<div class="container__col container__col_4 container__col_md-0">
<div class="blog-tile-wrapper">
<div class="blog-tile">
<div class="blog-tile__title">Популярное</div>
<div class="blog-tile__item blog-tile__item_last">Сегодня тут пусто</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="blog-footer">
<div class="container cookies__container">
<div class="cookies__margin-block cookies_hide js-cookie">
<div class="cookies">
<div class="cookies__title">
Посещая наш сайт, вы принимаете <a class="cookies__link" href="/legal/cookie/" target="_blank">политику использования cookie-файлов</a>
</div>
<button class="js-cookie-accept cookies__button">ОК</button>
</div>
</div>
</div>
<footer class="footer2 footer2_header3 footer2_desktop footer2_not-subscribed no-print ">
<div class="footer2__container container">
<div class="footer2__container-box">
<div class="footer2__content">
<div class="container__row">
<div class="container__col container__col_4">
<div class="container__row">
<div class="container__col container__col_6 container__col_md-5">
<div class="footer2__links">
<div class="footer2__links-row">
<a href="/about" class="footer2__link" title="О нас">О нас</a>
</div>
<div class="footer2__links-row">
<a href="/smi/" class="footer2__link" title="СМИ о нас">СМИ о нас</a>
</div>
<div class="footer2__links-row">
<a href="/reviews" class="footer2__link" title="Отзывы">Отзывы</a>
</div>
<div class="footer2__links-row">
<a href="/contacts/" class="footer2__link" title="Контакты">Контакты</a>
</div>
<div class="footer2__links-row">
<a href="/journal/" class="footer2__link" title="Блог">Блог</a>
</div>
<div class="footer2__links-row">
<a href="/faq/" class="footer2__link" title="FAQ">FAQ</a>
</div>
</div>
</div>
<div class="container__col container__col_6 container__col_md-7">
<div class="footer2__icons">
<div class="footer2__social">
<a href="https://vk.com/club145052891" target="_blank" rel="noreferrer nofollow" class="footer2__icon ic ic-vk-footer2 ic-vk-footer2-hover"></a>
<a href="https://zen.yandex.ru/id/5bbcbc1ba5bd5400a990e7d9" target="_blank" rel="noreferrer nofollow" class="footer2__icon ic ic-zen ic-zen-hover"></a>
<a href="https://www.youtube.com/channel/UCetgtvy93o3i3CvyGXKFU3g" target="_blank" rel="noreferrer nofollow" class="footer2__icon ic ic-yt-footer2 ic-yt-footer2-hover"></a>
</div>
<a href="https://ttttt.me/Otusjava" target="_blank" rel="noreferrer nofollow" class="footer2__link-extended hover-ic">
<div class="footer2__icon ic ic-inline ic-tlgrm-footer2 ic-tlgrm-footer2-hover"></div>
<p class="footer2__link-extended-text">Канал в Telegram</p>
</a>
<a href="https://ttttt.me/joinchat/JMakp0NXc-L8nNneHCtx7A" target="_blank" rel="noreferrer nofollow" class="footer2__link-extended hover-ic" >
<div class="footer2__icon ic ic-inline ic-tlgrm-footer2 ic-tlgrm-footer2-hover"></div>
<p class="footer2__link-extended-text">Группа в Telegram</p>
</a>
</div>
</div>
</div>
</div>
<div class="container__col container__col_4">
<div class="footer2__links footer2__links_center">
<div class="footer2__links-row">
<a href="/b2b" class="footer2__link" rel="nofollow" title="Корпоративное обучение">
Корпоративное обучение
</a>
</div>
<div class="footer2__links-row">
<a href="/lessons/" class="footer2__link" title="Каталог курсов">
Каталог курсов
</a>
</div>
<div class="footer2__links-row">
<a href="/about/loyalty/" class="footer2__link" title="Программы лояльности">Программы лояльности</a>
</div>
<div class="footer2__links-row">
<a href="/professions/" class="footer2__link" title="Каталог профессий">Каталог профессий</a>
</div>
<div class="footer2__links-row">
<a href="/employers/all/" class="footer2__link" title="Наши партнеры">Наши партнеры</a>
</div>
<div class="footer2__links-row">
<a href="/teach/" class="footer2__link" title="Стать преподавателем">
Стать преподавателем
</a>
</div>
</div>
</div>
<div class="container__col container__col_4">
<p class="footer2__text footer2__text_margin-bot">Подписка на новости IT, анонсы открытых уроков, спец. предложения</p>
<form method="post" class="footer2__subscribe js-subscribe" action="/lessons/subscribe/">
<input type="hidden" name="csrfmiddlewaretoken" value="lXYhBGOxOepaErVDt9EvPCNH9v8oLtyRgmCExQIGsHTWoKQEjexlmkk5IShqMan6">
<input
required
type="email"
name="email"
class="input footer2__subscribe-input"
placeholder="Электронная почта"
value=""
/>
<button
class="footer2__subscribe-button button button_blue button_as-input"
type="submit"
disabled
>
Подписаться
</button>
<div class="new-input-line new-input-line_relative new-input-line_triple footer2__subscribe-policy">
<label class="new-checkbox new-checkbox_vertical-center new-log-reg__checkbox">
<input type="checkbox" checked name="terms_agree" value="true"
class="js-remove-field-error">
<div class="new-checkbox__label">
Я принимаю условия
<a
class="new-link-dotted-blue"
target="_blank"
href="/legal/privacy/"
>
Политики обработки персональных данных
</a>
и
<a
class="new-link-dotted-blue"
target="_blank"
href="/legal/terms/"
>
Пользовательского соглашения
</a>
и даю
<a target="_blank" class="new-link-dotted-blue" href="
/legal/lead_privacy_agree/"
">
свое согласие на обработку персональных данных
</a>
</div>
</label>
</div>
</form>
<p class="footer2__text footer2__text_margin-bot">
По всем вопросам пишите на
<a class="footer2__link" href="mailto:help@otus.ru" target="_blank" rel="nofollow noreferer" title="help@otus.ru">help@otus.ru</a>
</p>
<p class="footer2__text footer2__text_margin-bot">
<a
class="footer2__link"
href="/legal/common/"
target="_blank"
rel="nofollow noreferer"
title="Сведения об образовательной организации"
>
Сведения об образовательной организации
</a>
</p>
<p class="footer2__text footer2__text_margin-bot">
<a
class="footer2__link"
href="/legal/it_company_accreditation/"
target="_blank"
rel="nofollow noreferer"
title="OTUS является аккредитованной IT-компанией"
>
OTUS является аккредитованной IT-компанией
</a>
</p>
<p class="footer2__text footer2__text_margin-bot">
<a
class="footer2__link"
href="/legal/recommendations/"
target="_blank"
rel="nofollow noreferer"
title="Сведения о рекомендательных технологиях"
>
Сведения о рекомендательных технологиях
</a>
</p>
<p class="footer2__text footer2__text_margin-bot">
<a
class="footer2__link"
href="https://reestr.digital.gov.ru/reestr/2704482/"
target="_blank"
rel="nofollow noreferer"
title="В реестре отечественного ПО №24216"
>
В реестре отечественного ПО №24216
</a>
</p>
</div>
</div>
</div>
<div class="footer2__info">
<div class="container__row">
<div class="container__col container__col_bottom container__col_8">
<div class="container__row">
<div class="container__col container__col_3 container__col_md-7">
<p
class="footer2__text footer2__text_nowrap footer2__text_margin-right "
>
© 2015-2026 OTUS
</p>
</div>
<div class="container__col container__col_3 container__col_md-5">
<a
class="footer2__link footer2__link-white-space-normal"
href="/legal/terms/"
title="Условия использования сервиса"
>
Условия использования сервиса
</a>
</div>
</div>
</div>
<div class="container__col container__col_middle container__col_4">
<div class="footer2__info-logos footer2__info-logos_desktop">
<div class="footer2__info-logos-row">
<div class="footer2__info-logo-box">
<div class="runet__wrapper">
<div class="runet">
<div class="runet__text">Премия Рунета <br>2018</div>
</div>
</div>
</div>
<div class="footer2__info-logo-box">
<a class="footer2__info-logo footer2__info-logo_skolkovo" href="http://sk.ru/?utm_source=otus" target="_blank" rel="noreferrer nofollow"></a>
</div>
</div>
<div class="footer2__info-logos-row">
<div class="footer2__info-logo-box footer2__info-logo-box_not-first footer2__info-logo-box_kts">
<a class="footer2__info-logo footer2__info-logo_kts" href="https://ktsstudio.com/?utm_source=otus" target="_blank" rel="noreferrer nofollow"></a>
</div>
<div class="footer2__info-logo-box footer2__info-logo-box_not-first">
<a class="footer2__info-logo footer2__info-logo_uno" href="http://goodlookin.ru/?utm_source=otus" target="_blank" rel="noreferrer nofollow"></a>
</div>
</div>
<div class="footer2__info-logos-row">
<div class="footer2__info-logo-box"></div>
<div class="footer2__info-logo-box">
<div class="runet-mobile">
<div class="runet-mobile__text">Премия Рунета <br>2018</div>
</div>
</div>
<div class="footer2__info-logo-box"></div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</footer>
<div class="footer2 footer2_mobile no-print ">
<div class="container footer2-mobile__container">
<div class="footer2-mobile__wrapper">
<div class="footer2-mobile__row">
<div class="container__row">
<div class="container__col container__col_12">
<p class="footer2__text footer2__text_margin-bot">Подписка на новости IT, анонсы открытых уроков, спец. предложения</p>
<form method="post" class="footer2__subscribe js-subscribe" action="/lessons/subscribe/">
<input type="hidden" name="csrfmiddlewaretoken" value="lXYhBGOxOepaErVDt9EvPCNH9v8oLtyRgmCExQIGsHTWoKQEjexlmkk5IShqMan6">
<input
required
type="email"
name="email"
class="input footer2__subscribe-input"
placeholder="Электронная почта"
value=""
/>
<button
class="footer2__subscribe-button button button_blue button_as-input"
type="submit"
disabled
>
Подписаться
</button>
<div class="new-input-line new-input-line_relative new-input-line_triple footer2__subscribe-policy">
<label class="new-checkbox new-checkbox_vertical-center new-log-reg__checkbox">
<input type="checkbox" checked name="terms_agree" value="true"
class="js-remove-field-error">
<div class="new-checkbox__label">
Я принимаю условия
<a
class="new-link-dotted-blue"
target="_blank"
href="/legal/privacy/"
>
Политики обработки персональных данных
</a>
и
<a
class="new-link-dotted-blue"
target="_blank"
href="/legal/terms/"
>
Пользовательского соглашения
</a>
и даю
<a target="_blank" class="new-link-dotted-blue" href="
/legal/lead_privacy_agree/"
">
свое согласие на обработку персональных данных
</a>
</div>
</label>
</div>
</form>
<p class="footer2__text footer2__text_margin-bot">
По всем вопросам пишите на
<a class="footer2__link"
href="mailto:help@otus.ru"
target="_blank" rel="nofollow noreferer"
title="help@otus.ru">help@otus.ru</a>
</p>
<p class="footer2__text footer2__text_margin-bot">
<a class="footer2__link"
href="/legal/common/"
target="_blank" rel="nofollow noreferer"
title="Сведения об образовательной организации"
>
Сведения об образовательной организации
</a>
</p>
<p class="footer2__text footer2__text_margin-bot">
<a
class="footer2__link"
href="/legal/it_company_accreditation/"
target="_blank"
rel="nofollow noreferer"
title="OTUS является аккредитованной IT-компанией"
>
OTUS является аккредитованной IT-компанией
</a>
</p>
<p class="footer2__text footer2__text_margin-bot">
<a
class="footer2__link"
href="/legal/recommendations/"
target="_blank"
rel="nofollow noreferer"
title="Сведения о рекомендательных технологиях"
>
Сведения о рекомендательных технологиях
</a>
</p>
<p class="footer2__text footer2__text_margin-bot">
<a
class="footer2__link"
href="https://reestr.digital.gov.ru/reestr/2704482/"
target="_blank"
rel="nofollow noreferer"
title="В реестре отечественного ПО №24216"
>
В реестре отечественного ПО №24216
</a>
</p>
</div>
</div>
</div>
<div class="footer2-mobile__row footer2-mobile__row_social">
<div class="container__row">
<div class="container__col container__col_7 footer2-mobile__col_tg">
<a href="https://ttttt.me/Otusjava" target="_blank" rel="noreferrer nofollow" class="footer2__link-extended hover-ic">
<div class="footer2__icon ic ic-inline ic-tlgrm-footer2 ic-tlgrm-footer2-hover"></div>
<p class="footer2__link-extended-text">Канал в Telegram</p>
</a>
<a href="https://ttttt.me/joinchat/JMakp0NXc-L8nNneHCtx7A" target="_blank" rel="noreferrer nofollow" class="footer2__link-extended hover-ic" >
<div class="footer2__icon ic ic-inline ic-tlgrm-footer2 ic-tlgrm-footer2-hover"></div>
<p class="footer2__link-extended-text">Группа в Telegram</p>
</a>
</div>
<div class="container__col container__col_5 footer2-mobile__col_social">
<div class="footer2__social footer2__social_mobile">
<a href="https://vk.com/club145052891" target="_blank" rel="noreferrer nofollow" class="footer2__icon ic ic-vk-footer2 ic-vk-footer2-hover"></a>
<a href="https://zen.yandex.ru/id/5bbcbc1ba5bd5400a990e7d9" target="_blank" rel="noreferrer nofollow" class="footer2__icon ic ic-zen ic-zen-hover"></a>
<a href="https://www.youtube.com/channel/UCetgtvy93o3i3CvyGXKFU3g" target="_blank" rel="noreferrer nofollow" class="footer2__icon ic ic-yt-footer2 ic-yt-footer2-hover"></a>
</div>
</div>
</div>
</div>
<div class="footer2-mobile__row">
<div class="container__row">
<div class="container__col container__col_6 container__col_xs375-8">
<div class="footer2__links ">
<div class="footer2__links-row">
<a href="/b2b" class="footer2__link" rel="nofollow" title="Корпоративное обучение">
Корпоративное обучение
</a>
</div>
<div class="footer2__links-row">
<a href="/lessons/" class="footer2__link" rel=nofollow title="Каталог курсов">
Каталог курсов
</a>
</div>
<div class="footer2__links-row">
<a href="/about/loyalty/" class="footer2__link" rel=nofollow title="Программы лояльности">Программы лояльности</a>
</div>
<div class="footer2__links-row">
<a href="/professions/" class="footer2__link" rel=nofollow title="Каталог профессий">Каталог профессий</a>
</div>
<div class="footer2__links-row">
<a href="/employers/all/" class="footer2__link" rel=nofollow title="Наши партнеры">Наши партнеры</a>
</div>
<div class="footer2__links-row">
<a href="/teach/" class="footer2__link" rel=nofollow title="Стать преподавателем">
Стать преподавателем
</a>
</div>
</div>
</div>
<div class="container__col container__col_6 container__col_xs375-4">
<div class="footer2__links">
<div class="footer2__links-row">
<a href="/about" class="footer2__link" rel=nofollow title="О нас">О нас</a>
</div>
<div class="footer2__links-row">
<a href="/smi/" class="footer2__link" rel=nofollow title="СМИ о нас">СМИ о нас</a>
</div>
<div class="footer2__links-row">
<a href="/reviews" class="footer2__link" rel=nofollow title="Отзывы">Отзывы</a>
</div>
<div class="footer2__links-row">
<a href="/contacts/" class="footer2__link" rel=nofollow title="Контакты">Контакты</a>
</div>
<div class="footer2__links-row">
<a href="/journal/" class="footer2__link" rel=nofollow title="Блог">Блог</a>
</div>
<div class="footer2__links-row">
<a href="/faq/" class="footer2__link" rel=nofollow title="FAQ">FAQ</a>
</div>
</div>
</div>
</div>
</div>
<div class="footer2-mobile__row footer2-mobile__row_mark">
<div class="footer2-mobile__mark">
<p
class="footer2__text footer2__text_nowrap footer2__text_margin-right footer2__text_margin-right"
>
© 2015-2026 OTUS
</p>
<a
class="footer2__link"
href="/legal/terms/"
title="Пользовательское соглашение"
>
Пользовательское соглашение
</a>
</div>
<div class="footer2-mobile__logos">
<div class="footer2__info-logos footer2__info-logos_tablet">
<div class="footer2__info-logos-row_mobile">
<div class="footer2__info-logo-box">
<a class="footer2__info-logo footer2__info-logo_skolkovo" href="http://sk.ru/?utm_source=otus" target="_blank" rel="noreferrer nofollow"></a>
</div>
<div class="footer2__info-logo-box">
<div class="runet-mobile">
<div class="runet-mobile__text">Премия Рунета <br>2018</div>
</div>
</div>
<div class="footer2__info-logo-box footer2__info-logo-box_not-first footer2__info-logo-box_kts">
<a class="footer2__info-logo footer2__info-logo_kts" href="https://ktsstudio.com/?utm_source=otus" target="_blank" rel="noreferrer nofollow"></a>
</div>
<div class="footer2__info-logo-box footer2__info-logo-box_not-first">
<a class="footer2__info-logo footer2__info-logo_uno" href="http://goodlookin.ru/?utm_source=otus" target="_blank" rel="noreferrer nofollow"></a>
</div>
</div>
</div>
<div class="footer2__info-logos footer2__info-logos_mobile">
<div class="footer2__info-logos-row">
<div class="footer2__info-logo-box">
<div class="runet-mobile">
<div class="runet-mobile__text">Премия Рунета <br>2018</div>
</div>
</div>
<div class="footer2__info-logo-box">
<a class="footer2__info-logo footer2__info-logo_skolkovo" href="http://sk.ru/?utm_source=otus" target="_blank" rel="noreferrer nofollow"></a>
</div>
</div>
<div class="footer2__info-logos-row">
<div class="footer2__info-logo-box footer2__info-logo-box_not-first footer2__info-logo-box_kts">
<a class="footer2__info-logo footer2__info-logo_kts" href="https://ktsstudio.com/?utm_source=otus" target="_blank" rel="noreferrer nofollow"></a>
</div>
<div class="footer2__info-logo-box footer2__info-logo-box_not-first">
<a class="footer2__info-logo footer2__info-logo_uno" href="http://goodlookin.ru/?utm_source=otus" target="_blank" rel="noreferrer nofollow"></a>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="modal-container hide-top hide-transparent new-log-reg-container"
data-query='{"next": "/nest/post/145/"}'
data-modal-ajax="/login-registration/"
data-modal-ajax-once="true"
data-modal-id="new-log-reg"
data-modal-hide-remove-class="hide-top,hide-transparent"
data-modal-hide-add-class="hide-transparent"
data-modal-loading="true"
data-modal-hide-add-delay-class="hide-top">
<div class="new-log-reg-loader">
<div class="new-log-reg__login">
<div class="loader loader_md loader_absolute-center ic-loader ic"></div>
</div>
</div>
<div class="new-log-reg-wrapper js-modal-content">
<div class="new-log-reg__login"></div>
</div>
</div>
<div class="modal-container hide-top hide-transparent new-log-reg-container"
data-query='{"next": "/nest/post/145/"}'
data-modal-ajax="/login-registration/"
data-modal-id="new-log-reg-event"
data-modal-hide-remove-class="hide-top,hide-transparent"
data-modal-hide-add-class="hide-transparent"
data-modal-loading="true"
data-modal-hide-add-delay-class="hide-top">
<div class="new-log-reg-loader">
<div class="new-log-reg__login">
<div class="loader loader_md loader_absolute-center ic-loader ic"></div>
</div>
</div>
<div class="new-log-reg-wrapper js-modal-content">
<div class="new-log-reg__login"></div>
</div>
</div>
<div class="modal-container hide" data-modal-id="restore-password">
<div class="modal-wrapper modal-wrapper_dark js-modal-wrapper">
<div class="modal new-log-reg__popup">
<div class="new-log-reg__popup-body">
<div class="modal__close new-ic new-ic-close-inverse new-log-reg__popup-close js-close-modal"></div>
<p class="new-log-reg__popup-title">Восстановление пароля</p>
<form method="post" class="js-restore-password" action="/api/restore_password.send_email">
<input type="hidden" name="csrfmiddlewaretoken" value="lXYhBGOxOepaErVDt9EvPCNH9v8oLtyRgmCExQIGsHTWoKQEjexlmkk5IShqMan6">
<div class="new-log-reg__popup-text new-log-reg__popup-text_slim">
Введите электронную почту для восстановления пароля
</div>
<div class="new-input-line new-input-line_slim new-input-line_relative">
<input type="email"
class="new-input new-input_full js-placeholder js-input
js-required
"
maxlength=""
name="email"
autocomplete="off"
required
placeholder="Электронная почта"
/>
<div class="new-input-error new-input-error_bottom js-validation-error hide "></div>
<div class="new-input-error new-input-error_info new-input-error_bottom hide"></div>
</div>
<div class="new-input-line js-error hide">
<span class="new-input-line__error new-log-reg__popup-error">
Пользователь с таким email не найден
</span>
</div>
<div class="new-input-line">
<button class="new-button new-button_md new-button_full new-button_blue"
type="submit">Восстановить
</button>
</div>
</form>
</div>
</div>
</div>
</div>
<div class="modal-container hide" data-modal-id="phone_duplicate">
<div class="modal-wrapper modal-wrapper_dark js-modal-wrapper" data-no-wrapper-close="true">
<div class="modal new-log-reg__popup new-log-reg__popup-body">
<button class="new-log-reg__popup-close new-ic-close-inverse js-close-modal js-stats"
data-event="Register;double_tel_bad"
style=""></button>
<form>
<p class="new-log-reg__popup-title">
Ваш телефон уже привязан к<br/>другой учетной записи
</p>
<p class="new-log-reg__popup-text">
Ваш телефон <b><nobr class="js-phone-duplicate-phone"></nobr></b> уже
привязан к учетной записи <b><nobr class="js-phone-duplicate-acc"></nobr></b>.
Выберите учетную запись, с которой желаете продолжить работу.
Мы привяжем к ней телефон.
</p>
<input class="js-dod-phone-dup-modal-phone" type="hidden" value="" name="phone">
<div class="dod-phone-dup-modal__loader-container">
<div class="js-loader-content">
<div class="new-input-line">
<button
class="new-button new-button_md new-button_full new-button_one-line
new-button_blue-inverse js-stats js-dod-phone-dup-modal-submit"
type="submit"
data-value="1"
data-event="Register;double_tel_ok"
>
Текущий аккаунт <nobr class="js-phone-duplicate-self-acc"></nobr>
</button>
</div>
<div class="new-input-line">
<button
class="new-button new-button_md new-button_full new-button_one-line
new-button_blue js-stats js-dod-phone-dup-modal-submit"
type="submit"
data-value="0"
data-event="Register;double_tel_ok"
>
Войти в аккаунт <nobr class="js-phone-duplicate-acc"></nobr>
</button>
</div>
<div class="new-input-line js-error hide">
<span class="new-input-line__error new-log-reg__popup-error js-error-text"></span>
</div>
</div>
<div
class="js-loader loader loader_absolute-center loader_md ic ic-loader"
style="display: none;"
></div>
</div>
</form>
</div>
</div>
</div>
<div class="modal-container hide" data-modal-id="teacher-enrollment">
<div class="modal-wrapper modal-wrapper_dark js-modal-wrapper" data-no-wrapper-close="true">
<div class="modal new-log-reg__popup">
<form class="new-log-reg__popup-body js-email-terms-update js-form-in-modal"
action="/teacher/request/"
data-modal="teacher-enrollment">
<div class="modal__close new-ic new-ic-close-inverse new-log-reg__popup-close js-close-modal"></div>
<div class="new-log-reg__popup-title">
Заполните номер телефона
</div>
<div class="new-log-reg__popup-text new-log-reg__popup-text_slim">
Для отправки заявки в преподаватели заполните номер телефона
</div>
<div class="new-input-line new-input-line_slim new-input-line_relative">
<input type="text"
class="new-input new-input_full js-placeholder js-input
js-required
"
maxlength="255"
name="phone"
autocomplete="off"
data-js-mask="phone"
required
placeholder="Телефон"
/>
<div class="new-input-error new-input-error_bottom js-validation-error hide "></div>
<div class="new-input-error new-input-error_info new-input-error_bottom hide"></div>
</div>
<div class="new-input-line js-error hide">
<span class="new-input-line__error new-log-reg__popup-error js-text"></span>
</div>
<div class="new-input-line">
<button class="new-button new-button_md new-button_full new-button_blue"
type="submit">Отправить
</button>
</div>
</form>
</div>
</div>
</div>
<script src="https://otus.ru/static/js/common.792bd.js" ></script>
<script src="https://otus.ru/static/js/vendor.common.2ae83.js" ></script>
<script src="https://otus.ru/static/js/vendor.otus.90c85.js" ></script>
<script src="https://otus.ru/static/js/otus.714df.js" ></script>
<script src="https://otus.ru/static/js/vendor.react.3b8bf.js" ></script>
<script src="https://otus.ru/static/js/vendor.common.2ae83.js" ></script>
<script src="https://otus.ru/static/js/vendor.otus-react:header-search.706a9.js" ></script>
<script src="https://otus.ru/static/js/otus-react:header-search.34d9b.js" ></script>
<script type="text/javascript">(window.Image ? (new Image()) : document.createElement('img')).src = 'https://vk.com/rtrg?r=w1EsimsoVZRm*rb04TRjaay1IST5bHFNpthXLyHrq1GCPGzgOmMRcY3mDDdxA17TQOlx6ykxwlrtBtW7sMri/2f364oz1QGDuVvxHFqMqER5NT9mlhp1lYbEMKPIB7NgDRQQp5s2IxBuu*caPsHzTtfILgZJsV5bbkH0m8*GleA-&pixel_id=1000094479';</script>
<script type="text/javascript">(window.Image ? (new Image()) : document.createElement('img')).src = 'https://vk.com/rtrg?p=VK-RTRG-182645-Hs1B';</script>
<script type="text/javascript">(window.Image ? (new Image()) : document.createElement('img')).src = 'https://vk.com/rtrg?p=VK-RTRG-110005-dVnpE';</script>
<script type="text/javascript">(window.Image ? (new Image()) : document.createElement('img')).src = 'https://vk.com/rtrg?p=VK-RTRG-226047-a8wzo';</script>
<noscript>
<div>
<img src="https://mc.yandex.ru/watch/34531570" style="position:absolute; left:-9999px; top: 0;" alt=""/>
<img src="https://mc.yandex.ru/watch/82755226" style="position:absolute; left:-9999px; top: 0;" alt=""/>
<img src="https://mc.yandex.ru/watch/93715742" style="position:absolute; left:-9999px; top: 0;" alt=""/>
</div>
</noscript>
<img height="1" width="1" src="https://happy.otus.ru/pixel/otus.gif?jwt=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJyZWZlcmVyIjoiaHR0cHM6Ly9vdHVzLnJ1L25lc3QvcG9zdC8xNDUvP3V0bV9zb3VyY2U9dHlwZWluJnV0bV9tZWRpdW09ZGlyZWN0JnV0bV9jYW1wYWlnbj1Ob25lIn0.xiOu8RiKsS1iCyRnwDGJ49arN5iW3tS96N6wsPFvrW0" style="position:absolute; left:-9999px; top: 0;" alt="" />
</body>
</html>