C ++ std :: pairのC#アナログとは何ですか?


284

興味がある:std::pairC ++でのC#の類似物は何ですか?System.Web.UI.Pairクラスを見つけましたが、テンプレートベースのものがいいです。

ありがとうございました!


11
少し前に同じリクエストがありましたが、考えてみれば、一般的な「First」と「Second」の代わりに、明示的なクラスタイプとフィールドを使用して、独自のペアリングクラスをロールすることもできます。コードを読みやすくします。ペアリングクラスはわずか4行なので、一般的なPair <T、U>クラスを再利用してもそれほど節約できず、コードが読みやすくなります。
マークラカタ2012

回答:


325

タプルは.NET4.0以降で使用でき、ジェネリックをサポートしています

Tuple<string, int> t = new Tuple<string, int>("Hello", 4);

以前のバージョンではSystem.Collections.Generic.KeyValuePair<K, V>、次のようなソリューションを使用できます。

public class Pair<T, U> {
    public Pair() {
    }

    public Pair(T first, U second) {
        this.First = first;
        this.Second = second;
    }

    public T First { get; set; }
    public U Second { get; set; }
};

次のように使用します。

Pair<String, int> pair = new Pair<String, int>("test", 2);
Console.WriteLine(pair.First);
Console.WriteLine(pair.Second);

これは出力します:

test
2

または、この連鎖したペア:

Pair<Pair<String, int>, bool> pair = new Pair<Pair<String, int>, bool>();
pair.First = new Pair<String, int>();
pair.First.First = "test";
pair.First.Second = 12;
pair.Second = true;

Console.WriteLine(pair.First.First);
Console.WriteLine(pair.First.Second);
Console.WriteLine(pair.Second);

それは出力します:

test
12
true

Equalsメソッドの追加に関する私の投稿を参照してください
Andrew Stein

Tuple <>がより優れたソリューションになりました。
dkantowitz 2012

6
ジェネリッククラスに属する型パラメーターはオブジェクト作成式(コンストラクター呼び出し)で推論できないため、BCLの作成者はと呼ばれる非ジェネリックヘルパークラスを作成しましたTuple。したがってTuple.Create("Hello", 4)、どちらが少し簡単かを言うことができますnew Tuple<string, int>("Hello", 4)。(ちなみに、.NET4.0は2010年からすでに導入されています。)
Jeppe Stig Nielsen

4
あなたの心Tuple<>の固形実装EqualsGetHashCode大きな値のセマンティクスで。独自のタプルを実装する場合は注意してください。
nawfal 2014年

これは明らかにEqualsとGetHashCodeにより壊れています
julx

90

System.Web.UIPairASP.NET 1.1では内部ViewState構造体として多用されていたため、このクラスが含まれていました。

2017年8月更新: C#7.0 / .NET Framework 4.7は、System.ValueTuple構造体を使用して名前付きアイテムでタプルを宣言する構文を提供します。

//explicit Item typing
(string Message, int SomeNumber) t = ("Hello", 4);
//or using implicit typing 
var t = (Message:"Hello", SomeNumber:4);

Console.WriteLine("{0} {1}", t.Message, t.SomeNumber);

構文の例については、MSDNを参照してください。

2012年6月の更新: Tuplesバージョン4.0以降、.NETの一部となっています。

これは、.NET4.0への組み込みとジェネリックのサポートについて説明した以前の記事です。

Tuple<string, int> t = new Tuple<string, int>("Hello", 4);

2
タプルは読み取り専用であることに注意してください。:つまり、あなたはこれを行うことはできませんtuple.Item1 = 4;
skybluecodeflier

2
タプルはまさに私が探していたものです。ありがとう。
グリゴラン

38

残念ながらありません。System.Collections.Generic.KeyValuePair<K, V>は多くの状況で使用できます。

あるいは、匿名型を使用して、少なくともローカルでタプルを処理できます。

var x = new { First = "x", Second = 42 };

最後の選択肢は、独自のクラスを作成することです。


2
明確にするために、匿名型も読み取り専用です-msdn
bsegraves 2012年


11

一部の回答は間違っているように見えますが、

  1. (a、b)と(a、c)のペアをどのように格納するかを辞書を使用することはできません。ペアの概念は、キーと値の連想ルックアップと混同しないでください
  2. 上記のコードの多くは疑わしいようです

これが私のペアクラスです

public class Pair<X, Y>
{
    private X _x;
    private Y _y;

    public Pair(X first, Y second)
    {
        _x = first;
        _y = second;
    }

    public X first { get { return _x; } }

    public Y second { get { return _y; } }

    public override bool Equals(object obj)
    {
        if (obj == null)
            return false;
        if (obj == this)
            return true;
        Pair<X, Y> other = obj as Pair<X, Y>;
        if (other == null)
            return false;

        return
            (((first == null) && (other.first == null))
                || ((first != null) && first.Equals(other.first)))
              &&
            (((second == null) && (other.second == null))
                || ((second != null) && second.Equals(other.second)));
    }

    public override int GetHashCode()
    {
        int hashcode = 0;
        if (first != null)
            hashcode += first.GetHashCode();
        if (second != null)
            hashcode += second.GetHashCode();

        return hashcode;
    }
}

ここにいくつかのテストコードがあります:

