Предикаты делегатов в C #

256

Можешь мне объяснить:

  • Что такое предикатный делегат?
  • Где мы должны использовать предикаты?
  • Какие-либо лучшие практики при использовании предикатов?

Описательный исходный код будет оценен.

Canavar
источник

Ответы:

319

Предикат - это функция, которая возвращает trueили false. Делегат предиката является ссылкой на предикат.

Таким образом, в основном делегат предиката является ссылкой на функцию, которая возвращает trueили false. Предикаты очень полезны для фильтрации списка значений - вот пример.

using System;
using System.Collections.Generic;

class Program
{
    static void Main()
    {
        List<int> list = new List<int> { 1, 2, 3 };

        Predicate<int> predicate = new Predicate<int>(greaterThanTwo);

        List<int> newList = list.FindAll(predicate);
    }

    static bool greaterThanTwo(int arg)
    {
        return arg > 2;
    }
}

Теперь, если вы используете C # 3, вы можете использовать лямбду для более четкого представления предиката:

using System;
using System.Collections.Generic;

class Program
{
    static void Main()
    {
        List<int> list = new List<int> { 1, 2, 3 };

        List<int> newList = list.FindAll(i => i > 2);
    }
}
Эндрю Хэйр
источник
@ Эндрю Харе: в вашем первом фрагменте кода это должно быть yeild returnвместо этого? Или как это работает, как оно перебирает весь список?
VoodooChild
5
@VoodooChild: Помните, что предикат будет вызываться для каждого элемента в последовательности по очереди . Так greaterThanTwoесть returnне yield returnтак как это FindAllметод , который обрабатывает последовательность для вас.
Эндрю Харе
1
@AndrewHare, можно ли i > valвместо того i > 2, где valнаходится значение, введенное пользователем.
Mourya
81

Исходя из ответа Эндрю относительно c # 2 и c # 3 ... вы также можете сделать их встроенными для одноразовой функции поиска (см. Ниже).

using System;
using System.Collections.Generic;

class Program
{
    static void Main()
    {
        List<int> list = new List<int> { 1, 2, 3 };

        List<int> newList = list.FindAll(delegate(int arg)
                           {
                               return arg> 2;
                           });
    }
}

Надеюсь это поможет.

WestDiscGolf
источник
11

Просто делегат, который возвращает логическое значение. Он часто используется при фильтрации списков, но может использоваться везде, где вы хотите.

List<DateRangeClass>  myList = new List<DateRangeClass<GetSomeDateRangeArrayToPopulate);
myList.FindAll(x => (x.StartTime <= minDateToReturn && x.EndTime >= maxDateToReturn):
Адам Карр
источник
9

Там хорошая статья о предикатах здесь , хотя это от .NET2 эпохи, так что нет никакого упоминания о лямбда - выражений там.

LukeH
источник
Ссылка в вашем ответе больше не ссылается на актуальную статью
Дэвид Крам
@ Дэвид Крэм: Спасибо, я обновил ссылку, чтобы использовать Wayback Machine, хотя в настоящее время статья выглядит действительно устаревшей.
LukeH
6

Что такое предикатный делегат?

1) Предикат - это функция, которая возвращает истину или ложь. Эта концепция появилась в .net 2.0 framework. 2) Используется с лямбда-выражением (=>). Он принимает универсальный тип в качестве аргумента. 3) Позволяет определить функцию предиката и передать ее в качестве параметра другой функции. 4) Это особый случай Func, в котором он принимает только один параметр и всегда возвращает логическое значение.

В пространстве имен C #:

namespace System
{   
    public delegate bool Predicate<in T>(T obj);
}

Он определяется в пространстве имен System.

Где мы должны использовать предикатный делегат?

Мы должны использовать Predicate Delegate в следующих случаях:

1) Для поиска предметов в общей коллекции. например

var employeeDetails = employees.Where(o=>o.employeeId == 1237).FirstOrDefault();

2) Базовый пример, который сокращает код и возвращает true или false:

Predicate<int> isValueOne = x => x == 1;

Теперь вызов выше предиката:

Console.WriteLine(isValueOne.Invoke(1)); // -- returns true.

3) Анонимный метод также может быть назначен типу делегата Predicate, как показано ниже:

Predicate<string> isUpper = delegate(string s) { return s.Equals(s.ToUpper());};
    bool result = isUpper("Hello Chap!!");

Какие-нибудь лучшие практики о предикатах?

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

Гюль Мд Эршад
источник
5

