什么是C ++ std :: pair的C#类似物?


284

我感兴趣:C#std::pair在C ++中的模拟是什么?我找到了System.Web.UI.Pair课程,但是我更喜欢基于模板的东西。

谢谢!


11
不久前,我有同样的请求,但是我想得更多,您可能只想滚动自己的配对类,使用显式的类类型和字段,而不是通用的“ First”和“ Second”即可。它使您的代码更具可读性。配对类最多可以包含4行,因此,通过重用通用的Pair <T,U>类,您不会节省太多,而且代码更易读。
Mark Lakata 2012年

Answers:


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方法
安德鲁·斯坦

Tuple <>现在是一个更好的解决方案。
dkantowitz

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.UIPair之所以包含该类,是因为该类在ASP.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;
skybluecodeflier11年

2
元组正是我想要的。谢谢。
gligoran 2011年

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,您将得到拳击。要正确完成课程,还有更多工作要做。
2012年

8

如果是关于字典之类的内容,那么您正在寻找System.Collections.Generic.KeyValuePair <TKey,TValue>。


3

根据您要完成的工作,您可能需要尝试KeyValuePair

您不能更改条目键的事实当然可以通过简单地将整个条目替换为KeyValuePair的新实例来纠正。




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,您可以轻松查看Tuple上的.NET 3.5文档
Serge Mikhailov

他们在底部说:版本信息 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);
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.