[TestClass]
public class PairTest
{
    [TestMethod]
    public void pairTest()
    {
        string s = "abc";
        Pair<int, string> foo = new Pair<int, string>(10, s);
        Pair<int, string> bar = new Pair<int, string>(10, s);
        Pair<int, string> qux = new Pair<int, string>(20, s);
        Pair<int, int> aaa = new Pair<int, int>(10, 20);

        Assert.IsTrue(10 == foo.first);
        Assert.AreEqual(s, foo.second);
        Assert.AreEqual(foo, bar);
        Assert.IsTrue(foo.GetHashCode() == bar.GetHashCode());
        Assert.IsFalse(foo.Equals(qux));
        Assert.IsFalse(foo.Equals(null));
        Assert.IsFalse(foo.Equals(aaa));

        Pair<string, string> s1 = new Pair<string, string>("a", "b");
        Pair<string, string> s2 = new Pair<string, string>(null, "b");
        Pair<string, string> s3 = new Pair<string, string>("a", null);
        Pair<string, string> s4 = new Pair<string, string>(null, null);
        Assert.IsFalse(s1.Equals(s2));
        Assert.IsFalse(s1.Equals(s3));
        Assert.IsFalse(s1.Equals(s4));
        Assert.IsFalse(s2.Equals(s1));
        Assert.IsFalse(s3.Equals(s1));
        Assert.IsFalse(s2.Equals(s3));
        Assert.IsFalse(s4.Equals(s1));
        Assert.IsFalse(s1.Equals(s4));
    }
}

3
IEquatableを実装しないと、ボクシングが発生します。クラスを正しく完了するには、さらに多くの作業が必要です。
ジャック


3

達成したいことに応じて、KeyValuePairを試してみるとよいでしょう。

エントリのキーを変更できないという事実は、もちろん、エントリ全体をKeyValuePairの新しいインスタンスに置き換えるだけで修正できます。



2

さっそくグーグルの後で同じ質問をしていたところ、System.Web.UI以外に.NETにペアクラスがあることがわかりました^〜^(http://msdn.microsoft.com/en-us/library/system.web.ui.pair.aspx)コレクションフレームワークの代わりにそこに配置した理由を知っている


System.Web.UI.Pairについて知っています。しかし、ジェネリッククラスが必要でした。
Alexander Prokofyev、

System.Web.UI.Pairはシールされています。そこから派生することはできません(型セーフアクセサーを追加する場合)。
Martin Vobr

2

.NET 4.0以降、次のSystem.Tuple<T1, T2>クラスがあります。

// pair is implicitly typed local variable (method scope)
var pair = System.Tuple.Create("Current century", 21);


下部には、次のように書かれています。 バージョン情報 NET Frameworkサポート対象:4
Alexander Prokofyev

2
@アレクサンダー:はい、そうです。(なぜ彼らはなぜこのページを.NET 3.5固有にしたのかと思っていましたが)
Serge Mikhailov

2

私は通常Tuple、次のようにクラスを独自の汎用ラッパーに拡張します。

public class Statistic<T> : Tuple<string, T>
{
    public Statistic(string name, T value) : base(name, value) { }
    public string Name { get { return this.Item1; } }
    public T Value { get { return this.Item2; } }
}

そしてそれを次のように使用します:

public class StatSummary{
      public Statistic<double> NetProfit { get; set; }
      public Statistic<int> NumberOfTrades { get; set; }

      public StatSummary(double totalNetProfit, int numberOfTrades)
      {
          this.TotalNetProfit = new Statistic<double>("Total Net Profit", totalNetProfit);
          this.NumberOfTrades = new Statistic<int>("Number of Trades", numberOfTrades);
      }
}

StatSummary summary = new StatSummary(750.50, 30);
Console.WriteLine("Name: " + summary.NetProfit.Name + "    Value: " + summary.NetProfit.Value);
Console.WriteLine("Name: " + summary.NumberOfTrades.Value + "    Value: " + summary.NumberOfTrades.Value);

1

上記を機能させるには(辞書のキーとしてペアが必要でした)。追加する必要がありました:

    public override Boolean Equals(Object o)
    {
        Pair<T, U> that = o as Pair<T, U>;
        if (that == null)
            return false;
        else
            return this.First.Equals(that.First) && this.Second.Equals(that.Second);
    }

それをしたら、私も追加しました

    public override Int32 GetHashCode()
    {
        return First.GetHashCode() ^ Second.GetHashCode();
    }

コンパイラの警告を抑制します。


1
あなたはそれよりも良いハッシュコードアルゴリズムを見つける必要があります、37 + 23 *(h1 + 23 *(h2 + 23 *(h3 + ...)))を使用してみてくださいこれにより、(A、B)が(B、A)と区別されます)、つまり 並べ替えはコードに影響します。
Lasse V. Karlsen、

コメントは受け入れられます。.私の場合、コンパイラの衰退を抑制しようとしていましたが、とにかくTは文字列でUはInt32です...
Andrew Stein


1

カスタムクラスまたは.Net 4.0タプルとは別に、C#7.0以降、ValueTupleと呼ばれる新しい機能があり、これはこの場合に使用できる構造体です。書く代わりに:

Tuple<string, int> t = new Tuple<string, int>("Hello", 4);

およびを介して値にアクセスするt.Item1t.Item2、次のように簡単に実行できます。

(string message, int count) = ("Hello", 4);

あるいは:

(var message, var count) = ("Hello", 4);
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.