Вопрос

Мне интересно: что такое аналог C # std :: pair в C ++? Я нашел класс System.Web.UI.Pair , но предпочел бы что-нибудь на основе шаблонов.

Спасибо!

Это было полезно?

Решение

Кортежи доступны с .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

Другие советы

System.Web.UI содержал класс Pair , поскольку он широко использовался в ASP.NET 1.1 в качестве внутренней структуры ViewState.

Обновление от августа 2017 г .: C # 7.0 / .NET Framework 4.7 предоставляет синтаксис для объявления кортежа с именованными элементами с помощью System.ValueTuple struct.

//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 года: Кортежи входят в состав .NET начиная с версии 4.0.

Вот более ранняя статья, описывающая включение в .NET4.0 и поддержка дженериков:

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

К сожалению, нет ни одного. Вы можете использовать System.Collections.Generic.KeyValuePair < K, V > во многих ситуациях.

Кроме того, вы можете использовать анонимные типы для обработки кортежей, по крайней мере локально:

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

Последний вариант - создать собственный класс.

C # имеет кортежи начиная с версии 4.0.

Некоторые ответы кажутся просто неправильными,

<Ол>
  • вы не можете использовать словарь, как бы хранить пары (a, b) и (a, c). Понятие пар не следует путать с ассоциативным поиском ключа и ценностей.
  • большая часть приведенного выше кода кажется подозрительной
  • Вот мой класс пары

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

    Если речь идет о словарях и т. п., вы ищете System.Collections.Generic.KeyValuePair < TKey, TValue >.

    В зависимости от того, чего вы хотите достичь, вы можете попробовать KeyValuePair .

    Тот факт, что вы не можете изменить ключ записи, можно, конечно, исправить, просто заменив всю запись новым экземпляром KeyValuePair.

    Я создал реализацию Tuples на C #, которая решает проблему в общем случае для двух или пяти значений - вот пост блога , который содержит ссылку на источник.

    Я задал тот же вопрос только сейчас, после того как я быстро нашел Google, и обнаружил, что в .NET существует класс пар, кроме его в System.Web.UI ^ ~ ^ ( http://msdn.microsoft.com/en-us/library/system.web.ui.pair.aspx ) Боже мой знает, почему они положили его туда, а не рамки коллекций

    Начиная с .NET 4.0 у вас есть класс System.Tuple < T1, T2 > :

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

    Обычно я расширяю класс 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);
    

    Для того, чтобы вышеперечисленное сработало (мне нужна была пара в качестве ключа словаря). Я должен был добавить:

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

    для подавления предупреждения компилятора.

    Библиотека PowerCollections (ранее доступная в Wintellect, но теперь размещенная на Codeplex @ http://powercollections.codeplex.com ) имеет общую структуру Pair.

    Помимо пользовательского класса или кортежей .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);
    
    Лицензировано под: CC-BY-SA с атрибуция
    Не связан с StackOverflow
    scroll top