Использование ключевого слова C # params в конструкторе универсальных типов - PullRequest
8 голосов
/ 17 октября 2008

У меня есть универсальный класс в C # с 2 конструкторами:

public Houses(params T[] InitialiseElements)
{}
public Houses(int Num, T DefaultValue)
{}

Построение объекта с использованием int в качестве универсального типа и передача двух целочисленных значений в качестве аргументов приводит к вызову «неправильного» конструктора (с моей точки зрения).

например. Houses<int> houses = new Houses<int>(1,2) - звонит 2-му строителю. Передача любого другого числа int в конструктор вызовет 1-й конструктор.

Есть ли способ обойти это, кроме удаления ключевого слова params и принуждения пользователей передавать массив T при использовании первого конструктора?

Ответы [ 4 ]

15 голосов
/ 17 октября 2008

Более ясным решением было бы использование двух статических заводских методов. Если вы поместите их в неуниверсальный класс, вы также можете воспользоваться выводом типа:

public static class Houses
{
    public static Houses<T> CreateFromElements<T>(params T[] initialElements)
    {
        return new Houses<T>(initialElements);
    }

    public Houses<T> CreateFromDefault<T>(int count, T defaultValue)
    {
        return new Houses<T>(count, defaultValue);
    }
}

Пример звонка:

Houses<string> x = Houses.CreateFromDefault(10, "hi");
Houses<int> y = Houses.CreateFromElements(20, 30, 40);

Тогда конструктору вашего универсального типа не понадобится бит "params", и путаницы не будет.

2 голосов
/ 18 октября 2008

Учитывая следующее, поскольку в оригинале не было слишком много информации о классе и т. Д.

Компилятор решит, что новый House (1,2) точно соответствует второму конструктору, и используйте его, обратите внимание, что я получил ответ с наибольшим количеством голосов, и он не сработал.

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace GenericTest
{
    public class House<T>
    {
        public House(params T[] values)
        {
            System.Console.WriteLine("Params T[]");
        }
        public House(int num, T defaultVal)
        {
            System.Console.WriteLine("int, T");
        }

        public static House<T> CreateFromDefault<T>(int count, T defaultVal)
        {
            return new House<T>(count, defaultVal);
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            House<int> test = new House<int>(1, 2);                         // prints int, t
            House<int> test1 = new House<int>(new int[] {1, 2});            // prints parms
            House<string> test2 = new House<string>(1, "string");           // print int, t
            House<string> test3 = new House<string>("string", "string");    // print parms
            House<int> test4 = House<int>.CreateFromDefault<int>(1, 2);     // print int, t
        }
    }
}
2 голосов
/ 17 октября 2008

2-й конструктор - это более точное точное совпадение, которое является критерием, используемым для оценки правильности конструктора.

2 голосов
/ 17 октября 2008

Возможно, вместо Params вы могли бы передать в IEnumerable

public Houses(IEnumerable<T> InitialiseElements){}
...