Эта статья — для тех, кто только присматривается к Rust или слышал о нем лишь обрывки восторженных или невосторженных отзывов. Если вы не до конца понимаете, зачем миру понадобился еще один системный язык, какие «боли» C++ он лечит и где реально используется сегодня, — здесь вы найдете ответы на эти вопросы. Мы постарались структурированно представить информацию, чтобы у вас сложилась полная картина: что это за язык, зачем его учить и с чего начать освоение. Сразу оговоримся: если «The Book» давно стала вашей настольной книгой, вы уже собаку съели на управлении памятью и знаете все о владении и заимствовании, эта статья вряд ли вас удивит. Остальным же — добро пожаловать.

В апреле 2026 года произошло сразу два события, заставивших вновь говорить о Rust. 16 апреля вышел очередной стабильный релиз — Rust 1.95.0. А вскоре после этого Илон Маск заявил, что новый мессенджер XChat построен на Rust и «whole new architecture».

В этой статье мы попробуем понять, почему для новых систем, где важны скорость, безопасность и надежность, все чаще выбирают Rust?

Что такое Rust, если объяснить в одном абзаце

Rust — это системный язык программирования, созданный для разработки высокопроизводительного и отказоустойчивого софта. Он компилируется в машинный код, не требует виртуальной машины вроде JVM и не использует сборщик мусора как Java, Go, JavaScript или Python.

Главная особенность Rust — безопасность памяти без garbage collector. То есть язык старается не допустить use-after-free, dangling pointer, double free, data race и другие неприятности еще на этапе компиляции.

Если совсем коротко:

Rust — это язык для случаев, когда хочется производительности C++, но не хочется жить с типичными ошибками C++.

Почему Rust вообще появился

Rust был создан Грейдоном Хоаром в Mozilla в период с 2006 по 2010 год. Хоар работал над языком в личное время, а затем Mozilla поддержала проект как перспективную платформу для разработки браузерных движков. Стабильная версия Rust 1.0 вышла 15 мая 2015 года.

Уже тогда команда формулировала задачу языка довольно четко: помочь строить надежные и эффективные системы, сочетая низкоуровневый контроль с высокоуровневыми гарантиями безопасности — без обязательного garbage collector и runtime.

Чтобы понять, зачем Rust был нужен, нужно вспомнить реальность системного программирования. Огромная часть критически важного ПО десятилетиями писалась на C и C++: браузеры, ядра ОС, драйверы, базы данных, сетевые сервисы. Эти языки невероятно мощные, но они дают разработчику очень много свободы. А свобода в работе с памятью часто означает, что программа может сделать что-то опасное.

Например, обратиться к памяти, которая уже освобождена. Или прочитать за пределами буфера. Или изменить данные из двух потоков одновременно без синхронизации. В лучшем случае программа упадет. В худшем — станет уязвимостью.

Chromium Project писал, что около 70% high severity security bugs в Chromium были memory unsafety problems, то есть ошибками работы с C/C++ pointers. Половина из них — use-after-free. Это не “70% всех багов Chrome”, а именно серьезные security bugs, но цифра все равно очень показательная.

Rust появился как попытка закрыть эту инженерную дыру:

  • дать производительность уровня C/C++;

  • убрать значительную часть ошибок памяти;

  • не вводить GC;

  • сделать современный toolchain (менеджер пакетов, тестирование, документация);

  • создать строгий компилятор, который берет на себя проверки вместо надежды на дисциплину разработчика.

Короткая формула:

Rust появился не потому, что миру не хватало еще одного языка программирования. Он появился потому, что у старых подходов был болезненный компромисс: либо быстро, но опасно; либо безопаснее, но с runtime, GC и меньшим контролем.

Какие проблемы других языков Rust пытался решить

Чтобы понять ценность Rust, нужно посмотреть на недостатки существующих языков.

C/C++: быстро, но опасно

C и C++ дают почти максимальный контроль над памятью. Разработчик сам решает, где и когда выделять память, когда освобождать, как передавать указатели, как работать с буферами.

Это делает C/C++ быстрыми и гибкими. Но цена — высокая вероятность ошибок.

Основные проблемы:

  • Memory safety (безопасность работы с памятью) полностью лежит на плечах разработчика.

  • Buffer overflow и buffer over-read — запись или чтение за пределами выделенного буфера.

  • Use-after-free — использование указателя после освобождения памяти.

  • Double free — повторное освобождение одной и той же области памяти.

  • Dangling pointer — указатель, который больше не указывает на валидный объект.

Пример на C:

#include <stdio.h>
#include <string.h>

int main() {
    char name[8];

    strcpy(name, "very very long name");

    printf("%s\n", name);
    return 0;
}

Здесь массив name рассчитан на 8 символов, но мы записываем туда строку сильно длиннее. В C это может привести к повреждению памяти. Иногда программа просто упадет. Иногда будет вести себя странно. Иногда такая ошибка станет частью exploit chain.

Rust не разрешает такой стиль работы в safe-коде. Например, если мы работаем с вектором, доступ по индексу проверяется:

fn main() {
    let items = vec![1, 2, 3];

    // println!("{}", items[10]); // panic: индекс вне границ
    println!("{:?}", items.get(10)); // None
}

items[10] приведtт к panic, потому что индекс вне границ. А items.get(10) вернет None, то есть безопасно скажет: “такого элемента нет”.

Хороший исторический пример — Heartbleed в OpenSSL, 2014 год. В реализации механизма heartbeat для проверки активности соединения в OpenSSL отсутствовала корректная проверка границ. Клиент мог запросить «отправь мне 500 байт», а отправить только 1 байт данных. Библиотека на C честно читала 500 байт из памяти — в том числе чужие ключи шифрования, пароли, сессионные cookie. Технически это buffer over-read (чтение за пределы), но корень тот же: отсутствие проверки границ.

