質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
Rust

Rustは、MoFoが支援するプログラミング言語。高速性を維持しつつも、メモリ管理を安全に行うことが可能な言語です。同じコンパイル言語であるC言語やC++では困難だったマルチスレッドを実装しやすく、並行性という点においても優れています。

Q&A

1回答

1096閲覧

Rust Ordを定義したStructに実装したい。

izmrui2020

総合スコア0

Rust

Rustは、MoFoが支援するプログラミング言語。高速性を維持しつつも、メモリ管理を安全に行うことが可能な言語です。同じコンパイル言語であるC言語やC++では困難だったマルチスレッドを実装しやすく、並行性という点においても優れています。

0グッド

0クリップ

投稿2022/12/05 12:16

編集2022/12/06 12:00

前提

Binary searchを使いたいので、VectorではなくBtreeSetをcollectionとして使いたいです。
自分で定義したEntry型は外部モジュールなのでderive macroが使えない状況を想定していただけるとありがたいです。

実現したいこと

  • BtreeSetに対してreplace関数によってidと合致したものは上書きをする,

IDがないものに対しては追加する。

  • BtreeSetへの追加で自動的にordがされる。

ordの実装が上手できなくて困っています。
何かわかることがあればぜひご教授お願いします。

Code

外部モジュール

#[derive(Debug, Clone, Hash, PartialEq, Eq)] struct Entry { id: String, value: BigDecimal } impl Entry { fn new(id: String, value: BigDecimal) -> Entry { Entry { id, value } } }

##main.rs

use bigdecimal::BigDecimal; use std::collections::BTreeSet; use std::cmp::Ordering; #[derive(Debug)] struct WrapEntry(Entry); impl WrapEntry { fn new(e: Entry) -> WrapEntry { WrapEntry(e) } } impl Ord for WrapEntry { fn cmp(&self, other: &Self) -> Ordering { if self.0.id == other.0.id { Ordering::Equal } else { match self.0.value.cmp(&other.0.value) { Ordering::Equal => Ordering::Greater, Ordering::Greater => Ordering::Greater, Ordering::Less => Ordering::Less, } } } } impl PartialOrd for WrapEntry { fn partial_cmp(&self, other: &Self) -> Option<Ordering> { Some(self.cmp(&other)) } } impl Eq for WrapEntry {} impl PartialEq for WrapEntry { fn eq(&self, other: &Self) -> bool { self.0.id == other.0.id } } #[tokio::main] async fn main() -> anyhow::Result<()> { println!("Start"); let mut set: BTreeSet<WrapEntry> = BTreeSet::new(); let a1 = WrapEntry::new(Entry::new("aaa".to_string(), BigDecimal::from(100))); let a2 = WrapEntry::new(Entry::new("aaa".to_string(), BigDecimal::from(103))); let a3 = WrapEntry::new(Entry::new("aaa".to_string(), BigDecimal::from(97))); let b1 = WrapEntry::new(Entry::new("bbb".to_string(), BigDecimal::from(100))); let b2 = WrapEntry::new(Entry::new("bbb".to_string(), BigDecimal::from(106))); let b3 = WrapEntry::new(Entry::new("bbb".to_string(), BigDecimal::from(94))); let c1 = WrapEntry::new(Entry::new("ccc".to_string(), BigDecimal::from(100))); let c2 = WrapEntry::new(Entry::new("ccc".to_string(), BigDecimal::from(110))); let c3 = WrapEntry::new(Entry::new("ccc".to_string(), BigDecimal::from(84))); println!("{:?}", set.replace(a1)); println!("{:?}", set.replace(b2)); println!("{:?}", set.replace(c1)); println!("{:?}", set.replace(b3)); println!("{:?}", set.replace(a2)); println!("{:?}", set.replace(c2)); println!("{:?}", set.replace(c3)); println!("{:?}", set.replace(a3)); println!("{:?}", set.replace(b1)); println!("xxxxxxxxxxxxxxxxxx"); for i in set.iter() { println!("{:?}", i); } Ok(()) }

結果

idが上手く認識されないです。

Start None None None None Some(WrapEntry(Entry { id: "aaa", value: BigDecimal("100") })) Some(WrapEntry(Entry { id: "ccc", value: BigDecimal("100") })) None Some(WrapEntry(Entry { id: "aaa", value: BigDecimal("103") })) Some(WrapEntry(Entry { id: "bbb", value: BigDecimal("94") })) xxxxxxxxxxxxxxxxxxxxx WrapEntry(Entry { id: "ccc", value: BigDecimal("84") }) WrapEntry(Entry { id: "bbb", value: BigDecimal("100") }) WrapEntry(Entry { id: "aaa", value: BigDecimal("97") }) WrapEntry(Entry { id: "ccc", value: BigDecimal("110") }) WrapEntry(Entry { id: "bbb", value: BigDecimal("106") })

試したこと

  • 自分のアイディアはEntry型をWrapしてOrdを実装することでWrapEntry型をOrdできるようにしたいと思ってます。

idが違う場合に違うobjectと認識させるためにはどのようにしたらいいか教えてほしいです。
よろしくお願いします。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

blackenedgold

2022/12/16 15:10

回答をしたいのですが、実現したいことが分からないので質問させて下さい。 1. > BtreeSetへの追加で自動的にordがされる。 とはどういう意味でしょう。 2. Entryに対してどのような順序関係を導入しようとしていますか。例えば以下の4つの `Entry` を小さい順に並べるとどうなりますか。 `Entry("aaa", 1)`, `Entry("aaa", 2)`, `Entry("bbb", 1)`, `Entry("bbb", 2)`
guest

回答1

0

<WrapEntity as Ord>::cmpの実装のif self.0.id == self.0.id {if self.0.id == other.0.id {の間違いではないですか?
たぶんそれでちゃんとIDで識別してくれる気がする

投稿2022/12/05 18:16

S.Percentage

総合スコア283

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

izmrui2020

2022/12/06 12:01

ありがとうございます。 確かにselfの部分が間違ってました。 なお、 idが同じ場合に上書きが成功しないところがあるのですが、もし何かわかりましたら教えていただけるとありがたいです。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

まだベストアンサーが選ばれていません

会員登録して回答してみよう

アカウントをお持ちの方は

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問