Is there a way to split up a variable inside an Option without having to use if statements?












0















I want the code below to work, but since map() takes ownership of Option, and there doesn't seem to be a clone() function for Option, the following doesn't compile.



fn main() {
struct ItemA {
a: String,
b: String,
}
let foo = Some(ItemA {
a: "A String".to_owned(),
b: "B String".to_owned(),
});
// OR
// let foo = None;

let opA: Option<String> = foo.map(|c| c.a);
let opB: Option<String> = foo.map(|c| c.b);
}


error[E0382]: use of moved value: `foo`
--> src/main.rs:15:31
|
14 | let opA: Option<String> = foo.map(|c| c.a);
| --- value moved here
15 | let opB: Option<String> = foo.map(|c| c.b);
| ^^^ value used here after move
|
= note: move occurs because `foo` has type `std::option::Option<main::ItemA>`, which does not implement the `Copy` trait


It would be nice if opA can take ownership of ItemA.a (so it doesn't have to clone the strings), and opB can take ownership of ItemA.b



Is this possible to do without having to use if statements to check if the Option is Some or None, unwrapping, and wrapping it back up individually.










share|improve this question

























  • Idiomatic Rust uses snake_case for variables, methods, macros, and fields; UpperCamelCase for types; and SCREAMING_SNAKE_CASE for statics and constants. Use op_a / op_b instead, please.

    – Shepmaster
    Nov 22 '18 at 0:00
















0















I want the code below to work, but since map() takes ownership of Option, and there doesn't seem to be a clone() function for Option, the following doesn't compile.



fn main() {
struct ItemA {
a: String,
b: String,
}
let foo = Some(ItemA {
a: "A String".to_owned(),
b: "B String".to_owned(),
});
// OR
// let foo = None;

let opA: Option<String> = foo.map(|c| c.a);
let opB: Option<String> = foo.map(|c| c.b);
}


error[E0382]: use of moved value: `foo`
--> src/main.rs:15:31
|
14 | let opA: Option<String> = foo.map(|c| c.a);
| --- value moved here
15 | let opB: Option<String> = foo.map(|c| c.b);
| ^^^ value used here after move
|
= note: move occurs because `foo` has type `std::option::Option<main::ItemA>`, which does not implement the `Copy` trait


It would be nice if opA can take ownership of ItemA.a (so it doesn't have to clone the strings), and opB can take ownership of ItemA.b



Is this possible to do without having to use if statements to check if the Option is Some or None, unwrapping, and wrapping it back up individually.










share|improve this question

























  • Idiomatic Rust uses snake_case for variables, methods, macros, and fields; UpperCamelCase for types; and SCREAMING_SNAKE_CASE for statics and constants. Use op_a / op_b instead, please.

    – Shepmaster
    Nov 22 '18 at 0:00














0












0








0








I want the code below to work, but since map() takes ownership of Option, and there doesn't seem to be a clone() function for Option, the following doesn't compile.



fn main() {
struct ItemA {
a: String,
b: String,
}
let foo = Some(ItemA {
a: "A String".to_owned(),
b: "B String".to_owned(),
});
// OR
// let foo = None;

let opA: Option<String> = foo.map(|c| c.a);
let opB: Option<String> = foo.map(|c| c.b);
}


error[E0382]: use of moved value: `foo`
--> src/main.rs:15:31
|
14 | let opA: Option<String> = foo.map(|c| c.a);
| --- value moved here
15 | let opB: Option<String> = foo.map(|c| c.b);
| ^^^ value used here after move
|
= note: move occurs because `foo` has type `std::option::Option<main::ItemA>`, which does not implement the `Copy` trait


It would be nice if opA can take ownership of ItemA.a (so it doesn't have to clone the strings), and opB can take ownership of ItemA.b



Is this possible to do without having to use if statements to check if the Option is Some or None, unwrapping, and wrapping it back up individually.










share|improve this question
















I want the code below to work, but since map() takes ownership of Option, and there doesn't seem to be a clone() function for Option, the following doesn't compile.



fn main() {
struct ItemA {
a: String,
b: String,
}
let foo = Some(ItemA {
a: "A String".to_owned(),
b: "B String".to_owned(),
});
// OR
// let foo = None;

let opA: Option<String> = foo.map(|c| c.a);
let opB: Option<String> = foo.map(|c| c.b);
}


error[E0382]: use of moved value: `foo`
--> src/main.rs:15:31
|
14 | let opA: Option<String> = foo.map(|c| c.a);
| --- value moved here
15 | let opB: Option<String> = foo.map(|c| c.b);
| ^^^ value used here after move
|
= note: move occurs because `foo` has type `std::option::Option<main::ItemA>`, which does not implement the `Copy` trait


It would be nice if opA can take ownership of ItemA.a (so it doesn't have to clone the strings), and opB can take ownership of ItemA.b



Is this possible to do without having to use if statements to check if the Option is Some or None, unwrapping, and wrapping it back up individually.







rust ownership






share|improve this question















share|improve this question













share|improve this question




share|improve this question








edited Nov 22 '18 at 0:00









Shepmaster

155k14307452




155k14307452










asked Nov 21 '18 at 22:57









Prime_AqasixPrime_Aqasix

510514




510514













  • Idiomatic Rust uses snake_case for variables, methods, macros, and fields; UpperCamelCase for types; and SCREAMING_SNAKE_CASE for statics and constants. Use op_a / op_b instead, please.

    – Shepmaster
    Nov 22 '18 at 0:00



















  • Idiomatic Rust uses snake_case for variables, methods, macros, and fields; UpperCamelCase for types; and SCREAMING_SNAKE_CASE for statics and constants. Use op_a / op_b instead, please.

    – Shepmaster
    Nov 22 '18 at 0:00

















Idiomatic Rust uses snake_case for variables, methods, macros, and fields; UpperCamelCase for types; and SCREAMING_SNAKE_CASE for statics and constants. Use op_a / op_b instead, please.

– Shepmaster
Nov 22 '18 at 0:00





Idiomatic Rust uses snake_case for variables, methods, macros, and fields; UpperCamelCase for types; and SCREAMING_SNAKE_CASE for statics and constants. Use op_a / op_b instead, please.

– Shepmaster
Nov 22 '18 at 0:00












2 Answers
2






active

oldest

votes


















1














You can use map_or_else.



let (opA, opB) = foo.map_or_else(
|| (None, None),
|c| (Some(c.a), Some(c.b))
);


The first function is called if foo is None, and returns two Nones. The second function is called if foo is Some and splits the members into a tuple.



Of course, this doesn't really save you much over a simple match, and will probably be harder to follow.



let (opA, opB) = match foo {
None => (None, None),
Some(c) => (Some(c.a), Some(c.b))
};


By the way, Option does implement Clone, but it requires that the contained type implements Clone.






share|improve this answer

































    -1















    [T]here doesn't seem to be a clone() function for Option [...]




    That's wrong, there is, if the inner type implements clone as well. So in your case, just add #[derive(Clone)] to your struct ItemA.



    Next, you can use as_ref to create an Option<&T> and then use map.



    let op_a = foo.as_ref().map(|c| &c.a);
    let op_b = foo.as_ref().map(|c| &c.b);

    println!("{:?}", op_a);
    println!("{:?}", op_b);





    share|improve this answer























      Your Answer






      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: "1"
      };
      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: true,
      noModals: true,
      showLowRepImageUploadWarning: true,
      reputationToPostImages: 10,
      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
      });


      }
      });














      draft saved

      draft discarded


















      StackExchange.ready(
      function () {
      StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f53421617%2fis-there-a-way-to-split-up-a-variable-inside-an-option-without-having-to-use-if%23new-answer', 'question_page');
      }
      );

      Post as a guest















      Required, but never shown

























      2 Answers
      2






      active

      oldest

      votes








      2 Answers
      2






      active

      oldest

      votes









      active

      oldest

      votes






      active

      oldest

      votes









      1














      You can use map_or_else.



      let (opA, opB) = foo.map_or_else(
      || (None, None),
      |c| (Some(c.a), Some(c.b))
      );


      The first function is called if foo is None, and returns two Nones. The second function is called if foo is Some and splits the members into a tuple.



      Of course, this doesn't really save you much over a simple match, and will probably be harder to follow.



      let (opA, opB) = match foo {
      None => (None, None),
      Some(c) => (Some(c.a), Some(c.b))
      };


      By the way, Option does implement Clone, but it requires that the contained type implements Clone.






      share|improve this answer






























        1














        You can use map_or_else.



        let (opA, opB) = foo.map_or_else(
        || (None, None),
        |c| (Some(c.a), Some(c.b))
        );


        The first function is called if foo is None, and returns two Nones. The second function is called if foo is Some and splits the members into a tuple.



        Of course, this doesn't really save you much over a simple match, and will probably be harder to follow.



        let (opA, opB) = match foo {
        None => (None, None),
        Some(c) => (Some(c.a), Some(c.b))
        };


        By the way, Option does implement Clone, but it requires that the contained type implements Clone.






        share|improve this answer




























          1












          1








          1







          You can use map_or_else.



          let (opA, opB) = foo.map_or_else(
          || (None, None),
          |c| (Some(c.a), Some(c.b))
          );


          The first function is called if foo is None, and returns two Nones. The second function is called if foo is Some and splits the members into a tuple.



          Of course, this doesn't really save you much over a simple match, and will probably be harder to follow.



          let (opA, opB) = match foo {
          None => (None, None),
          Some(c) => (Some(c.a), Some(c.b))
          };


          By the way, Option does implement Clone, but it requires that the contained type implements Clone.






          share|improve this answer















          You can use map_or_else.



          let (opA, opB) = foo.map_or_else(
          || (None, None),
          |c| (Some(c.a), Some(c.b))
          );


          The first function is called if foo is None, and returns two Nones. The second function is called if foo is Some and splits the members into a tuple.



          Of course, this doesn't really save you much over a simple match, and will probably be harder to follow.



          let (opA, opB) = match foo {
          None => (None, None),
          Some(c) => (Some(c.a), Some(c.b))
          };


          By the way, Option does implement Clone, but it requires that the contained type implements Clone.







          share|improve this answer














          share|improve this answer



          share|improve this answer








          edited Nov 21 '18 at 23:41

























          answered Nov 21 '18 at 23:18









          Benjamin LindleyBenjamin Lindley

          86.1k4137227




          86.1k4137227

























              -1















              [T]here doesn't seem to be a clone() function for Option [...]




              That's wrong, there is, if the inner type implements clone as well. So in your case, just add #[derive(Clone)] to your struct ItemA.



              Next, you can use as_ref to create an Option<&T> and then use map.



              let op_a = foo.as_ref().map(|c| &c.a);
              let op_b = foo.as_ref().map(|c| &c.b);

              println!("{:?}", op_a);
              println!("{:?}", op_b);





              share|improve this answer




























                -1















                [T]here doesn't seem to be a clone() function for Option [...]




                That's wrong, there is, if the inner type implements clone as well. So in your case, just add #[derive(Clone)] to your struct ItemA.



                Next, you can use as_ref to create an Option<&T> and then use map.



                let op_a = foo.as_ref().map(|c| &c.a);
                let op_b = foo.as_ref().map(|c| &c.b);

                println!("{:?}", op_a);
                println!("{:?}", op_b);





                share|improve this answer


























                  -1












                  -1








                  -1








                  [T]here doesn't seem to be a clone() function for Option [...]




                  That's wrong, there is, if the inner type implements clone as well. So in your case, just add #[derive(Clone)] to your struct ItemA.



                  Next, you can use as_ref to create an Option<&T> and then use map.



                  let op_a = foo.as_ref().map(|c| &c.a);
                  let op_b = foo.as_ref().map(|c| &c.b);

                  println!("{:?}", op_a);
                  println!("{:?}", op_b);





                  share|improve this answer














                  [T]here doesn't seem to be a clone() function for Option [...]




                  That's wrong, there is, if the inner type implements clone as well. So in your case, just add #[derive(Clone)] to your struct ItemA.



                  Next, you can use as_ref to create an Option<&T> and then use map.



                  let op_a = foo.as_ref().map(|c| &c.a);
                  let op_b = foo.as_ref().map(|c| &c.b);

                  println!("{:?}", op_a);
                  println!("{:?}", op_b);






                  share|improve this answer












                  share|improve this answer



                  share|improve this answer










                  answered Nov 22 '18 at 7:15









                  hellowhellow

                  5,19242242




                  5,19242242






























                      draft saved

                      draft discarded




















































                      Thanks for contributing an answer to Stack Overflow!


                      • 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.


                      To learn more, see our tips on writing great answers.




                      draft saved


                      draft discarded














                      StackExchange.ready(
                      function () {
                      StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f53421617%2fis-there-a-way-to-split-up-a-variable-inside-an-option-without-having-to-use-if%23new-answer', 'question_page');
                      }
                      );

                      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







                      Popular posts from this blog

                      MongoDB - Not Authorized To Execute Command

                      in spring boot 2.1 many test slices are not allowed anymore due to multiple @BootstrapWith

                      How to fix TextFormField cause rebuild widget in Flutter