Rust не делает невозможной любую уязвимость, но он резко снижает вероятность целого класса memory-safety ошибок в safe-коде.

Еще один пример — dangling pointer. В C++ можно случайно сохранить указатель на объект, который уже уничтожен:

#include <iostream>

int* make_number() {
    int value = 42;
    return &value; // опасно: возвращаем адрес локальной переменной
}

int main() {
    int* p = make_number();
    std::cout << *p << std::endl; // неопределенное поведение
}

value живет только внутри функции make_number. После выхода из функции память больше невалидна, но указатель p все еще “куда-то” указывает.

Rust такой код в safe-варианте не пропустит:

fn make_number() -> &i32 {
    let value = 42;
    &value
}

Компилятор скажет: нельзя вернуть ссылку на локальную переменную, потому что она будет уничтожена при выходе из функции.

Это и есть одна из главных идей Rust: лучше получить ошибку компиляции, чем production-инцидент.

Java: безопаснее, но есть JVM и GC

Java решила многие проблемы C/C++ другим способом. Разработчик не освобождает память вручную. Этим занимается garbage collector — сборщик мусора.

  • JVM — Java Virtual Machine, виртуальная машина, внутри которой выполняется Java-программа.

  • Runtime — дополнительное программное обеспечение, которое работает параллельно с программой и управляет ее выполнением.

  • GC — garbage collector, механизм автоматического удаления объектов, которые больше не используются.

  • Pause time — паузы, которые могут возникать из-за работы GC. Самый заметный тип таких пауз — Stop-the-world (STW): моменты, когда JVM временно приостанавливает выполнение потоков приложения, чтобы выполнить часть работы сборщика мусора.

Для enterprise/backend Java — прекрасный выбор. Она зрелая, стабильная, с огромной экосистемой. Но для драйверов, ядра ОС, embedded, криптографии, некоторых low-latency систем и очень чувствительных к задержкам компонентов JVM, GC и возможные STW-паузы могут быть сильным ограничением.

Например, в Java можно получить NullPointerException:

class Main {
    static String findUserName(int id) {
        if (id == 1) {
            return "Alice";
        }
        return null;
    }

    public static void main(String[] args) {
        String name = findUserName(2);
        System.out.println(name.length()); // NullPointerException
    }
}

Проблема не в том, что Java плохая. Проблема в том, что null — это особое значение, которое может оказаться почти где угодно, если API это допускает.

В Rust отсутствие значения выражается типом:

fn find_user_name(id: u32) -> Option<String> {
    if id == 1 {
        Some(String::from("Alice"))
    } else {
        None
    }
}

fn main() {
    match find_user_name(2) {
        Some(name) => println!("{}", name.len()),
        None => println!("user not found"),
    }
}

Option<T> означает: значение либо есть (Some), либо его нет (None). Компилятор заставляет обработать оба случая. Это не гарантия, что в программе не будет логических ошибок, но шанс забыть про “пустое значение” становится гораздо ниже.

Go: проще, но меньше контроля

Go — отличный язык для написания инфраструктурного кода. Он проще Rust, имеет удобный garbage collector и отличную поддержку concurrency через горутины.

Однако Go тоже использует GC, а значит, имеет паузы и меньший контроль над памятью. Кроме того, Go не предотвращает data race (гонки данных) на этапе компиляции.

Data race — ситуация, когда два потока или горутины одновременно обращаются к одним и тем же данным, и хотя бы один из них эти данные изменяет. В Go есть встроенный race detector, который запускается через -race, но он ищет гонки, проявившиеся во время выполнения тестов или программы. Rust же старается не дать написать многие такие сценарии уже на этапе компиляции.

Пример идеи на Rust:

use std::thread;

fn main() {
    let mut counter = 0;

    thread::scope(|s| {
        s.spawn(|| {
            counter += 1;
        });

        s.spawn(|| {
            counter += 1;
        });
    });

    println!("{counter}");
}

Такой код не должен компилироваться: две задачи пытаются одновременно получить изменяемый доступ к counter.

Правильный вариант — явно использовать синхронизацию:

use std::sync::{Arc, Mutex};
use std::thread;

fn main() {
    let counter = Arc::new(Mutex::new(0));

    let mut handles = vec![];

    for _ in 0..2 {
        let counter = Arc::clone(&counter);

        handles.push(thread::spawn(move || {
            let mut value = counter.lock().unwrap();
            *value += 1;
        }));
    }

    for handle in handles {
        handle.join().unwrap();
    }

    println!("{}", *counter.lock().unwrap());
}

Arc — атомарный счетчик ссылок для совместного владения между потоками.
Mutex — механизм, который разрешает доступ к данным только одному потоку за раз.

Rust не делает многопоточность необъяснимо простой. Но он заставляет явно описывать, как именно данные разделяются и защищаются.

Python: быстро писать, но не для системного уровня

Python прекрасен для скриптов, data science, automation, backend и прототипов. Но он не создан для низкоуровневого управления памятью и предсказуемой производительности.

В реальных проектах часто встречается гибридный подход: бизнес-логика, orchestration или data science-пайплайн пишутся на Python, а performance-critical части выносятся в C, C++ или Rust.

Хороший пример направления — data processing. В экосистеме Rust есть Polars и DataFusion: быстрые инструменты обработки данных, которые часто используют как альтернативу или дополнение к Python-пайплайнам.

Как работает Rust? Основные концепции

Теперь к самому важному.

Главная идея Rust: ownership

