Как мне сформировать хороший делегат-предикат для Find() чего-то в моем List<T>?

StackOverflow https://stackoverflow.com/questions/242012

Вопрос

Посмотрев MSDN, мне до сих пор неясно, как мне сформировать правильный предикат для использования метода Find() в списке, используя переменную-член T (где T - класс).

Например:

public class Car
{
   public string Make;
   public string Model;
   public int Year;
}

{  // somewhere in my code
   List<Car> carList = new List<Car>();
   // ... code to add Cars ...

   Car myCar = new Car();

   // Find the first of each car made between 1980 and 2000
   for (int x = 1980; x < 2000; x++)
   {
       myCar = carList.Find(byYear(x));
       Console.Writeline(myCar.Make + myCar.Model);
   }
}

Как должен выглядеть мой предикат «byYear»?

(Пример MSDN говорит только о списке динозавров и ищет только неизменное значение «saurus» - он не показывает, как передать значение в предикат...)

РЕДАКТИРОВАТЬ:Я использую VS2005/.NET2.0, поэтому не думаю, что мне доступна лямбда-нотация...

РЕДАКТИРОВАТЬ2:В примере удалено «1999», поскольку мне может понадобиться «Найти» программно на основе разных значений.Пример изменен на диапазон автомобилей с 1980 по 2000 год с использованием цикла for-do.

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

Решение

Хорошо, в .NET 2.0 вы можете использовать делегатов, вот так:

static Predicate<Car> ByYear(int year)
{
    return delegate(Car car)
    {
        return car.Year == year;
    };
}

static void Main(string[] args)
{
    // yeah, this bit is C# 3.0, but ignore it - it's just setting up the list.
    List<Car> list = new List<Car>
    {
        new Car { Year = 1940 },
        new Car { Year = 1965 },
        new Car { Year = 1973 },
        new Car { Year = 1999 }
    };
    var car99 = list.Find(ByYear(1999));
    var car65 = list.Find(ByYear(1965));

    Console.WriteLine(car99.Year);
    Console.WriteLine(car65.Year);
}

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

Вы можете использовать лямбда-выражение следующим образом:

myCar = carList.Find(car => car.Year == 1999);

Или вы можете использовать анонимного делегата:

Car myCar = cars.Find(delegate(Car c) { return c.Year == x; });

// If not found myCar will be null
if (myCar != null)
{
     Console.Writeline(myCar.Make + myCar.Model);
}

Поскольку вы не можете использовать лямбду, вы можете просто заменить ее анонимным делегатом.

myCar = carList.Find(delegate(Car car) { return car.Year == i; });

Хм.Если подумать, вы можете использовать каррирование для возврата предиката.

Func<int, Predicate<Car>> byYear = i => (c => c.Year == i);

Теперь вы можете передать результат этой функции (который является предикатом) в метод Find:

my99Car = cars.Find(byYear(1999));
my65Car = cars.Find(byYear(1965));

Вы тоже можете использовать это:

var existData =
    cars.Find(
     c => c.Year== 1999);
Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top