Boost.Serialization
В этом примере мы создаем объект
Boost.Serialization — это часть обширной библиотеки Boost и предоставляет мощные возможности для сериализации и десериализации объектов. Она поддерживает большинство типов данных C++, включая пользовательские типы, и может сериализовать данные в различные форматы, включая двоичный, текстовый и XML.Boost.Serialization может использоваться для сохранения состояния программы, передачи данных между процессами и постоянного хранения данных.В этом примере мы создаем объект
MyClass с именем obj1 и сохраняем его состояние в файле с помощью boost::archive::text_oarchive. Затем мы восстанавливаем объект obj2 из этого файла с помощью boost::archive::text_iarchive. В конце мы выводим значения полей obj2, чтобы убедиться, что состояние было правильно восстановлено.cereal
Библиотека
В этом примере мы создаем структуру
Библиотека
cereal для C++ — это гибкая и эффективная библиотека для сериализации, которая поддерживает множество форматов, включая бинарные, XML и JSON. Она может быть использована в различных областях, где требуется сохранение и восстановление состояния объектов. В этом примере мы создаем структуру
MyData и сохраняем ее состояние в файле с помощью cereal::BinaryOutputArchive. Затем мы восстанавливаем структуру из этого файла с помощью cereal::BinaryInputArchive. В конце мы выводим значения полей m2, чтобы убедиться, что состояние было правильно восстановлено.Uniform initialization
Uniform initialization — это способ инициализации переменных и объектов, который был введен в стандарте C++11. Он представляет собой универсальный и более предсказуемый способ инициализации, который использует фигурные скобки
Преимущества uniform initialization включают:
*Важно отметить, что uniform initialization не всегда работает так, как ожидается, особенно в случае с перегруженными конструкторами. В некоторых случаях, компилятор может выбрать не тот конструктор, который вы ожидали, что может привести к неожиданному поведению.
Uniform initialization — это способ инициализации переменных и объектов, который был введен в стандарте C++11. Он представляет собой универсальный и более предсказуемый способ инициализации, который использует фигурные скобки
{} вместо круглых () или присваивания =.Преимущества uniform initialization включают:
- Предотвращает узкое преобразование (narrowing conversion), которое может привести к потере данных.- Обеспечивает одинаковый синтаксис для инициализации всех типов данных и структур.- Позволяет инициализировать объекты, которые ранее не могли быть инициализированы, такие как массивы и структуры.*Важно отметить, что uniform initialization не всегда работает так, как ожидается, особенно в случае с перегруженными конструкторами. В некоторых случаях, компилятор может выбрать не тот конструктор, который вы ожидали, что может привести к неожиданному поведению.
Aggregate initialization
Aggregate initialization — это форма инициализации, которая позволяет инициализировать агрегаты (объекты определенных типов) с использованием фигурных скобок и списка значений. Агрегаты могут быть одним из следующих типов:
В этом примере мы создаем структуру
Aggregate initialization — это форма инициализации, которая позволяет инициализировать агрегаты (объекты определенных типов) с использованием фигурных скобок и списка значений. Агрегаты могут быть одним из следующих типов:
- Массивы- Структуры или классы без пользовательских конструкторов, без закрытых или защищенных нестатических членов данных, без базовых классов и без виртуальных функций.В этом примере мы создаем структуру
Point, которая содержит два целочисленных поля x и y. Затем мы инициализируем объект p1 этой структуры с помощью aggregate initialization, указывая значения для x и y в фигурных скобках. Аналогично, мы инициализируем массив arr с помощью списка значений в фигурных скобках.Сырые указатели
Сырые указатели — это переменные, которые хранят адрес другой переменной. Они используются во многих ситуациях, включая динамическое выделение памяти, создание связанных структур данных (например, деревьев и связанных списков), и для работы с массивами.
В этом примере
Однако использование сырых указателей может быть опасным, поскольку они могут привести к ошибкам, таким как утечки памяти, разыменование нулевого указателя и разыменование висячего указателя. По этой причине в современном C++ рекомендуется использовать умные указатели, такие как
Сырые указатели — это переменные, которые хранят адрес другой переменной. Они используются во многих ситуациях, включая динамическое выделение памяти, создание связанных структур данных (например, деревьев и связанных списков), и для работы с массивами.
В этом примере
p — это сырой указатель на переменную x. Мы можем получить значение x через указатель, используя оператор разыменования *, и мы можем изменить значение x через указатель. Вывод программы показывает, что значение x действительно изменяется через указатель.Однако использование сырых указателей может быть опасным, поскольку они могут привести к ошибкам, таким как утечки памяти, разыменование нулевого указателя и разыменование висячего указателя. По этой причине в современном C++ рекомендуется использовать умные указатели, такие как
std::unique_ptr, std::shared_ptr и std::weak_ptr, которые автоматически управляют жизненным циклом объектов.Template Method
Паттерн Template Method относится к поведенческим шаблонам проектирования и предоставляет скелет алгоритма в базовом классе, позволяя подклассам переопределять некоторые шаги алгоритма без изменения его структуры.
Этот паттерн часто используется в разработке фреймворков, где каждый подкласс реализует неизменные части архитектуры домена, оставляя "заполнители" для опций настройки.
В этом примере
Паттерн Template Method относится к поведенческим шаблонам проектирования и предоставляет скелет алгоритма в базовом классе, позволяя подклассам переопределять некоторые шаги алгоритма без изменения его структуры.
Этот паттерн часто используется в разработке фреймворков, где каждый подкласс реализует неизменные части архитектуры домена, оставляя "заполнители" для опций настройки.
В этом примере
AbstractClass определяет шаблонный метод TemplateMethod(), который состоит из вызовов различных операций в определенной последовательности. Некоторые из этих операций делегируются подклассам ConcreteClass1 и ConcreteClass2.#вопросы_с_собеседований
Что такое internal linkage?
internal linkage (внутреннее связывание) означает, что имя (например, переменная или функция) видимо и доступно только в пределах файла (или, точнее, в пределах трансляционной единицы), в котором оно определено. Это означает, что если у вас есть два разных файла с исходным кодом, и в каждом из них определено имя с внутренней связью, то эти два имени считаются разными и не конфликтуют друг с другом.
Внутреннюю связь в C++ можно установить несколькими способами. Например, если вы определите переменную или функцию как static, она будет иметь внутреннюю связь. Также, имена в безымянных пространствах имен (anonymous namespaces) имеют внутреннюю связь.
Что такое internal linkage?
Внутреннюю связь в C++ можно установить несколькими способами. Например, если вы определите переменную или функцию как static, она будет иметь внутреннюю связь. Также, имена в безымянных пространствах имен (anonymous namespaces) имеют внутреннюю связь.
Флаг компиляции -fPIC
Это означает, что сгенерированный код может быть исполнен независимо от его абсолютного адреса в памяти.
Это особенно полезно при создании динамических библиотек, которые могут быть загружены в произвольное место в памяти во время выполнения.
Результатом применения этого флага будет объектный файл, который содержит позиционно-независимый код. Этот объектный файл затем может быть использован для создания динамической библиотеки, которую можно загрузить и использовать во время выполнения других программ.
-fPIC используется для генерации позиционно-независимого кода (Position Independent Code, PIC). Это означает, что сгенерированный код может быть исполнен независимо от его абсолютного адреса в памяти.
Это особенно полезно при создании динамических библиотек, которые могут быть загружены в произвольное место в памяти во время выполнения.
Результатом применения этого флага будет объектный файл, который содержит позиционно-независимый код. Этот объектный файл затем может быть использован для создания динамической библиотеки, которую можно загрузить и использовать во время выполнения других программ.
Синхронизация между стандартными потоками C++ и C
Функция
По умолчанию, эта синхронизация включена, это означает, что потоки C++ и C могут быть использованы вместе, и их буферы будут иметь правильный порядок.
Вызов
Этот код используется для быстрого чтения и записи данных, что особенно полезно в соревновательном программировании.
Здесь мы также отвязываем
Функция
std::ios::sync_with_stdio используется для установки синхронизации между стандартными потоками C++ и стандартными потоками C.По умолчанию, эта синхронизация включена, это означает, что потоки C++ и C могут быть использованы вместе, и их буферы будут иметь правильный порядок.
Вызов
std::ios::sync_with_stdio(false) может увеличить производительность ввода/вывода, но после этого стандартные потоки C++ и C не должны использоваться вместе.Этот код используется для быстрого чтения и записи данных, что особенно полезно в соревновательном программировании.
Здесь мы также отвязываем
std::cin от std::cout, что дополнительно увеличивает скорость ввода/вывода и используем \n вместо std::endl, т. к. std::endl выполняет отчиску буфера и может замедлить вывод.Заполняем вектор последовательными значениями
С этим нам поможет функция
В этом примере мы создаем вектор из 10 элементов, заполняем его значениями от 1 до 10 с помощью
С этим нам поможет функция
std::iota, которая является частью библиотеки <numeric>. Она используется для заполнения диапазона последовательными значениями, начиная с определенного значения.В этом примере мы создаем вектор из 10 элементов, заполняем его значениями от 1 до 10 с помощью
std::iota и выводим вектор.#вопросы_с_собеседований
Можно ли выбрасывать exception из конструктора? Какие поля будут сконструированы, какие поля будут разрушены?
в C++ выбрасывать исключения из конструктора можно. Это обычно делается, когда в процессе инициализации объекта происходит ошибка, и объект не может быть корректно сконструирован.
Если исключение выбрасывается из конструктора, то все поля, которые были успешно сконструированы до момента выброса исключения, будут корректно разрушены. Это гарантируется механизмом исключений в C++.
Важно помнить, что только те поля, которые были успешно сконструированы, будут разрушены. Если исключение выбрасывается в процессе конструирования поля, то это поле не будет разрушено, так как его конструктор не был успешно завершен.
Можно ли выбрасывать exception из конструктора? Какие поля будут сконструированы, какие поля будут разрушены?
Если исключение выбрасывается из конструктора, то все поля, которые были успешно сконструированы до момента выброса исключения, будут корректно разрушены. Это гарантируется механизмом исключений в C++.
Важно помнить, что только те поля, которые были успешно сконструированы, будут разрушены. Если исключение выбрасывается в процессе конструирования поля, то это поле не будет разрушено, так как его конструктор не был успешно завершен.
Глубокое копирование
Термин глубокое копирование подразумевает создание нового объекта и копирование всех значений полей исходного объекта в новый объект. Если поле является указателем, то вместо копирования самого указателя создается новый объект, на который указывает исходный указатель, и новый указатель на этот новый объект сохраняется в новом объекте.
Это отличается от поверхностного копирования, при котором копируются только значения полей, включая указатели, но не объекты, на которые они указывают.
В этом примере у нас есть класс
В функции
Термин глубокое копирование подразумевает создание нового объекта и копирование всех значений полей исходного объекта в новый объект. Если поле является указателем, то вместо копирования самого указателя создается новый объект, на который указывает исходный указатель, и новый указатель на этот новый объект сохраняется в новом объекте.
Это отличается от поверхностного копирования, при котором копируются только значения полей, включая указатели, но не объекты, на которые они указывают.
В этом примере у нас есть класс
Deep, который содержит указатель data. В копирующем конструкторе мы создаем новый объект Deep, копируя значение, на которое указывает data в исходном объекте, а не сам указатель.В функции
main мы создаем объект obj1 и затем создаем obj2, используя копирующий конструктор. Затем мы меняем значение, на которое указывает data в obj2, и это не влияет на obj1, что подтверждает, что было выполнено глубокое копирование.std::tie
Она может использоваться для распаковки кортежей или пары значений в отдельные переменные. Например, если у вас есть функция, которая возвращает
В этом примере мы используем
Это позволяет нам проверить, было ли значение успешно вставлено в набор.
std::tie — это функция, которая создает кортеж ссылок на lvalue из своих аргументов или экземпляров std::ignore. Она может использоваться для распаковки кортежей или пары значений в отдельные переменные. Например, если у вас есть функция, которая возвращает
std::pair или std::tuple, вы можете использовать std::tie, чтобы присвоить значения этого кортежа отдельным переменным.В этом примере мы используем
std::tie для распаковки результата вызова set_of_s.insert(value) в две переменные: итератор iter и логическую переменную inserted. Это позволяет нам проверить, было ли значение успешно вставлено в набор.
constinit
Если переменная, объявленная с
Переменная может быть одновременно
constinit — это новый ключевое слово и спецификатор в C++20. Он используется для объявления переменных со статическим или потоковым временем хранения. Если переменная объявлена с constinit, ее инициализирующее объявление должно быть выполнено с constinit. Если переменная, объявленная с
constinit, имеет динамическую инициализацию (даже если она выполняется как статическая инициализация), программа является некорректной.constinit гарантирует, что переменная инициализируется на этапе компиляции, и что статическая инициализация не может привести к проблемам с порядком инициализации. Однако он не делает переменную неизменяемой и не подразумевает const или constexpr. Однако constexpr подразумевает constinit. Переменная может быть одновременно
const и constinit, но не может быть одновременно constexpr и constinit.#вопросы_с_собеседований
Мьютекс и Семафор: Какой из них вы бы использовали для защиты доступа к операции приращения и почему?
Мьютекс и семафор — это два разных механизма синхронизации. Мьютекс используется для обеспечения взаимного исключения доступа к общим ресурсам, в то время как семафор используется для управления доступом к ограниченному числу ресурсов.
В случае операции приращения, которая должна быть выполнена атомарно, стоит использовать мьютекс. Он гарантирует, что только один поток может получить доступ к общему ресурсу в определенный момент времени. Это предотвращает состояние гонки и обеспечивает корректность работы программы.
Мьютекс и Семафор: Какой из них вы бы использовали для защиты доступа к операции приращения и почему?
В случае операции приращения, которая должна быть выполнена атомарно, стоит использовать мьютекс. Он гарантирует, что только один поток может получить доступ к общему ресурсу в определенный момент времени. Это предотвращает состояние гонки и обеспечивает корректность работы программы.
using enum
Конструкция
Это удобно, когда вы хотите использовать перечисление внутри класса или функции без необходимости указывать полное имя перечисления.
Конструкция
using enum была добавлена в C++20 и позволяет импортировать и использовать имена элементов перечисления без указания имени перечисления.Это удобно, когда вы хотите использовать перечисление внутри класса или функции без необходимости указывать полное имя перечисления.
using enum также может использоваться в объявлениях классов для добавления элементов перечисления из именованного перечисления в область видимости класса.Форматирование текста в С++20
В этом примере мы используем
std::format — это функция форматирования текста, которая появилась в C++20. Она предлагает безопасную и расширяемую альтернативу семейству функций printf. Эта функция предназначена для дополнения существующей библиотеки C++ I/O streams.В этом примере мы используем
std::format для форматирования строки "Hello, {}!", где {} является заполнителем для аргумента "world". Результатом работы этого кода будет строка "Hello, world!", которая выводится на экран с помощью std::cout.std::stringstream
В этом примере мы создаем
std::stringstream — это класс, который позволяет выполнять операции ввода-вывода на строковых потоках. Он эффективно хранит экземпляр std::basic_string и выполняет операции ввода-вывода на нем.В этом примере мы создаем
std::stringstream из строки "10 20 30". Затем мы извлекаем три целых числа из потока с помощью оператора >> и сохраняем их в переменных x, y и z. Результатом работы этого кода будет вывод на экран чисел 10, 20 и 30.decltype
Один из примеров использования
В C++11 можно использовать
В этом примере кода показано, как
decltype — это ключевое слово, используемое для запроса типа выражения. Оно было введено в C++11 и используется в основном в обобщенном программировании, где часто трудно или даже невозможно определить тип выражения.Один из примеров использования
decltype — это объявление шаблонной функции, тип возвращаемого значения которой зависит от типов аргументов шаблона. В C++11 можно использовать
decltype для конечного возвращаемого типа вместе с ключевым словом auto, чтобы объявить шаблонную функцию, возвращаемый тип которой зависит от типов аргументов шаблона.В этом примере кода показано, как
decltype используется для определения типа выражения. Результат работы этого кода будет зависеть от того, какие значения будут присвоены переменным и какие функции будут вызваны.atoll
Функция
В этом примере кода мы считываем строку с помощью функции
Результат работы этого кода будет зависеть от того, какое значение будет введено пользователем.
Функция
atoll в языке программирования C++ является функцией библиотеки заголовка cstdlib. Она используется для преобразования строки в значение типа long long int.В этом примере кода мы считываем строку с помощью функции
fgets, а затем преобразуем ее в значение типа long long int с помощью функции atoll. Затем мы выводим полученное значение и его удвоенное значение.Результат работы этого кода будет зависеть от того, какое значение будет введено пользователем.
forward_list
Добавление, удаление и перемещение элементов внутри списка или между несколькими списками не инвалидирует итераторы, которые в настоящее время ссылаются на другие элементы в списке. Однако итератор или ссылка, ссылающиеся на элемент, инвалидируются, когда соответствующий элемент удаляется (через
В этом примере мы создаем
forward_list — это контейнер, который поддерживает быструю вставку и удаление элементов из любого места. forward_list реализован в виде односвязного списка, который хранит только ссылку на следующий элемент.Добавление, удаление и перемещение элементов внутри списка или между несколькими списками не инвалидирует итераторы, которые в настоящее время ссылаются на другие элементы в списке. Однако итератор или ссылка, ссылающиеся на элемент, инвалидируются, когда соответствующий элемент удаляется (через
erase_after) из списка.В этом примере мы создаем
forward_list с именем numbers, с помощью метода front() получаем первый элемент списка и выводим его на экран. Далее мы перебираем все элементы списка с помощью цикла for и выводим их на экран.