ภาคแสดงผู้ได้รับมอบหมายใน C #


256

คุณช่วยอธิบายให้ฉัน:

  • ผู้แทนภาครัฐคืออะไร
  • เราควรใช้เพรดิเคตที่ไหน
  • มีแนวทางปฏิบัติที่ดีที่สุดเมื่อใช้เพรดิเคตหรือไม่

รหัสที่มาอธิบายจะได้รับการชื่นชม

คำตอบ:


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

@Andrew Hare: ในข้อมูลโค้ดแรกของคุณควรจะเป็นyeild returnเช่นนั้นหรือ หรือวิธีการทำงานอย่างไรมันซ้ำไปทั่วทั้งรายการ?
VoodooChild

5
@VoodooChild: โปรดจำไว้ว่าคำกริยาจะถูกเรียกว่าองค์ประกอบในแต่ละลำดับในทางกลับกัน ดังนั้นgreaterThanTwoมีreturnไม่ได้yield returnเพราะมันเป็นFindAllวิธีการที่มีการจัดการลำดับสำหรับคุณ
Andrew Hare

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

หวังว่านี่จะช่วยได้


11

เพียงแค่ตัวแทนที่ส่งกลับบูลีน มันถูกใช้อย่างมากในการกรองรายการ แต่สามารถใช้ได้ทุกที่ที่คุณต้องการ

List<DateRangeClass>  myList = new List<DateRangeClass<GetSomeDateRangeArrayToPopulate);
myList.FindAll(x => (x.StartTime <= minDateToReturn && x.EndTime >= maxDateToReturn):

9

มีบทความที่ดีเกี่ยวกับเพรดิเคตที่นี่ถึงแม้ว่ามันจะมาจากยุค. NET2 ดังนั้นจึงไม่มีการเอ่ยถึงการแสดงออกแลมบ์ดาในนั้น


ลิงก์ในคำตอบของคุณจะไม่ลิงก์ไปยังบทความจริงอีกต่อไป
David Cram

@ David อัด: ขอบคุณผมได้ปรับปรุงการเชื่อมโยงไปใช้เครื่อง Wayback แม้ว่าบทความที่กำลังมองหาจริงๆลงวันที่ในปัจจุบัน
ลุค

6

ผู้แทนจากภาครัฐคืออะไร

1) เพรดิเคตเป็นคุณลักษณะที่ส่งคืนจริงหรือเท็จแนวคิดนี้มาในกรอบงาน. net 2.0 2) มันถูกใช้กับการแสดงออกแลมบ์ดา (=>) มันใช้ประเภททั่วไปเป็นอาร์กิวเมนต์ 3) อนุญาตให้ฟังก์ชันเพรดิเคตถูกกำหนดและส่งผ่านเป็นพารามิเตอร์ไปยังฟังก์ชันอื่น 4) มันเป็นกรณีพิเศษของFunc, ในการที่จะใช้เพียงพารามิเตอร์เดียวและส่งกลับค่าบูล

ใน C # namespace:

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

มันถูกกำหนดใน System namespace

เราควรใช้ผู้รับมอบสิทธิ์ภาคไหน

เราควรใช้ Predicate Delegate ในกรณีต่อไปนี้:

1) สำหรับการค้นหารายการในคอลเลกชันทั่วไป เช่น

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

2) ตัวอย่างพื้นฐานที่ย่อโค้ดและส่งคืนค่าจริงหรือเท็จ

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

ตอนนี้โทรไปที่ภาคแสดง:

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

3) วิธีการที่ไม่ระบุชื่อสามารถกำหนดให้กับประเภทผู้แทนภาคได้ดังนี้:

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

มีแนวทางปฏิบัติที่ดีที่สุดเกี่ยวกับเพรดิเคตหรือไม่

ใช้ Func, Lambda Expressions และ Delegates แทน Predicates


5

วิธีการค้นหาตามภาคอนุญาตให้ผู้มอบหมายวิธีหรือการแสดงออกแลมบ์ดาในการตัดสินใจว่าองค์ประกอบที่กำหนดเป็น“ การแข่งขัน” เพรดิเคตเป็นเพียงผู้รับมอบสิทธิ์ยอมรับวัตถุและส่งคืนจริงหรือเท็จ: สาธารณะมอบหมาย bool เพรดิเคต (วัตถุ T);

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

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)

0

ภาคแสดงอยู่ในหมวดหมู่ของผู้ได้รับมอบหมายทั่วไปใน C # สิ่งนี้เรียกว่าหนึ่งอาร์กิวเมนต์และส่งคืนชนิดบูลีนเสมอ โดยทั่วไปเพรดิเคตจะใช้ในการทดสอบเงื่อนไข - จริง / เท็จ คลาสจำนวนมากสนับสนุนเพรดิเคตเป็นอาร์กิวเมนต์ ยกตัวอย่างเช่น list.findall คาดว่าพารามิเตอร์กริยา นี่คือตัวอย่างของเพรดิเคต

ลองนึกภาพตัวชี้ฟังก์ชั่นที่มีลายเซ็น -

บูลมอบสิทธิ์ myDelegate (การแข่งขัน T);

นี่คือตัวอย่าง

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 # คือ

  • การประกาศ
  • instantiation
  • INVACATION

เรียนรู้รูปแบบเพิ่มเติม http://asp-net-by-parijat.blogspot.in/2015/08/what-is-delegates-in-c-how-to-declare.html

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.