Ownership — владение. В Rust у каждого значения есть ровно один владелец. Когда владелец выходит из области видимости, значение уничтожается, а ресурсы освобождаются. Это основной механизм, благодаря которому Rust управляет памятью без GC.

fn main() {
    let string1 = String::from("hello");
    let string2 = string1;

    // println!("{}", string1); // ошибка компиляции
    println!("{}", string2);
}

Что произошло? String хранит данные в куче. Когда мы пишем let string2 = string1, владение строкой переходит от string1 к string2. После этого string1 больше нельзя использовать.

В JavaScript, Java или Python такое поведение может показаться странным: “почему я не могу использовать старую переменную?” Но Rust делает это не из вредности. Он предотвращает ситуацию, когда две переменные считают себя владельцами одного ресурса и пытаются освободить его дважды.

Для простых типов вроде i32 все иначе:

fn main() {
    let x = 42;
    let y = x;

    println!("{x}");
    println!("{y}");
}

i32 копируется дешево, поэтому здесь работает Copy: обе переменные имеют собственное значение.

Borrowing: можно не отдавать, а одолжить

Если бы каждое использование значения требовало передачи владения, писать программы было бы неудобно. Поэтому в Rust есть borrowing — заимствование.

Заимствование — это временный доступ к значению без передачи владения.

fn main() {
    let name = String::from("Alice");

    print_name(&name);

    println!("{}", name);
}

fn print_name(value: &String) {
    println!("{}", value);
}

name владеет строкой. &name — ссылка, то есть временное заимствование. print_name не забирает строку себе. После вызова name все еще можно использовать.

Главное правило заимствования: можно иметь сколько угодно неизменяемых ссылок (&T) или одну изменяемую ссылку (&mut T), но не то и другое одновременно;

Lifetimes: ссылки не должны жить дольше данных

Lifetime — время жизни ссылки. Это период, в течение которого ссылка остается валидной.

В большинстве случаев Rust сам выводит lifetimes, и разработчик их не пишет. Но иногда компилятору нужно помочь.

fn longest<'a>(first: &'a str, second: &'a str) -> &'a str {
    if first.len() > second.len() {
        first
    } else {
        second
    }
}

'a означает: возвращаемая ссылка живет не дольше, чем входные ссылки. То есть Rust не позволит вернуть ссылку на данные, которые уже уничтожены.

Сначала lifetimes выглядят немного непонятно. Но идея простая: ссылка не должна пережить значение, на которое она указывает.

Borrow checker: строгий компилятор-друг

Borrow checker — часть компилятора Rust, которая проверяет правила владения, заимствования и времён жизни.

Новички часто воспринимают его как врага. Пишешь простой код — компилятор ругается. Передаешь значение — оно “переехало”. Берешь две mutable-ссылки — нельзя. Возвращаешь ссылку — lifetime не подходит.

Но через какое-то время становится понятно: borrow checker не мешает писать код. Он мешает писать код, о котором вы пожалеете через полгода.

Почему Rust не нужен garbage collector

В языках с GC объект живёт до тех пор, пока сборщик мусора не решит, что объект больше недостижим.

В Rust значение уничтожается, когда выходит из области видимости:

fn main() {
    {
        let message = String::from("hello");
        println!("{}", message);
    }

    // здесь message уже уничтожен
}

Здесь message живет только внутри блока { ... }. Когда блок заканчивается, Rust вызывает деструктор и освобождает ресурсы.

Это похоже на автоматическое управление памятью, но без отдельного GC-процесса. Rust заранее проверяет, что после уничтожения значение больше не используется.

Stack и heap простыми словами

Чтобы понять Rust, полезно вспомнить разницу между stack и heap.

Stack — стек. Быстрая область памяти для локальных значений фиксированного размера. Работает по принципу LIFO: last in, first out. Функция вызвалась — ее локальные переменные появились на стеке. Функция завершилась — они исчезли.

fn sum(a: i32, b: i32) -> i32 {
    let result = a + b;
    result
}

a, b, result — простые значения, которые обычно живут на стеке.

Heap — куча. Область памяти для данных динамического размера: строк, векторов, деревьев, объектов, размер которых может быть неизвестен на этапе компиляции.

fn main() {
    let text = String::from("hello");
    let numbers = vec![1, 2, 3];
}

String и Vec хранят метаданные на стеке, а реальные данные — в куче.

Для явного размещения значения в куче есть Box<T>:

fn main() {
    let number = Box::new(10);

    println!("{}", number);
}

Box полезен, когда нужен собственный указатель на данные в куче, например для рекурсивных структур данных.

Option вместо null

В Rust нет привычного null.

Если значения может не быть, это должно быть видно в типе:

fn find_user(id: u32) -> Option<String> {
    if id == 1 {
        Some(String::from("Alice"))
    } else {
        None
    }
}

fn main() {
    let user = find_user(2);

    match user {
        Some(name) => println!("User: {}", name),
        None => println!("User not found"),
    }
}

Option<T> — это enum:

enum Option<T> {
    Some(T),
    None,
}

Самое важное: функция честно говорит вызывающему коду, что результата может не быть.

В Java вы часто узнаёте о null уже в runtime. В Rust отсутствие значения — часть модели типов, и компилятор заставляет с этим считаться.

Result вместо исключений

Rust не использует исключения как основной способ обработки ошибок. Для восстанавливаемых ошибок есть Result<T, E>.

fn divide(a: i32, b: i32) -> Result<i32, String> {
    if b == 0 {
        Err(String::from("division by zero"))
    } else {
        Ok(a / b)
    }
}

fn main() {
    match divide(10, 0) {
        Ok(value) => println!("Result: {}", value),
        Err(error) => println!("Error: {}", error),
    }
}

