mirror of
https://github.com/notohh/rustlings.git
synced 2024-10-30 08:49:12 -04:00
cow1 solution
This commit is contained in:
parent
a943f5ba32
commit
663a03a17b
3 changed files with 106 additions and 47 deletions
|
@ -1,24 +1,18 @@
|
||||||
// This exercise explores the Cow, or Clone-On-Write type. Cow is a
|
// This exercise explores the `Cow` (Clone-On-Write) smart pointer. It can
|
||||||
// clone-on-write smart pointer. It can enclose and provide immutable access to
|
// enclose and provide immutable access to borrowed data and clone the data
|
||||||
// borrowed data, and clone the data lazily when mutation or ownership is
|
// lazily when mutation or ownership is required. The type is designed to work
|
||||||
// required. The type is designed to work with general borrowed data via the
|
// with general borrowed data via the `Borrow` trait.
|
||||||
// Borrow trait.
|
|
||||||
//
|
|
||||||
// This exercise is meant to show you what to expect when passing data to Cow.
|
|
||||||
// Fix the unit tests by checking for Cow::Owned(_) and Cow::Borrowed(_) at the
|
|
||||||
// TODO markers.
|
|
||||||
|
|
||||||
use std::borrow::Cow;
|
use std::borrow::Cow;
|
||||||
|
|
||||||
fn abs_all<'a, 'b>(input: &'a mut Cow<'b, [i32]>) -> &'a mut Cow<'b, [i32]> {
|
fn abs_all(input: &mut Cow<[i32]>) {
|
||||||
for i in 0..input.len() {
|
for ind in 0..input.len() {
|
||||||
let v = input[i];
|
let value = input[ind];
|
||||||
if v < 0 {
|
if value < 0 {
|
||||||
// Clones into a vector if not already owned.
|
// Clones into a vector if not already owned.
|
||||||
input.to_mut()[i] = -v;
|
input.to_mut()[ind] = -value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
input
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
|
@ -30,47 +24,45 @@ mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn reference_mutation() -> Result<(), &'static str> {
|
fn reference_mutation() {
|
||||||
// Clone occurs because `input` needs to be mutated.
|
// Clone occurs because `input` needs to be mutated.
|
||||||
let slice = [-1, 0, 1];
|
let vec = vec![-1, 0, 1];
|
||||||
let mut input = Cow::from(&slice[..]);
|
let mut input = Cow::from(&vec);
|
||||||
match abs_all(&mut input) {
|
abs_all(&mut input);
|
||||||
Cow::Owned(_) => Ok(()),
|
assert!(matches!(input, Cow::Owned(_)));
|
||||||
_ => Err("Expected owned value"),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn reference_no_mutation() -> Result<(), &'static str> {
|
fn reference_no_mutation() {
|
||||||
// No clone occurs because `input` doesn't need to be mutated.
|
// No clone occurs because `input` doesn't need to be mutated.
|
||||||
let slice = [0, 1, 2];
|
let vec = vec![0, 1, 2];
|
||||||
let mut input = Cow::from(&slice[..]);
|
let mut input = Cow::from(&vec);
|
||||||
match abs_all(&mut input) {
|
abs_all(&mut input);
|
||||||
// TODO
|
// TODO: Replace `todo!()` with `Cow::Owned(_)` or `Cow::Borrowed(_)`.
|
||||||
}
|
assert!(matches!(input, todo!()));
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn owned_no_mutation() -> Result<(), &'static str> {
|
fn owned_no_mutation() {
|
||||||
// We can also pass `slice` without `&` so Cow owns it directly. In this
|
// We can also pass `vec` without `&` so `Cow` owns it directly. In this
|
||||||
// case no mutation occurs and thus also no clone, but the result is
|
// case, no mutation occurs and thus also no clone. But the result is
|
||||||
// still owned because it was never borrowed or mutated.
|
// still owned because it was never borrowed or mutated.
|
||||||
let slice = vec![0, 1, 2];
|
let vec = vec![0, 1, 2];
|
||||||
let mut input = Cow::from(slice);
|
let mut input = Cow::from(vec);
|
||||||
match abs_all(&mut input) {
|
abs_all(&mut input);
|
||||||
// TODO
|
// TODO: Replace `todo!()` with `Cow::Owned(_)` or `Cow::Borrowed(_)`.
|
||||||
}
|
assert!(matches!(input, todo!()));
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn owned_mutation() -> Result<(), &'static str> {
|
fn owned_mutation() {
|
||||||
// Of course this is also the case if a mutation does occur. In this
|
// Of course this is also the case if a mutation does occur. In this
|
||||||
// case the call to `to_mut()` in the abs_all() function returns a
|
// case, the call to `to_mut()` in the `abs_all` function returns a
|
||||||
// reference to the same data as before.
|
// reference to the same data as before.
|
||||||
let slice = vec![-1, 0, 1];
|
let vec = vec![-1, 0, 1];
|
||||||
let mut input = Cow::from(slice);
|
let mut input = Cow::from(vec);
|
||||||
match abs_all(&mut input) {
|
abs_all(&mut input);
|
||||||
// TODO
|
// TODO: Replace `todo!()` with `Cow::Owned(_)` or `Cow::Borrowed(_)`.
|
||||||
}
|
assert!(matches!(input, todo!()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1020,11 +1020,11 @@ https://doc.rust-lang.org/stable/book/ch16-00-concurrency.html"""
|
||||||
name = "cow1"
|
name = "cow1"
|
||||||
dir = "19_smart_pointers"
|
dir = "19_smart_pointers"
|
||||||
hint = """
|
hint = """
|
||||||
If `Cow` already owns the data it doesn't need to clone it when `to_mut()` is
|
If `Cow` already owns the data, it doesn't need to clone it when `to_mut()` is
|
||||||
called.
|
called.
|
||||||
|
|
||||||
Check out https://doc.rust-lang.org/std/borrow/enum.Cow.html for documentation
|
Check out the documentation of the `Cow` type:
|
||||||
on the `Cow` type."""
|
https://doc.rust-lang.org/std/borrow/enum.Cow.html"""
|
||||||
|
|
||||||
# THREADS
|
# THREADS
|
||||||
|
|
||||||
|
|
|
@ -1 +1,68 @@
|
||||||
// Solutions will be available before the stable release. Thank you for testing the beta version 🥰
|
// This exercise explores the `Cow` (Clone-On-Write) smart pointer. It can
|
||||||
|
// enclose and provide immutable access to borrowed data and clone the data
|
||||||
|
// lazily when mutation or ownership is required. The type is designed to work
|
||||||
|
// with general borrowed data via the `Borrow` trait.
|
||||||
|
|
||||||
|
use std::borrow::Cow;
|
||||||
|
|
||||||
|
fn abs_all(input: &mut Cow<[i32]>) {
|
||||||
|
for ind in 0..input.len() {
|
||||||
|
let value = input[ind];
|
||||||
|
if value < 0 {
|
||||||
|
// Clones into a vector if not already owned.
|
||||||
|
input.to_mut()[ind] = -value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
// You can optionally experiment here.
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn reference_mutation() {
|
||||||
|
// Clone occurs because `input` needs to be mutated.
|
||||||
|
let vec = vec![-1, 0, 1];
|
||||||
|
let mut input = Cow::from(&vec);
|
||||||
|
abs_all(&mut input);
|
||||||
|
assert!(matches!(input, Cow::Owned(_)));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn reference_no_mutation() {
|
||||||
|
// No clone occurs because `input` doesn't need to be mutated.
|
||||||
|
let vec = vec![0, 1, 2];
|
||||||
|
let mut input = Cow::from(&vec);
|
||||||
|
abs_all(&mut input);
|
||||||
|
assert!(matches!(input, Cow::Borrowed(_)));
|
||||||
|
// ^^^^^^^^^^^^^^^^
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn owned_no_mutation() {
|
||||||
|
// We can also pass `vec` without `&` so `Cow` owns it directly. In this
|
||||||
|
// case, no mutation occurs and thus also no clone. But the result is
|
||||||
|
// still owned because it was never borrowed or mutated.
|
||||||
|
let vec = vec![0, 1, 2];
|
||||||
|
let mut input = Cow::from(vec);
|
||||||
|
abs_all(&mut input);
|
||||||
|
assert!(matches!(input, Cow::Owned(_)));
|
||||||
|
// ^^^^^^^^^^^^^
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn owned_mutation() {
|
||||||
|
// Of course this is also the case if a mutation does occur. In this
|
||||||
|
// case, the call to `to_mut()` in the `abs_all` function returns a
|
||||||
|
// reference to the same data as before.
|
||||||
|
let vec = vec![-1, 0, 1];
|
||||||
|
let mut input = Cow::from(vec);
|
||||||
|
abs_all(&mut input);
|
||||||
|
assert!(matches!(input, Cow::Owned(_)));
|
||||||
|
// ^^^^^^^^^^^^^
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue