C # analog ของ C ++ std :: pair คืออะไร?


284

ฉันสนใจ: C # ของอะนาล็อกstd::pairใน C ++ คืออะไร? ฉันพบSystem.Web.UI.Pairชั้นเรียน แต่ฉันต้องการสิ่งที่ใช้เทมเพลต

ขอบคุณ!


11
ฉันมีคำขอเดียวกันเมื่อไม่นานมานี้ แต่ยิ่งฉันคิดถึงมันมากขึ้นคุณอาจต้องการเพิ่มระดับการจับคู่ของคุณเองด้วยประเภทและเขตข้อมูลที่ชัดเจนแทนที่จะเป็น "First" และ "Second" ทั่วไป ทำให้โค้ดของคุณอ่านง่ายขึ้น คลาสการจับคู่อาจมีขนาดเพียง 4 บรรทัดดังนั้นคุณจึงไม่ประหยัดมากนักโดยการใช้คลาส Pair <T, U> ซ้ำอีกครั้งและรหัสของคุณจะสามารถอ่านได้มากขึ้น
Mark Lakata

คำตอบ:


325

Tuples พร้อมใช้งานตั้งแต่. 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

เห็นโพสต์ของฉันเกี่ยวกับการเพิ่มเท่ากับวิธีการ
แอนดรูสไตน์

Tuple <> เป็นทางออกที่ดีกว่า
dkantowitz

6
ตั้งแต่พารามิเตอร์ชนิดที่อยู่ในชั้นทั่วไปไม่สามารถสรุปในการแสดงออกสร้างวัตถุ (โทรคอนสตรัค), ผู้เขียนของ BCL Tupleทำระดับผู้ช่วยที่ไม่ทั่วไปเรียกว่า ดังนั้นคุณสามารถพูดได้ว่าTuple.Create("Hello", 4)ง่ายกว่าnew Tuple<string, int>("Hello", 4)ไหน (โดยวิธี. NET4.0 ได้มาที่นี่ตั้งแต่ปี 2010)
Jeppe Stig Nielsen

4
ใจคุณTuple<>ดำเนินการที่มั่นคงEqualsและGetHashCodeมีความหมายที่คุ้มค่าซึ่งเป็นสิ่งที่ดี โปรดทราบเมื่อใช้สิ่งอันดับของคุณเอง
nawfal

เห็นได้ชัดว่ามันแตกเพราะ Equals และ GetHashCode
julx

90

System.Web.UIมีPairคลาสเนื่องจากมีการใช้งานอย่างหนักใน ASP.NET 1.1 เป็นโครงสร้าง ViewState ภายใน

อัปเดต ส.ค. 2560: C # 7.0 / .NET Framework 4.7 จัดเตรียมไวยากรณ์เพื่อประกาศ Tuple ด้วยรายการที่ระบุชื่อโดยใช้System.ValueTuplestruct

//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สำหรับตัวอย่างไวยากรณ์เพิ่มเติม

อัปเดตมิถุนายน 2555: Tuplesเป็นส่วนหนึ่งของ .NET ตั้งแต่รุ่น 4.0

นี่คือบทความก่อนหน้านี้ที่อธิบายถึงการรวม in.NET4.0และการสนับสนุนสำหรับข้อมูลทั่วไป:

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

2
โปรดทราบว่าสิ่งอันดับเป็นแบบอ่านอย่างเดียว นั่นคือคุณไม่สามารถทำได้:tuple.Item1 = 4;
skybluecodeflier

2
สิ่งอันดับคือสิ่งที่ฉันกำลังมองหา ขอบคุณ
gligoran

38

น่าเสียดายที่ไม่มี คุณสามารถใช้System.Collections.Generic.KeyValuePair<K, V>ในหลาย ๆ สถานการณ์

หรือคุณสามารถใช้ประเภทที่ไม่ระบุชื่อเพื่อจัดการสิ่งอันดับอย่างน้อยในเครื่อง:

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

ทางเลือกสุดท้ายคือการสร้างชั้นเรียนของตัวเอง


2
เพียงเพื่อจะชัดเจนชนิดที่ไม่ระบุชื่อยังอ่านอย่างเดียว - MSDN
bsegraves


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 คุณจะได้รับการชกมวย มีงานอีกมากที่ต้องทำเพื่อให้ชั้นเรียนของคุณถูกต้อง
แจ็ค

8

หากเป็นเรื่องเกี่ยวกับพจนานุกรมและสิ่งที่คล้ายกันคุณกำลังมองหา System.Collections.Generic.KeyValuePair <TKey, TValue>


3

ทั้งนี้ขึ้นอยู่กับสิ่งที่คุณต้องการที่จะบรรลุคุณอาจต้องการที่จะลองKeyValuePair

ความจริงที่ว่าคุณไม่สามารถเปลี่ยนคีย์ของรายการสามารถแก้ไขได้โดยเพียงแค่แทนที่ทั้งรายการด้วยอินสแตนซ์ใหม่ของ KeyValuePair



2

ฉันถามคำถามเดียวกันหลังจาก google รวดเร็วฉันพบว่ามีคู่คลาสใน. NET ยกเว้นใน System.Web.UI ^ ~ ^ (http://msdn.microsoft.com/en-us/library/system.web.ui.pair.aspx ) ความดีรู้ว่าทำไมพวกเขาถึงวางไว้ที่นั่นแทนที่จะเป็นเฟรมเวิร์กคอลเล็กชัน


ฉันรู้เกี่ยวกับ System.Web.UI.Pair ต้องการเรียนทั่วไป
Alexander Prokofyev

System.Web.UI.Pair ถูกปิดผนึก คุณไม่สามารถได้มาจากมัน (ในกรณีที่คุณต้องการเพิ่ม accessors ที่ปลอดภัยประเภท)
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);

@Alexander คุณสามารถดู. NET 3.5 เอกสารบน Tuple
Serge Mikhailov

ที่ด้านล่างพวกเขาพูดว่า: Version Information NET Frameworkรองรับใน: 4
Alexander Prokofyev

2
@Alexander: ตกลงถูกต้อง (แม้ว่ามันจะทำให้ฉันสงสัยว่าทำไมพวกเขาถึงสร้างหน้านี้เฉพาะ. 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 ...
แอนดรูสไตน์


1

นอกเหนือจากคลาสที่กำหนดเองหรือ. Net 4.0 Tuples เนื่องจาก C # 7.0 มีคุณลักษณะใหม่ที่เรียกว่า ValueTuple ซึ่งเป็นโครงสร้างที่สามารถใช้ในกรณีนี้ แทนที่จะเขียน:

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

และเข้าถึงคุณค่าผ่านทางt.Item1และt.Item2คุณสามารถทำได้เช่นนั้น:

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

หรือแม้กระทั่ง:

(var message, var count) = ("Hello", 4);
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.