Result<T, E> означает: либо Ok(T), либо Err(E).

Это делает ошибку частью сигнатуры функции. Разработчик сразу видит, что операция может завершиться неуспешно.

Для короткой передачи ошибки наверх используется оператор ?:

use std::fs;

fn read_config() -> Result<String, std::io::Error> {
    let content = fs::read_to_string("config.txt")?;
    Ok(content)
}

Если read_to_string вернёт Ok(String), строка попадет в content. Если вернет Err(std::io::Error), ошибка сразу вернется из read_config.

? заменяет типичный шаблон match Ok/Err и делает код короче без скрытых исключений.

match: не switch, а мощнее

match похож на switch, но намного сильнее. Он работает с enum, структурами, кортежами, диапазонами и шаблонами.

fn main() {
    let status = divide(10, 2);

    match status {
        Ok(value) => println!("Result: {}", value),
        Err(error) => println!("Error: {}", error),
    }
}

Если у enum есть несколько вариантов, Rust заставляет обработать все. Это снижает вероятность забыть какой-то сценарий.

Например:

enum PaymentStatus {
    Paid,
    Pending,
    Failed(String),
}

fn print_status(status: PaymentStatus) {
    match status {
        PaymentStatus::Paid => println!("paid"),
        PaymentStatus::Pending => println!("pending"),
        PaymentStatus::Failed(reason) => println!("failed: {}", reason),
    }
}

Если завтра мы добавим новый вариант Refunded, компилятор покажет места, где match больше не покрывает все варианты. Для поддержки кода это очень сильная вещь.

Traits: интерфейсы, но не совсем

trait в Rust похож на интерфейс в Java или TypeScript: он описывает поведение, которое должен реализовать тип.

trait Drawable {
    fn draw(&self);
}

struct Button {
    label: String,
}

impl Drawable for Button {
    fn draw(&self) {
        println!("Drawing button: {}", self.label);
    }
}

Теперь можно написать функцию, которая принимает все, что умеет рисоваться:

fn render(item: &impl Drawable) {
    item.draw();
}

Функции важно не то, какой конкретно тип передали, а какое поведение он умеет предоставлять.

Fearless Concurrency (бесстрашная многопоточность)

Rust помогает писать многопоточный код безопаснее. Компилятор не даст вам случайно создать гонку данных. Это особенно ценно для серверов, мессенджеров, сетевых сервисов и высоконагруженных систем.

Zero-cost abstractions: высокоуровневый код без лишней цены

Один из принципов Rust — zero-cost abstractions.

Смысл не в том, что любая абстракция всегда бесплатна. Смысл в том, что многие удобные конструкции компилируются в машинный код без лишней runtime-стоимости по сравнению с ручным низкоуровневым вариантом.

Например, итераторы:

fn sum_even_squares(values: &[i32]) -> i32 {
    values
        .iter()
        .filter(|x| **x % 2 == 0)
        .map(|x| x * x)
        .sum()
}

Код выглядит высокоуровневым: filter, map, sum. Но компилятор может оптимизировать такую цепочку в эффективный машинный код. Операции вроде map, filter, reduce должны создавать код, столь же эффективный, как эквивалентные императивные реализации.

Unsafe Rust: если Rust безопасный, зачем ему unsafe

Rust не запрещает низкоуровневое программирование. Для задач, где компилятор не может полностью доказать безопасность, есть unsafe.

Официальная книга Rust описывает пять возможностей unsafe: разыменование raw pointer, вызов unsafe-функций, доступ или изменение mutable static, реализация unsafe trait и доступ к полям union. При этом unsafe не выключает весь borrow checker и все проверки Rust: оно только дает доступ к операциям, безопасность которых должен гарантировать сам разработчик.

Зачем это нужно? Например:

  • писать драйверы;

  • работать с C-библиотеками через FFI;

  • реализовывать низкоуровневые структуры данных;

  • писать ядро ОС;

  • оптимизировать горячие участки кода;

  • взаимодействовать с железом.

Пример raw pointer:

fn main() {
    let mut value = 10;

    let ptr = &mut value as *mut i32;

    unsafe {
        *ptr += 1;
    }

    println!("{}", value);
}

В safe Rust мы работаем со ссылками &T и &mut T. Raw pointers const T и mut T похожи на указатели C/C++, но Rust не гарантирует их безопасность. Создать raw pointer можно в safe-коде, а вот разыменование требует unsafe.

Хорошая практика: unsafe-кода должно быть мало, он должен быть изолирован и закрыт безопасным API.

Как работает Rust-программа

Rust — компилируемый язык

Rust — компилируемый язык. У него нет интерпретатора в стиле Python или JavaScript. Вы пишете код, компилятор превращает его в машинный код (через промежуточное представление LLVM), и на выходе получается готовый исполняемый файл (бинарник).

Ключевые термины:

  • компилятор — программа, которая переводит исходный код в исполняемый файл (rustc);

  • интерпретатор — программа, которая выполняет код напрямую (Python, Ruby, JS);

  • LLVM — инфраструктура компиляции, используемая Rust, Swift, Clang и другими для генерации оптимизированного кода под разные архитектуры;

  • crate — минимальная единица компиляции и базовый элемент модульной системы языка. Если проводить аналогии с другими языками, то это аналог «библиотеки» (library) или «пакета» (package). Все, что вы пишете на Rust, в конечном итоге упаковывается в крейт.

  • Cargo — система сборки и менеджер зависимостей.

Два типа крейтов

