Rust - generic closure exercise
$begingroup$
The official Rust book chapter 13.1 includes an exercise to expand on the example provided in the chapter:
Try modifying Cacher to hold a hash map rather than a single value. The keys of the hash map will be the arg values that are passed in, and the values of the hash map will be the result of calling the closure on that key. Instead of looking at whether self.value directly has a Some or a None value, the value function will look up the arg in the hash map and return the value if it’s present. If it’s not present, the Cacher will call the closure and save the resulting value in the hash map associated with its arg value.
The second problem with the current Cacher implementation is that it only accepts closures that take one parameter of type u32 and return a u32. We might want to cache the results of closures that take a string slice and return usize values, for example. To fix this issue, try introducing more generic parameters to increase the flexibility of the Cacher functionality.
The following is what I have:
use std::thread;
use std::time::Duration;
use std::collections::HashMap;
use std::hash::Hash;
struct Cacher<T, K, J>
where T: Fn(&K) -> J,
K: Hash + Eq,
J: Clone
{
calculation: T,
value: HashMap<K, J>,
}
impl<T, K, J> Cacher<T, K, J>
where T: Fn(&K) -> J,
K: Hash + Eq,
J: Clone
{
fn new(calculation: T) -> Cacher<T, K, J> {
Cacher {
calculation,
value: HashMap::new(),
}
}
fn value(&mut self, arg: K) -> J {
if let Some(v) = self.value.get(&arg) {
v.clone()
} else {
let v = (self.calculation)(&arg);
self.value.insert(arg, v.clone());
v
}
}
}
fn generate_workout(intensity: u32, random_number: u32) {
let mut expensive_result = Cacher::new(|&num| {
println!("calculating slowly...");
thread::sleep(Duration::from_secs(2));
num
});
if intensity < 25 {
println!(
"Today, do {} pushups!",
expensive_result.value(&intensity)
);
println!(
"Next, do {} situps!",
expensive_result.value(&intensity)
);
} else {
if random_number == 3 {
println!("Take a break today! Remember to stay hydrated!");
} else {
println!(
"Today, run for {} minutes!",
expensive_result.value(&intensity)
)
}
}
}
fn main() {
let simulated_user_specified_value = 10;
let simulated_random_number = 7;
generate_workout(
simulated_user_specified_value,
simulated_random_number
);
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn call_with_different_values() {
let mut c = Cacher::new(|&a| a);
let v1 = c.value(1);
let v2 = c.value(2);
assert_eq!(v1, 1);
assert_eq!(v2, 2);
let mut d = Cacher::new (|a: &String| a.len());
let str1 = String::from("abc");
let v3 = d.value(str1);
assert_eq!(v3, 3);
}
}
I don't like the fact that my approach has the generic parameter J
bounded by Clone
trait. How can I make this work without the bound?
Any other feedback is appreciated.
beginner rust
New contributor
$endgroup$
add a comment |
$begingroup$
The official Rust book chapter 13.1 includes an exercise to expand on the example provided in the chapter:
Try modifying Cacher to hold a hash map rather than a single value. The keys of the hash map will be the arg values that are passed in, and the values of the hash map will be the result of calling the closure on that key. Instead of looking at whether self.value directly has a Some or a None value, the value function will look up the arg in the hash map and return the value if it’s present. If it’s not present, the Cacher will call the closure and save the resulting value in the hash map associated with its arg value.
The second problem with the current Cacher implementation is that it only accepts closures that take one parameter of type u32 and return a u32. We might want to cache the results of closures that take a string slice and return usize values, for example. To fix this issue, try introducing more generic parameters to increase the flexibility of the Cacher functionality.
The following is what I have:
use std::thread;
use std::time::Duration;
use std::collections::HashMap;
use std::hash::Hash;
struct Cacher<T, K, J>
where T: Fn(&K) -> J,
K: Hash + Eq,
J: Clone
{
calculation: T,
value: HashMap<K, J>,
}
impl<T, K, J> Cacher<T, K, J>
where T: Fn(&K) -> J,
K: Hash + Eq,
J: Clone
{
fn new(calculation: T) -> Cacher<T, K, J> {
Cacher {
calculation,
value: HashMap::new(),
}
}
fn value(&mut self, arg: K) -> J {
if let Some(v) = self.value.get(&arg) {
v.clone()
} else {
let v = (self.calculation)(&arg);
self.value.insert(arg, v.clone());
v
}
}
}
fn generate_workout(intensity: u32, random_number: u32) {
let mut expensive_result = Cacher::new(|&num| {
println!("calculating slowly...");
thread::sleep(Duration::from_secs(2));
num
});
if intensity < 25 {
println!(
"Today, do {} pushups!",
expensive_result.value(&intensity)
);
println!(
"Next, do {} situps!",
expensive_result.value(&intensity)
);
} else {
if random_number == 3 {
println!("Take a break today! Remember to stay hydrated!");
} else {
println!(
"Today, run for {} minutes!",
expensive_result.value(&intensity)
)
}
}
}
fn main() {
let simulated_user_specified_value = 10;
let simulated_random_number = 7;
generate_workout(
simulated_user_specified_value,
simulated_random_number
);
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn call_with_different_values() {
let mut c = Cacher::new(|&a| a);
let v1 = c.value(1);
let v2 = c.value(2);
assert_eq!(v1, 1);
assert_eq!(v2, 2);
let mut d = Cacher::new (|a: &String| a.len());
let str1 = String::from("abc");
let v3 = d.value(str1);
assert_eq!(v3, 3);
}
}
I don't like the fact that my approach has the generic parameter J
bounded by Clone
trait. How can I make this work without the bound?
Any other feedback is appreciated.
beginner rust
New contributor
$endgroup$
add a comment |
$begingroup$
The official Rust book chapter 13.1 includes an exercise to expand on the example provided in the chapter:
Try modifying Cacher to hold a hash map rather than a single value. The keys of the hash map will be the arg values that are passed in, and the values of the hash map will be the result of calling the closure on that key. Instead of looking at whether self.value directly has a Some or a None value, the value function will look up the arg in the hash map and return the value if it’s present. If it’s not present, the Cacher will call the closure and save the resulting value in the hash map associated with its arg value.
The second problem with the current Cacher implementation is that it only accepts closures that take one parameter of type u32 and return a u32. We might want to cache the results of closures that take a string slice and return usize values, for example. To fix this issue, try introducing more generic parameters to increase the flexibility of the Cacher functionality.
The following is what I have:
use std::thread;
use std::time::Duration;
use std::collections::HashMap;
use std::hash::Hash;
struct Cacher<T, K, J>
where T: Fn(&K) -> J,
K: Hash + Eq,
J: Clone
{
calculation: T,
value: HashMap<K, J>,
}
impl<T, K, J> Cacher<T, K, J>
where T: Fn(&K) -> J,
K: Hash + Eq,
J: Clone
{
fn new(calculation: T) -> Cacher<T, K, J> {
Cacher {
calculation,
value: HashMap::new(),
}
}
fn value(&mut self, arg: K) -> J {
if let Some(v) = self.value.get(&arg) {
v.clone()
} else {
let v = (self.calculation)(&arg);
self.value.insert(arg, v.clone());
v
}
}
}
fn generate_workout(intensity: u32, random_number: u32) {
let mut expensive_result = Cacher::new(|&num| {
println!("calculating slowly...");
thread::sleep(Duration::from_secs(2));
num
});
if intensity < 25 {
println!(
"Today, do {} pushups!",
expensive_result.value(&intensity)
);
println!(
"Next, do {} situps!",
expensive_result.value(&intensity)
);
} else {
if random_number == 3 {
println!("Take a break today! Remember to stay hydrated!");
} else {
println!(
"Today, run for {} minutes!",
expensive_result.value(&intensity)
)
}
}
}
fn main() {
let simulated_user_specified_value = 10;
let simulated_random_number = 7;
generate_workout(
simulated_user_specified_value,
simulated_random_number
);
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn call_with_different_values() {
let mut c = Cacher::new(|&a| a);
let v1 = c.value(1);
let v2 = c.value(2);
assert_eq!(v1, 1);
assert_eq!(v2, 2);
let mut d = Cacher::new (|a: &String| a.len());
let str1 = String::from("abc");
let v3 = d.value(str1);
assert_eq!(v3, 3);
}
}
I don't like the fact that my approach has the generic parameter J
bounded by Clone
trait. How can I make this work without the bound?
Any other feedback is appreciated.
beginner rust
New contributor
$endgroup$
The official Rust book chapter 13.1 includes an exercise to expand on the example provided in the chapter:
Try modifying Cacher to hold a hash map rather than a single value. The keys of the hash map will be the arg values that are passed in, and the values of the hash map will be the result of calling the closure on that key. Instead of looking at whether self.value directly has a Some or a None value, the value function will look up the arg in the hash map and return the value if it’s present. If it’s not present, the Cacher will call the closure and save the resulting value in the hash map associated with its arg value.
The second problem with the current Cacher implementation is that it only accepts closures that take one parameter of type u32 and return a u32. We might want to cache the results of closures that take a string slice and return usize values, for example. To fix this issue, try introducing more generic parameters to increase the flexibility of the Cacher functionality.
The following is what I have:
use std::thread;
use std::time::Duration;
use std::collections::HashMap;
use std::hash::Hash;
struct Cacher<T, K, J>
where T: Fn(&K) -> J,
K: Hash + Eq,
J: Clone
{
calculation: T,
value: HashMap<K, J>,
}
impl<T, K, J> Cacher<T, K, J>
where T: Fn(&K) -> J,
K: Hash + Eq,
J: Clone
{
fn new(calculation: T) -> Cacher<T, K, J> {
Cacher {
calculation,
value: HashMap::new(),
}
}
fn value(&mut self, arg: K) -> J {
if let Some(v) = self.value.get(&arg) {
v.clone()
} else {
let v = (self.calculation)(&arg);
self.value.insert(arg, v.clone());
v
}
}
}
fn generate_workout(intensity: u32, random_number: u32) {
let mut expensive_result = Cacher::new(|&num| {
println!("calculating slowly...");
thread::sleep(Duration::from_secs(2));
num
});
if intensity < 25 {
println!(
"Today, do {} pushups!",
expensive_result.value(&intensity)
);
println!(
"Next, do {} situps!",
expensive_result.value(&intensity)
);
} else {
if random_number == 3 {
println!("Take a break today! Remember to stay hydrated!");
} else {
println!(
"Today, run for {} minutes!",
expensive_result.value(&intensity)
)
}
}
}
fn main() {
let simulated_user_specified_value = 10;
let simulated_random_number = 7;
generate_workout(
simulated_user_specified_value,
simulated_random_number
);
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn call_with_different_values() {
let mut c = Cacher::new(|&a| a);
let v1 = c.value(1);
let v2 = c.value(2);
assert_eq!(v1, 1);
assert_eq!(v2, 2);
let mut d = Cacher::new (|a: &String| a.len());
let str1 = String::from("abc");
let v3 = d.value(str1);
assert_eq!(v3, 3);
}
}
I don't like the fact that my approach has the generic parameter J
bounded by Clone
trait. How can I make this work without the bound?
Any other feedback is appreciated.
beginner rust
beginner rust
New contributor
New contributor
New contributor
asked 1 hour ago
qwertyqwerty
11
11
New contributor
New contributor
add a comment |
add a comment |
0
active
oldest
votes
Your Answer
StackExchange.ifUsing("editor", function () {
return StackExchange.using("mathjaxEditing", function () {
StackExchange.MarkdownEditor.creationCallbacks.add(function (editor, postfix) {
StackExchange.mathjaxEditing.prepareWmdForMathJax(editor, postfix, [["\$", "\$"]]);
});
});
}, "mathjax-editing");
StackExchange.ifUsing("editor", function () {
StackExchange.using("externalEditor", function () {
StackExchange.using("snippets", function () {
StackExchange.snippets.init();
});
});
}, "code-snippets");
StackExchange.ready(function() {
var channelOptions = {
tags: "".split(" "),
id: "196"
};
initTagRenderer("".split(" "), "".split(" "), channelOptions);
StackExchange.using("externalEditor", function() {
// Have to fire editor after snippets, if snippets enabled
if (StackExchange.settings.snippets.snippetsEnabled) {
StackExchange.using("snippets", function() {
createEditor();
});
}
else {
createEditor();
}
});
function createEditor() {
StackExchange.prepareEditor({
heartbeatType: 'answer',
autoActivateHeartbeat: false,
convertImagesToLinks: false,
noModals: true,
showLowRepImageUploadWarning: true,
reputationToPostImages: null,
bindNavPrevention: true,
postfix: "",
imageUploader: {
brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
allowUrls: true
},
onDemand: true,
discardSelector: ".discard-answer"
,immediatelyShowMarkdownHelp:true
});
}
});
qwerty is a new contributor. Be nice, and check out our Code of Conduct.
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f213927%2frust-generic-closure-exercise%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
0
active
oldest
votes
0
active
oldest
votes
active
oldest
votes
active
oldest
votes
qwerty is a new contributor. Be nice, and check out our Code of Conduct.
qwerty is a new contributor. Be nice, and check out our Code of Conduct.
qwerty is a new contributor. Be nice, and check out our Code of Conduct.
qwerty is a new contributor. Be nice, and check out our Code of Conduct.
Thanks for contributing an answer to Code Review Stack Exchange!
- Please be sure to answer the question. Provide details and share your research!
But avoid …
- Asking for help, clarification, or responding to other answers.
- Making statements based on opinion; back them up with references or personal experience.
Use MathJax to format equations. MathJax reference.
To learn more, see our tips on writing great answers.
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f213927%2frust-generic-closure-exercise%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown