Выпуск Rust 1.17
Команда Rust рада представить выпуск Rust 1.17.0. Rust — это системный язык программирования, нацеленный на безопасность, скорость и параллельное выполнение кода.
Если у вас установлена предыдущая версия Rust, то для обновления достаточно выполнить:
1 | $ rustup update stable
|
Если у вас ещё не установлен Rust, то вы можете установить rustup
c
соответствующей страницы нашего веб-сайта и ознакомиться с подробным примечанием
к выпуску 1.17.0 на GitHub.
Что вошло в стабильную версию 1.17.0
Выпуск Rust 1.17.0 в основном вносит небольшие улучшения, преимущественно касающиеся удобства использования.
Например, время жизни 'static
теперь автоматически подразумевается для констант или статических переменных. При создании константы или статической переменной:
1 2 | const NAME: &'static str = "Ferris"; static NAME: &'static str = "Ferris"; |
Rust 1.17 позволит вам больше не писать 'static
, так как это единственное время жизни, которое имеет смысл:
1 2 | const NAME: &str = "Ferris"; static NAME: &str = "Ferris"; |
В некоторых ситуациях это позволит избавиться от лишних повторений:
1 2 3 4 5 | // было const NAMES: &'static [&'static str; 2] = &["Ferris", "Bors"]; // стало const NAMES: &[&str; 2] = &["Ferris", "Bors"]; |
Другим подобным косметическим улучшением является «короткая инициализация полей». Подобно ECMAScript 6, который называет это «Сокращение Значения Свойства Объектного Литерала» («Object Literal Property Value Shorthand»), дублирование может быть удалено при объявлении структур, к примеру:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | // определение struct Point { x: i32, y: i32, } let x = 5; let y = 6; // было let p = Point { x: x, y: y, }; // стало let p = Point { x, y, }; |
То есть форма записи x, y
будет предполагать, что значения x
и y
соответствуют переменным с аналогичными именами, находящимися в данной области видимости.
Другое маленькое, но полезное улучшение касается в основном новичков в Rust, которые пытаются использовать +
, чтобы соединить две &str
вместе. Но это не работает, соединить вы можете лишь String + &str
. Поэтому, было добавлено новое сообщение об ошибке, чтобы помочь пользователям, которые совершают подобную ошибку:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 | // код "foo" + "bar" // было error[E0369]: binary operation `+` cannot be applied to type `&'static str` --> <anon>:2:5 | 2 | "foo" + "bar" | ^^^^^ | note: an implementation of `std::ops::Add` might be missing for `&'static str` --> <anon>:2:5 | 2 | "foo" + "bar" | ^^^^^ // стало error[E0369]: binary operation `+` cannot be applied to type `&'static str` --> <anon>:2:5 | 2 | "foo" + "bar" | ^^^^^ | = note: `+` can't be used to concatenate two `&str` strings help: to_owned() can be used to create an owned `String` from a string reference. String concatenation appends the string on the right to the string on the left and may require reallocation. This requires ownership of the string on the left. | "foo".to_owned() + "bar" |
При использовании сценариев сборки Cargo вы должны указать расположение скрипта в вашем Cargo.toml
. Однако, подавляющее большинство людей писали build = "build.rs"
, тем самым используя файл build.rs
, расположенный в корне проекта. Теперь это соглашение поддерживается самим Cargo, и будет использовано по умолчанию, если существует файл build.rs
. Мы предупреждали об этом изменении в течение нескольких последних выпусков. Вы также можете использовать build = false
для отказа от этого соглашения.
В этом выпуске удалена старая система сборки на основе Makefile
. Новая система, анонсированная в Rust 1.15, написана на Rust и в основном использует Cargo для управления сборкой. На данный момент она уже достаточно зрелая, чтобы быть единственной системой сборки.
В рамках этого изменения, пакеты из crates.io теперь можно использовать в системе сборки Rust. Первым был добавлен mdBook, и он теперь используется при сборки нашей разнообразной книжной документации:
- Книга «Язык программирования Rust» (репозиторий)
- Книга «Справка по Rust» (репозиторий)
- Книга «Растономикон» (репозиторий)
Обратите внимание на ссылки на соответствующие репозитории; документы были перемещены из основного дерева. Кроме того, мы добавили четвёртую книгу, которая все ещё расположена в основном дереве: Книга «Нестабильные возможности Rust». Она описывает нестабильные возможности, содержит ссылки на задачи, связанные с их стабилизацией, и может содержать исходную документацию. Если есть возможность, которую вы хотите увидеть стабилизированной, то пожалуйста, примите участие в её обсуждении!
Несколько выпусков назад rustup
перестал по умолчанию устанавливать документацию. Мы внесли это изменение, чтобы немного разгрузить канал, а также потому, что не все пользователи в действительности хотят хранить локальную копию документации. Однако это создало ловушку: некоторые пользователи не были в курсе, что произошло изменение, и заметили бы это, только если бы у них пропало подключение к Интернету. Кроме того, некоторые пользователи хотели иметь локальную копию документации, независимо от их подключения. Таким образом, мы откатили данное изменение, и документация снова устанавливается по умолчанию.
Наконец, хотя этот выпуск и полон улучшений, но есть один маленький шажок назад, о котором мы хотим сообщить вам с сожалением. В Windows была выпущена Visual Studio 2017, и Microsoft изменила структуру установки программного обеспечения. Rust не может автоматически определить её местоположение, и хотя мы работали над необходимыми изменениями, но не успели к этому выпуску. До тех пор, Visual Studio 2015 все ещё работает нормально, или же вы можете запустить vcvars.bat
в командной строке. Мы надеемся, что эта работа будет выполнена в кратчайшие сроки.
Подробнее смотрите примечания к выпуску.
Стабилизация библиотек
Был стабилизирован 21 новый интерфейс:
Arc::into_raw
иRc::into_raw
позволят вам забратьArc
илиRc
и получить сырой указатель.Arc::from_raw
иRc::from_raw
позволят вам забрать сырой указатель и получитьArc
илиRc
.Arc::ptr_eq
иRc::ptr_eq
возвращаетtrue
если обаArc
или обаRc
указывают на одно и то же значение (не обязательно значения, которые сравниваются, равны).Ordering::then
позволит вам сцепить дваOrdering
вместе, иOrdering::then_with
позволит сделать это с помощью функции.BTreeMap::range
позволит вам итерировать лишь по частиBTreeMap
, иBTreeMap::range_mut
позволит вам сделать это с возможностью изменения.collections::Bound
может дать вам ещё больше контроля.process::abort
будет полностью завершать процесс анормальным образом.ptr::read_unaligned
иptr::write_unaligned
аналогичныptr::read
иptr::write
, но без требований к выравниванию.Result::expect_err
зеркально подобенResult::expect
, то есть работает с вариантомErr
, а не с вариантомOk
.Cell::swap
аналогиченstd::mem::swap
, но позволяет вам делать это с&Cell
вместо&mut T
.Cell::replace
аналогиченstd::mem::replace
, но позволяет вам делать это с&Cell
вместо&mut T
.Cell::into_inner
позволит вам взятьCell
, и извлечь его значение.Cell::take
позволит вам забрать значениеCell
наружу, заменив его наDefault::default
.
Что касается других изменений, для многих методов Cell<T>
требовалось ограничение T: Copy
, но теперь это требование значительно ослаблено.
Box<T>
теперь реализует более дюжины новых преобразований с помощью From
.
SocketAddr
и IpAddr
также теперь имеют несколько новых преобразований. Раньше вы должны были писать код вроде этого:
1 | "127.0.0.1:3000".parse().unwrap() |
Сейчас же вы можете писать
1 2 3 | SocketAddr::from(([127, 0, 0, 1], 3000)) // или ([127, 0, 0, 1], 3000).into()) |
Что позволит удалить ненужный парсинг во время выполнения. Оба варианта примерно одинаково удобно читаются, все зависит от ваших предпочтений.
Обратная трассировка теперь имеет более приятное форматирование, опуская некоторые незначительные детали по умолчанию. Например, полная обратная трассировка:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 | thread 'main' panicked at 'explicit panic', foo.rs:2 stack backtrace: 1: 0x55c39a23372c - std::sys::imp::backtrace::tracing::imp::write::hf33ae72d0baa11ed at /buildslave/rust-buildbot/slave/stable-dist-rustc-linux/build/src/libstd/sys/unix/backtrace/tracing/gcc_s.rs:42 2: 0x55c39a23571e - std::panicking::default_hook::::h59672b733cc6a455 at /buildslave/rust-buildbot/slave/stable-dist-rustc-linux/build/src/libstd/panicking.rs:351 3: 0x55c39a235324 - std::panicking::default_hook::h1670459d2f3f8843 at /buildslave/rust-buildbot/slave/stable-dist-rustc-linux/build/src/libstd/panicking.rs:367 4: 0x55c39a235afb - std::panicking::rust_panic_with_hook::hcf0ddb069e7beee7 at /buildslave/rust-buildbot/slave/stable-dist-rustc-linux/build/src/libstd/panicking.rs:555 5: 0x55c39a22e866 - std::panicking::begin_panic::heb433e9aa28a7408 6: 0x55c39a22e9bf - foo::main::hd216d4a160fcce19 7: 0x55c39a23d44a - __rust_maybe_catch_panic at /buildslave/rust-buildbot/slave/stable-dist-rustc-linux/build/src/libpanic_unwind/lib.rs:98 8: 0x55c39a236006 - std::rt::lang_start::hd7c880a37a646e81 at /buildslave/rust-buildbot/slave/stable-dist-rustc-linux/build/src/libstd/panicking.rs:436 at /buildslave/rust-buildbot/slave/stable-dist-rustc-linux/build/src/libstd/panic.rs:361 at /buildslave/rust-buildbot/slave/stable-dist-rustc-linux/build/src/libstd/rt.rs:57 9: 0x55c39a22e9e9 - main 10: 0x7f5e5ed3382f - __libc_start_main 11: 0x55c39a22e6b8 - _start 12: 0x0 - <unknown> |
теперь имеет вид
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 | thread 'main' panicked at 'explicit panic', foo.rs:2 stack backtrace: 0: std::sys::imp::backtrace::tracing::imp::unwind_backtrace at /checkout/src/libstd/sys/unix/backtrace/tracing/gcc_s.rs:49 1: std::sys_common::backtrace::_print at /checkout/src/libstd/sys_common/backtrace.rs:71 2: std::panicking::default_hook:: at /checkout/src/libstd/sys_common/backtrace.rs:60 at /checkout/src/libstd/panicking.rs:355 3: std::panicking::default_hook at /checkout/src/libstd/panicking.rs:371 4: std::panicking::rust_panic_with_hook at /checkout/src/libstd/panicking.rs:549 5: std::panicking::begin_panic 6: foo::main 7: __rust_maybe_catch_panic at /checkout/src/libpanic_unwind/lib.rs:98 8: std::rt::lang_start at /checkout/src/libstd/panicking.rs:433 at /checkout/src/libstd/panic.rs:361 at /checkout/src/libstd/rt.rs:57 9: main 10: __libc_start_main 11: _start |
по умолчанию. Вы можете установить переменную окружения RUST_BACKTRACE = full
, чтобы получить полную обратную трассировку. В будущем мы планируем убрать ещё больше лишней информации; См. эту ошибку.
Подробнее смотрите примечания к выпуску.
Возможности Cargo
Помимо ранее упомянутых изменений, касающихся build.rs
, у Cargo есть ещё несколько новых улучшений. cargo check --all
и cargo run --package
— два отсутствовавших до этого момента флага, которые теперь поддерживаются.
Теперь вы можете игнорировать проверки отзыва SSL. Конечно, по умолчанию проверка все ещё производится.
Новое поле в Cargo.toml
, required-features
, позволяет указать конкретные возможности, которые должны быть установлены для цели, которую нужно собрать. Вот пример: предположим, что мы пишем контейнер, который взаимодействует с базами данных, и хотим, чтобы он поддерживал несколько баз данных. Мы могли бы это сделать в нашем Cargo.toml
:
1 2 3 4 5 | [features] # ... postgres = [] sqlite = [] tools = [] |
Возможность tools
позволяет нам включить дополнительные инструменты, а возможности postgres
и sqlite
указывают, какие базы данных мы хотим поддерживать.
Раньше cargo build
пыталась собрать все цели, которые вам нужны. Но что, если у нас есть файл src/bin/postgres-tool.rs
, который является нужным только при условии, что возможности postgres
и tools
были включены? Раньше нам приходилось писать что-то вроде этого:
1 2 3 4 5 6 7 8 9 | #[cfg(not(all(feature = "postgres", feature = "tools")))] fn main() { println!("This tool requires the `postgres` and `tools` features to be enabled."); } #[cfg(all(feature = "postgres", feature = "tools"))] fn main() { // код } |
Слишком много шаблонного кода для работы с cargo build
. Ещё печальнее дело обстояло с examples/
, которые должны демонстрировать, как использовать вашу библиотеку. Но такие махинации возможны только при работе внутри самого пакета, поэтому вы потерпите неудачу, если попытаетесь использовать пример вне этого пакета.
С помощью нового ключа required-features
мы можем добавить следующее:
1 2 3 | [[bin]] # ... required-features = ["postgres", "tools"] |
Теперь cargo build
будет собирать наш postgres-tool
, только если у нас включены две эти возможности, и поэтому мы можем написать нормальный fn main
без всяких нагромождений вроде cfg
.
Подробнее смотрите примечания к выпуску.
Разработчики версии 1.17.0
Много людей внесли свой вклад в создание Rust 1.17. Мы не смогли бы сделать это без всех вас. Спасибо!