Крейты делятся на две основные категории в зависимости от их назначения:

  • Binary Crate (Бинарный крейт): Это программа, которую можно скомпилировать в исполняемый файл. Такой крейт обязательно содержит функцию main(), которая является точкой входа. Обычно основным файлом такого крейта является src/main.rs.

  • Library Crate (Библиотечный крейт): Это набор кода (функций, типов, структур), предназначенный для повторного использования в других проектах. В нем нет функции main(), и он не запускается сам по себе. Основной файл — src/lib.rs.

Cargo: почему tooling Rust так хвалят

Cargo — это package manager, build system, test runner и инструмент публикации пакетов в одном. Официальная документация Cargo описывает его как менеджер пакетов Rust, который скачивает зависимости, компилирует пакеты, делает distributable packages и загружает их на crates.io.

Для разработчика из JS-мира Cargo можно грубо представить как “npm + build tool + test runner + documentation tool”. Но в отличие от многих экосистем, здесь это стандартный путь, а не набор разрозненных решений.

Stack Overflow Developer Survey 2025 отдельно отмечает Cargo как самый admired cloud development and infrastructure tool с показателем около 71%. Rust там же снова назван самым admired programming language с показателем около 72%.

Основные команды:

  • создание нового проекта (cargo new);

  • сборка (cargo buildcargo build --release);

  • запуск (cargo run);

  • тестирование (cargo test);

  • форматирование кода (cargo fmt);

  • линтинг (cargo clippy);

  • генерация документации (cargo doc).

Crates: библиотеки Rust

Crate — пакет или библиотека в Rust.

crates.io — официальный реестр пакетов.

Rust-экосистема меньше, чем npm, Maven или PyPI, но во многих направлениях она уже очень зрелая.

Чаще всего в реальных проектах встречаются:

  • serde, serde_json — сериализация и десериализация;

  • tokio, async-std — async runtime;

  • axum, actix-web, rocket — web backend;

  • reqwest, hyper — HTTP;

  • clap — CLI-аргументы;

  • anyhow, thiserror — ошибки;

  • tracing — логирование и observability;

  • sqlx, diesel, sea-orm — базы данных;

  • wasm-bindgen, wasm-pack — WebAssembly;

  • tauri, egui — desktop/cross-platform UI;

  • polars, datafusion — обработка данных;

  • rayon — параллельные вычисления.

Итого, что предотвращает и не предотвращает Rust

Rust не делает невозможными все ошибки, но резко снижает класс memory safety проблем.

Он предотвращает на этапе компиляции:

  • use-after-free,

  • buffer overflow/over-read (в safe-коде),

  • data race в многопоточном коде,

  • dangling pointers и т.д.

Однако важно понимать: Rust ≠ полная безопасность приложения.

Компилятор не проверит за вас:

  • криптографический протокол;

  • бизнес-логику;

  • хранение ключей;

  • privacy-модель;

  • архитектуру доступа;

  • уязвимости типа логических ошибок, инъекций или неверных разрешений.

Поэтому даже если мессенджер XChat, по заявлению, написан на Rust, это не доказывает его абсолютную безопасность. Язык — лишь один из слоев защиты.

Плюсы и минусы Rust

Плюсы Rust

Главные плюсы Rust:

  • безопасность памяти без GC;

  • высокая производительность;

  • отсутствие обязательного runtime;

  • строгая система типов;

  • Option<T> вместо null;

  • Result<T, E> вместо скрытых исключений;

  • безопасная многопоточность;

  • современный tooling;

  • Cargo и crates.io;

  • хорошая совместимость с C через FFI;

  • сильные возможности для системного программирования;

  • хорошие сообщения компилятора;

  • встроенная поддержка тестирования и документации.

Но главный плюс даже не технический. Rust меняет мышление. Он заставляет думать о владении, жизненном цикле данных, ошибках и потоках не после падения программы, а во время написания кода.

Минусы Rust

Минусы тоже реальные.

Самые заметные:

  • высокий порог входа;

  • ownership и borrowing раздражают новичков;

  • lifetimes могут быть сложными;

  • компиляция иногда долгая;

  • async Rust требует привыкания;

  • меньше библиотек, чем у Java/Node/Python в некоторых доменах;

  • меньше вакансий, чем для массовых языков;

  • не всегда подходит для быстрой продуктовой разработки;

  • unsafe-код требует дисциплины;

  • некоторые структуры данных, например графы с двунаправленными ссылками, писать сложнее, чем в языках с GC.

Rust удобен не во всем: двунаправленные ссылки, Rc, Weak, RefCell, внутренняя изменяемость и графовые структуры требуют понимания модели владения. Это не “невозможно”, но точно сложнее, чем просто хранить ссылки в языке с GC.

Почему компании переписывают части систем на Rust

Обычно компании не переписывают весь продукт на Rust. Это дорого, рискованно и часто не нужно.

Гораздо чаще Rust внедряют точечно: в сетевые компоненты, парсеры, sandbox, криптографию, CLI, агенты, runtime-компоненты, драйверы, сервисы с высокой нагрузкой.

Причины обычно прагматичные:

  • Снижение memory-safety рисков — меньше уязвимостей, с которыми приходится жить годами.

  • Выше надежность — Rust-код реже падает в production из-за неопределённого поведения.

  • Меньше инцидентов — команды тратят меньше времени на отладку багов, связанных с указателями.

  • Лучший контроль ресурсов — отсутствие GC позволяет точно предсказывать потребление памяти и CPU.

  • Экономия на инфраструктуре — оптимизированный код может обрабатывать ту же нагрузку на меньшем количестве серверов.

Google пишет, что в Android подход с Rust уже дал серьезный эффект: в данных 2025 года memory safety vulnerabilities впервые упали ниже 20% от общего числа уязвимостей, а Rust-код показал заметно меньшую density memory-safety уязвимостей по сравнению с C/C++.

