什么是 C++ std::pair 的 C# 模拟?

问题描述 投票:0回答:14

C# 中 C++ 中

std::pair
的类似物是什么?我找到了
System.Web.UI.Pair
类,但我更喜欢基于模板的东西。

c# .net data-structures std-pair base-class-library
14个回答
383
投票

元组自.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

98
投票

System.Web.UI
包含
Pair
类,因为它在 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);

39
投票

不幸的是,没有。您可以在很多情况下使用

System.Collections.Generic.KeyValuePair<K, V>

或者,您可以使用匿名类型来处理元组,至少在本地:

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

最后一个选择是创建一个自己的类。


21
投票

C# 从 4.0 版本开始具有 元组


10
投票

有些答案似乎是错误的,

  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));
    }
}

9
投票

除了自定义类或 .Net 4.0 元组之外,从 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);

7
投票

如果与字典等相关,您正在寻找 System.Collections.Generic.KeyValuePair


3
投票

我创建了元组的 C# 实现,它一般解决了 2 到 5 个值之间的问题 - 这是博客文章,其中包含源链接。


3
投票

我通常将

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);

2
投票

根据您想要实现的目标,您可能想尝试KeyValuePair

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


2
投票

我刚刚在快速谷歌后问了同样的问题,我发现 .NET 中有一个pair类,除了它在 System.Web.UI ^ ~ ^ (http://msdn.microsoft.com/ en-us/library/system.web.ui.pair.aspx) 天知道为什么他们把它放在那里而不是集合框架


2
投票

自 .NET 4.0 起,您拥有

System.Tuple<T1, T2>
类:

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

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
投票

PowerCollections 库(以前可从 Wintellect 获得,但现在托管在 Codeplex @ http://powercollections.codeplex.com)具有通用的 Pair 结构。

© www.soinside.com 2019 - 2024. All rights reserved.