C++ geek
3.61K subscribers
277 photos
5 videos
28 links
Учим C/C++ на примерах
Download Telegram
📦 std::move vs std::forward: Когда и зачем?

На собеседованиях часто спрашивают про rvalue-ссылки, но в реальном коде мы постоянно путаемся: когда делать move, а когда forward?

Давайте разберем на жизненных примерах.

1. std::move - "Это мое, но забирай!" 🚚

std::move - это безусловное приведение к rvalue. Вы говорите компилятору: "Мне этот объект больше не нужен. Можешь выпотрошить его и забрать данные, не копируя их".

Сценарий 1: Передача владения (unique_ptr)
Это классика. std::unique_ptr нельзя скопировать, его можно только переместить.


auto ptr = std::make_unique<BigData>();

// process(ptr); // Ошибка компиляции! Копирование запрещено.
process(std::move(ptr)); // ОК. Владение передано, ptr теперь пуст.



Сценарий 2: Оптимизация тяжелых объектов
У вас есть локальный вектор, который вы хотите сохранить в поле класса. Зачем его копировать?


void SetData(std::vector<int> newData) {
// Мы крадем буфер памяти у newData.
// Копирования элементов НЕ происходит.
this->data_ = std::move(newData);
}




2. std::forward - "Я просто посредник" 📮

std::forward используется почти исключительно в шаблонах. Его цель - Perfect Forwarding (Идеальная передача).

Представьте, что вы пишете функцию-обертку (wrapper). Она принимает аргумент и должна передать его дальше другой функции.

⚫️Если ей передали временный объект (rvalue) - она должна передать его как rvalue (чтобы сработал move).
⚫️Если передали обычную переменную (lvalue) - она должна передать как lvalue (копия).

std::move здесь всё испортит (он всё превратит в rvalue). Тут нужен std::forward.

Сценарий: Фабрики и Обертки


template <typename T>
void LogAndAdd(std::vector<T>& vec, T&& item) {
std::cout << "Adding item...";

// forward сохранит категорию значения item.
// Если item был временным — сработает push_back(T&&) (перемещение).
// Если item был переменной — сработает push_back(const T&) (копия).
vec.push_back(std::forward<T>(item));
}




⚡️ Шпаргалка

1. std::move используем, когда мы знаем, что объект нам больше не нужен, и мы хотим отдать его ресурсы (обычный код).
2. std::forward используем, когда мы пишем шаблон, который принимает "универсальную ссылку" (T&&), и нам нужно пробросить аргумент дальше "как есть" (библиотечный код).

#cpp #cpp11 #movesemantics #coding #interview #tips

➡️ @cpp_geek
Please open Telegram to view this post
VIEW IN TELEGRAM
👍83
🎭 Сколько стоит virtual? Вся правда о полиморфизме и магии final

Мы обожаем интерфейсы и ООП. Добавить virtual перед методом - минутное дело, и вот наш код уже гибкий и расширяемый. Но задумывались ли вы, чем мы за это платим на уровне железа?

⚙️ Анатомия виртуального вызова (vtable)

Если в классе есть хотя бы одна виртуальная функция, компилятор втайне добавляет в каждый объект этого класса скрытое поле - vptr (указатель на виртуальную таблицу). Сама таблица (vtable) хранится где-то в памяти и содержит адреса реальных функций.

Как происходит вызов obj->DoWork() под капотом:

1. Процессор идет по адресу объекта obj.
2. Читает скрытый указатель vptr.
3. Делает прыжок в память, где лежит vtable.
4. Находит там нужный адрес функции для конкретного класса-наследника.
5. Делает еще один прыжок, чтобы выполнить код.

🚨 Почему это бьет по производительности?

Дело даже не в лишних прыжках по памяти (хотя промахи кэша процессора - это больно).
Главная проблема: Виртуальность убивает оптимизации.

Когда компилятор видит вызов виртуальной функции через указатель, он "слепнет". Он не знает, код какого именно наследника будет вызван во время работы программы (Run-time). Из-за этого он не может применить Inlining (встраивание тела функции вместо вызова) - а это самая мощная оптимизация в C++.

🛡 Спаситель из C++11: ключевое слово final

Слово final запрещает дальнейшее наследование класса или переопределение метода. Но кроме защиты архитектуры, оно делает невероятное: включает Девиртуализацию (Devirtualization).


class Base {
public:
virtual void Process() = 0;
};

// Мы жестко фиксируем класс: от него нельзя наследоваться!
class Derived final : public Base {
public:
void Process() override {
/* важная логика */
}
};

void RunOptimized(Derived* obj) {
// Компилятор видит: тип obj — Derived.
// Derived помечен как final. Значит, никто физически
// не мог переопределить метод Process!

// 🚀 МАГИЯ: Компилятор выбрасывает vtable, игнорирует vptr
// и превращает вызов в обычный, или вообще инлайнит (встраивает) его!
obj->Process();
}



💡 Золотое правило современного C++:
Относитесь к классам как к запечатанным. Пишите final для всех классов (особенно тех, что реализуют интерфейсы), если только вы не проектируете их специально для дальнейшего наследования.

Вы получите защиту от глупых архитектурных ошибок и бесплатный прирост скорости!

#cpp #cpp11 #oop #optimization #performance #coding #tips

➡️ @cpp_geek
👍10