В Linux kernel есть официальная документация по Rust внутри ядра. Это не значит, что ядро Linux переписывают на Rust целиком, но Rust стал официально поддерживаемым направлением для части kernel-разработки.

Microsoft также развивает направление Rust for Windows Drivers: в 2025 году Microsoft писала о состоянии Rust для Windows-драйверов и своём видении развития этого направления.

AWS Firecracker — пример инфраструктурного проекта, написанного на Rust: AWS описывала Firecracker как lightweight virtualization technology для serverless workloads, а в анонсах подчеркивала выбор Rust из-за thread safety и предотвращения многих buffer overrun ошибок.

Cloudflare использует Rust в Pingora — своем фреймворке для быстрых и надежных сетевых систем. Cloudflare писала, что Pingora, построенный на Rust, обслуживает значительную часть инфраструктурного трафика компании.

Иными словами, Rust выбирают не потому, что он модный. Его выбирают там, где ошибка памяти, лишняя задержка или неэффективность инфраструктуры стоят дорого.

Где Rust лучше всего использовать

Rust особенно силен там, где одновременно важны скорость, надежность и контроль.

Хорошие сценарии:

  • backend и высоконагруженные сервисы;

  • API, proxy, realtime-сервисы, очереди;

  • CLI-инструменты;

  • системное программирование;

  • драйверы, ядро, низкоуровневые библиотеки;

  • embedded и IoT;

  • WebAssembly;

  • криптография и security-sensitive software;

  • парсеры;

  • data processing;

  • blockchain и инфраструктура.

Где Rust может быть плохим выбором

Rust не нужен везде.

Он может быть плохим выбором, если вы делаете MVP за два дня, простую админку, обычный CRUD-backend или продуктовую фичу, где команда гораздо сильнее в Java, Node.js, Go или Python.

Rust также может быть избыточен, если главная задача — быстро нанять команду, быстро проверить гипотезу или использовать экосистему, где Python/Java/JS дают готовые решения быстрее.

Еще один честный минус — learning curve. Ownership, borrowing и lifetimes сначала непривычны. Разработка на старте может идти медленнее. Компиляция больших проектов может занимать заметное время. Async Rust сложнее, чем async в JS или Go.

Поэтому Rust — не универсальная замена всем языкам. Его сила проявляется там, где цена ошибки, лишней памяти или непредсказуемой производительности действительно высока.

Rust в 2026 году: статистика

Rust уже не экспериментальный язык, но и не массовый язык уровня JavaScript, Python или Java.

По Stack Overflow Developer Survey 2025 Rust снова стал самым admired programming language с показателем около 72%, но доля worked-with среди всех респондентов — 14.8%. То есть Rust очень любят те, кто с ним работает, но он пока не стал массовым языком для всех подряд.

TIOBE Index за апрель 2026 показывает Rust в top 20, но отмечает замедление роста: после пика на #13 в начале года Rust опустился к #16. Сам TIOBE подчёркивает, что индекс показывает популярность, а не качество языка.

PYPL показывает Rust примерно в районе top 10 по tutorial/search interest: интерес к изучению сохраняется, хотя Rust не обгоняет массовые языки.

State of Rust Survey 2025, опубликованный в марте 2026 года, собрал 7156 ответов и показывает зрелое сообщество, но также фиксирует знакомые проблемы: сложность языка, ожидания вокруг compiler performance и развитие экосистемы.

Итог простой: Rust — не хайп на один сезон. Но его роль не в том, чтобы заменить все языки, а в том, чтобы занять сильные ниши: infrastructure, backend, systems, security, embedded, WebAssembly и performance-critical компоненты.

Как изучать Rust

Лучше не начинать с async, unsafe и макросов. Сначала нужно понять базу.

С чего начать:

  • базовый синтаксис (переменные, циклы, функции);

  • structenummatch;

  • модули;

  • Option и Result;

  • ownership, borrowing, lifetimes.

Лучшие ресурсы:

  • «The Rust Programming Language» (официальная книга, «The Book») — бесплатно онлайн, есть русский перевод. Обязательна к прочтению.

  • Rust by Example — учимся на готовых примерах (тоже переведена на русский).

  • Rustlings — маленькие упражнения в терминале, чтобы закрепить материал.

  • The Cargo Book — документация по Cargo.

  • Rustonomicon — для тех, кто хочет глубоко разобраться в unsafe.

  • Tokio Tutorial — асинхронное программирование.

  • «Zero To Production In Rust» — для backend-разработчиков.

  • «Command‑Line Rust» — для тех, кто хочет делать CLI-утилиты.

  • «Programming Rust» (ОʼReilly) — более глубокая книга.

  • «Rust for Rustaceans» — для тех, кто уже знает основы.

Rust лучше изучать через практику. Сначала CLI-проекты, потом backend, WASM, embedded или системные задачи — в зависимости от интереса.

Хорошие первые проекты: 

  1. CLI todo‑list.

  2. Mini grep (поиск строк в файле).

  3. Парсер логов (например, веб-сервера) с агрегацией статистики.

  4. JSON/CSV-парсер.

  5. HTTP API на Axum (CRUD по пользователям).

  6. Telegram/Discord бот.

  7. Многопоточный парсер логов (каждый файл в отдельном потоке, сбор статистики через mpsc).

  8. WebAssembly-модуль, вызываемый из JavaScript.

  9. Небольшой прокси-сервер (HTTP или TCP).

  10. Маленький embedded-проект (если есть интерес к железу, например, мигание светодиодом на STM32 или Raspberry Pi Pico).

Стоит ли учить Rust