Методы поиска на основе предикатов позволяют делегату метода или лямбда-выражению решать, является ли данный элемент «совпадающим». Предикат - это просто делегат, принимающий объект и возвращающий истину или ложь: открытый делегат bool Predicate (T object);

   static void Main()
        {
            string[] names = { "Lukasz", "Darek", "Milosz" };
            string match1 = Array.Find(names, delegate(string name) { return name.Contains("L"); });
            //or
            string match2 = Array.Find(names, delegate(string name) { return name.Contains("L"); });
            //or
            string match3 = Array.Find(names, x => x.Contains("L"));


            Console.WriteLine(match1 + " " + match2 + " " + match3);     // Lukasz Lukasz Lukasz
        }
        static bool ContainsL(string name) { return name.Contains("L"); }
lukaszk
источник
2

Если вы в VB 9 (VS2008), предикат может быть сложной функцией:

Dim list As New List(Of Integer)(New Integer() {1, 2, 3})
Dim newList = list.FindAll(AddressOf GreaterThanTwo)
...
Function GreaterThanTwo(ByVal item As Integer) As Boolean
    'do some work'
    Return item > 2
End Function

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

Dim list As New List(Of Integer)(New Integer() {1, 2, 3})
Dim newList = list.FindAll(Function(item) item > 2)
danlash
источник
0

Предикат подпадает под категорию общих делегатов в C #. Это вызывается с одним аргументом и всегда возвращает логический тип. По сути, предикат используется для проверки условия - истина / ложь. Многие классы поддерживают предикат в качестве аргумента. Например, list.findall ожидает предикат параметра. Вот пример предиката.

Представьте себе указатель на функцию с подписью -

bool делегат myDelegate (T match);

Вот пример

Node.cs

namespace PredicateExample
{
    class Node
    {
        public string Ip_Address { get; set; }
        public string Node_Name { get; set; }
        public uint Node_Area { get; set; }
    }
}

Основной класс -

using System;
using System.Threading;
using System.Collections.Generic;

namespace PredicateExample
{
    class Program
    {
        static void Main(string[] args)
        {
            Predicate<Node> backboneArea = Node =>  Node.Node_Area == 0 ;
            List<Node> Nodes = new List<Node>();
            Nodes.Add(new Node { Ip_Address = "1.1.1.1", Node_Area = 0, Node_Name = "Node1" });
            Nodes.Add(new Node { Ip_Address = "2.2.2.2", Node_Area = 1, Node_Name = "Node2" });
            Nodes.Add(new Node { Ip_Address = "3.3.3.3", Node_Area = 2, Node_Name = "Node3" });
            Nodes.Add(new Node { Ip_Address = "4.4.4.4", Node_Area = 0, Node_Name = "Node4" });
            Nodes.Add(new Node { Ip_Address = "5.5.5.5", Node_Area = 1, Node_Name = "Node5" });
            Nodes.Add(new Node { Ip_Address = "6.6.6.6", Node_Area = 0, Node_Name = "Node6" });
            Nodes.Add(new Node { Ip_Address = "7.7.7.7", Node_Area = 2, Node_Name = "Node7" });

            foreach( var item in Nodes.FindAll(backboneArea))
            {
                Console.WriteLine("Node Name " + item.Node_Name + " Node IP Address " + item.Ip_Address);
            }

            Console.ReadLine();
        }
    }
}
ловкий
источник
0

Просто -> они предоставляют Истинные / Ложные значения, основанные на условии, в основном используемом для запросов. в основном используется с делегатами

рассмотрим пример списка

List<Program> blabla= new List<Program>();
        blabla.Add(new Program("shubham", 1));
        blabla.Add(new Program("google", 3));
        blabla.Add(new Program("world",5));
        blabla.Add(new Program("hello", 5));
        blabla.Add(new Program("bye", 2));

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

    Predicate<Program> test = delegate (Program p) { return p.age > 3; };
        List<Program> matches = blabla.FindAll(test);
        Action<Program> print = Console.WriteLine;
        matches.ForEach(print);

пытался сохранить это простым!

Шубхам Харе
источник
-4

Делегат определяет ссылочный тип, который можно использовать для инкапсуляции метода с определенной сигнатурой. Жизненный цикл делегата C #: жизненный цикл делегата C #

  • декларация
  • Конкретизация
  • В ОТПУСКЕ

узнать больше формы http://asp-net-by-parijat.blogspot.in/2015/08/what-is-delegates-in-c-how-to-declare.html

парижат мишра
источник