Да, если:

  • Вы хотите понимать системное программирование на глубоком уровне: как работают память, указатели, стек, куча.

  • Работаете с высоконагруженным бэкендом, микросервисами, нуждаетесь в предсказуемой производительности.

  • Вам интересны безопасность, инфраструктура, блокчейн, embedded, WebAssembly.

  • Хотите прокачать мышление о памяти, типах, владении и конкурентности — эти навыки пригодятся в любом языке.

  • Вы готовы потратить несколько недель на преодоление начальной сложности.

Не обязательно прямо сейчас, если:

  • Вы занимаетесь только простым фронтендом (React/Vue/Svelte) и не планируете использовать WASM или bэкенд.

  • Ваш текущий стек (Java enterprise, .NET, Go) полностью закрывает потребности компании, и у вас нет задачи внедрять низкоуровневые оптимизации.

  • Команда не готова к переходу на Rust.

Rust не заменит JavaScript во frontend. Не заменит Python в data science. Не заменит Java во всем enterprise. Не заменит Go во всех микросервисах. И не обязан.

Его сила в другом: Rust постепенно забирает те области, где раньше почти автоматически выбирали C или C++, но где теперь хочется большей безопасности.

Заключение

Rust не появился, чтобы заменить все языки. Он появился, чтобы закрыть конкретную инженерную задачу: дать разработчику скорость и контроль системного языка, но убрать значительную часть ошибок, которые десятилетиями приводили к уязвимостям и падениям программ.

Именно поэтому Rust выбирают не для всего подряд, а для тех частей систем, где цена ошибки особенно высока: безопасность, память, производительность, многопоточность, сеть, инфраструктура и системный код.

XChat — хороший повод снова поговорить о Rust. Но главный вывод не в том, что “раз Маск выбрал Rust, значит всем нужно срочно переписывать проекты”. Главный вывод в другом: если новая система должна быть быстрой, надежной, безопасной по памяти и готовой к высокой нагрузке, Rust становится очень сильным кандидатом.

Если вы хотите писать код, который одновременно быстр, строг и надежен, Rust точно стоит попробовать. Но кофе действительно понадобится много.

Комментарии (13)


  1. liquidgel
    07.05.2026 15:07

    Проблему с unsafe как-то решили? Дико смешно было лет 8 назад читать как они настолько сильно упоролись безопасностью что без unsafe нельзя было написать примерно никакой серьезный софт, а любые подтянутые в проект либы (разумеется unsafe) по цепочке делали и твой софт unsafe. В итоге единственное конкурентное преимущество языка его же и хоронило


    1. Boneyan
      07.05.2026 15:07

      Нет проблемы с unsafe - это легитимная часть языка и никто от неё избавляться не собирается. Не для всех вещей компилятор может доказать безопасность, в таких случаях используется unsafe. unsafe не делает “по цепочке” весь остальной код unsafe. unsafe код можно вызывать внутри safe кода. Обычно это небольшие небезопасные участки, обёрнутые в безопасное api.


      1. TimurZhoraev
        07.05.2026 15:07

        Вообще говоря safe/unsafe это маркетинговый ход нежели технические подробности. Язык должен предоставить какое-либо число, чтобы это измерить. Например, в случае гонки - какие объекты пытаются обратиться к шаре, проблема чтение-модификация-запись при статическом анализе, теоретическая глубина стека при рекурсии или измеренная в реалтайм, сведения о динамических аллокаторах особенно в многопоточке, может даже что-то связанное с DMA, то есть если язык не поддерживает такого рода числа - то это сомнительное преимущество по сравнению с явным указанием флажков, счётчиков ссылок, дополнительных состояний объекта итд. Да это неудобно, но зато прозрачно и позволяет выявить при дебаге всевозможные комбинации. Тем более как только появляется FFI/mmap особенно в ембедде - даже на одном таком пороге ломается вся концепция. По сути язык стал обвязкой вокруг некоего memory and object manager. Но что-то явных преимуществ нет. То есть универсальная попытка встать между компиляцией и интерпретацией скорее всего тупиковая.


      1. liquidgel
        07.05.2026 15:07

        Нет проблемы с unsafe - это легитимная часть языка и никто от неё избавляться не собирается

        Т.е. язык как был мертворожденным, так им и остался


  1. TimurZhoraev
    07.05.2026 15:07

    Проблема главная и немаловажная - нет стандарта по которому можно получить гарантированные компиляторы. Это же касается и Питона. Версии 2.7 и 3 стали именем нарицательным. Выйдет Rust 2.0 вот и придётся думать что там нужно прогнать через Perl/Python Regexp-ом чтобы починить кодовую базу. Думается что с LLM актуальность не стандартизированных языков существенно снизилась, потому как для работы нужны предсказуемые инструменты а не синтаксический сахар, который уже LLM инкапсулирует до уровня отсутствия в нём необходимости, включая покрытие тестами и технический регламент (саммари). Вообщем останется только Python и K&R C или Страуструп С++ образца начала 90-х, включая диалекты близкие к синтаксису Java/OpenCL(MP)/CUDA, благодаря феноменальной наработанной кодовой базе по существу (может даже ещё Fortran), а не трансляторами Java/Python/C++ -> Rust. Да, Go/Ruby/Rust и др современные, но их начало совпало как раз с этими кодогенераторами, вообщем те языки разработка которых пришлась где-то на 2015-20й год скорее всего будут свёрнуты, так как датасетов для претренинга от них маловато, они достаточно далеки от железа и имеют коммьюнити с минимальной поддержкой, язык нужно учить, развивать и делать это промышленным способом а не энтузиазмом разработчиков.


    1. Dhwtj
      07.05.2026 15:07

      Rust обещает полную совместимость внутри edition

      Это лучше чем ломающие изменения питон и лучше чем дряхлый C++

      А вообще, несёте херню


      1. TimurZhoraev
        07.05.2026 15:07

        Rust обещает полную совместимость

        Разработчики языка могут этим заниматься сколько угодно, может появится 3 ветки языка Rust% какой-нибудь. Важно то, что дряхлые плюсы всё-таки имеют ISO/IEC 14882, это позволяет исключить бардак с инструментами и библиотеками. Во всяком случае можно апеллировать к документу уровня ГОСТ, описывающему язык. У Питона с этим тоже беда, но он по крайней мере стал BASIC-ом 21го века благодаря грамматике, умещающейся на тетрадный лист. Любой Bison/Flex влезит в контекст даже самой захудалой LLM-ки, это его очень хорошо вытаскивает + Pip-ы на все случаи жизни. Раст не может похвастать таким коммьюнити, так как язык скорее нишевый чем общего назначения. Если для Питона или того же Perl есть pip и Cpan, то что ищется для Раста с первой же строчки поисковика? Что там нужно набрать "rust library modules", "rust library hub" (по иронии попадаем сюда), "rust modules library collection"... все пути ведут к git. Язык без поддержки порталом по скачиванию чего угодно минуя заклинания git встроенными средствами - это путь к забвению. Мне охота высветить что-то на выводы малины нолики-единички. Куда обратиться кроме частных лиц (обратите внимание - автор свернул проект в архив)? Для питонов делаю в один клик. Для С коммьюнити куда больше так как там вся обвязка и есть "протухший" С. Тем более можно забыть что LLM уже генерит на С даже для эмбеда и микроконтроллеров-восьмибиток, там причёсывание из разряда поправить порядок инициализации перифирии и пару бит. Нашёл тут как-то либу на Раст для энкодера полнорегистровую, заставил LLM переписать на С, чтобы управлять скудной памятью самостоятельно, ESP32 это из пушки по воробьям. Прекрасно получилось хоть для атмелки хоть для кортекса м0/3.


        1. Dhwtj
          07.05.2026 15:07

          апеллировать к документу уровня ГОСТ, описывающему язык

          Не пойму к чему вы

          Вот проявится у Rust второй компилятор тогда может и понадобится что-то вроде ГОСТ

          Да и неопределенное поведение в c++ штука неприятная при наличии всех типа ГОСТ

          rust library modules

          https://crates.io/


          1. TimurZhoraev
            07.05.2026 15:07

            Craits.io - такое ощущение что это запрятано от всех поисковиков, а если более точно то https://crates.io/. Я захожу на главный портал языка https://rust-lang.org/ - и собственно где ссылка на это? Захожу сюда https://rust-lang.org/community/ - где этот кратэс... или разработчики его сами боятся. Я нашёл его только в https://rust-lang.org/tools/ мелким шрифтом где-то в глубине фрейма. Вообщем отношение такое к пользователю из разряда вот мы тут что-то придумали хорошее, но то, как это хорошо использовать - додумывайте сами. В плюсах и сях неопределённое поведение уже давным давно решается определёнными вставками. Любой санитайзер выловит любые утечки, зато это можно записать для любой платформы начиная от Спектрумов до ПЛИС-ов. Тем более с LLM-кой которая качественно обёртывает тестами своё творчество контроль за памятью уже не проблема как год или даже полтора.


          1. TimurZhoraev
            07.05.2026 15:07

            Что касается ГОСТ-а на язык коим является ISO/IEC на протяжении уже треть века или даже половины уже - то это гарантия что в энтерпрайзе есть что брать за основу не вдаваясь в подробности реализации - если соответствует - действуй. Это позволяет избежать vendor-lock или привязки к мыслям что будет со второй версией чего-то там. Комитет - это некая письменная захардкоженная гарантия результата. Разработчики же не могут дать такую бумагу. Только некие протоколы тестирования в частном порядке согласно каким-то там внутренним регламентам. Сегодня там есть этот прогер/архитектор, завтра женился, ушёл в Тибет, забил, стало душно, не интересно итд. Вот чтобы этого избежать и придуманы стандарты, фиксирующие в определённый момент времени то то необходимо. Горизонт планирования разработки без стандартов - вчера и сегодня. То есть для хобби и экспериментов. Поэтому Линус совершенно прав, пуская новые языки и фреймворки в ядро только на альтеративные функции, во всяком случае дубляж на стандартном языке имеется точно. Либо уже в комитете лежит драфт, но что-то сомнительно, хотя может и пришло время.


            1. black_warlock_iv
              07.05.2026 15:07

              То, к какому аду приводит стандарт хорошо видно на примере C++. Буквально на днях выяснилось, что контейнер, запланированный в новом стандарте, будет в 2 раза медленнее аналогичного контейнера в буст. А история с std::views::please_work, ой простите std::views::as_input, без которого std::views::filter тотально сломан? Это всё оттуда же идёт, от стандартизации и комитета.

              Стандартизация C++ — это не благо, а вынужденная мера из-за расплодившихся несовместимых реализаций. Rust идёт по пути референсного компилятора, это гораздо лучше. А к чему приводит искусственная стандартизация при наличии референсного компилятора известно на примере Haskell: всем плевать на стандарт и все пишут код конкретно для GHC.


  1. Dhwtj
    07.05.2026 15:07

    Rust также хорош для моделирования предметной области, наравне со scala. Но мне почему-то никто не верит.


  1. QtRoS
    07.05.2026 15:07

    В целом article получилась not bad, по сложности affordable для junior developers, хотя порой не хватает details. Думаю, Rust вполне может стать timeless, как C++