---------- 20190805 Predicate Delegate Represents the method that defines a set of criteria and determines whether the specified object meets those criteria. The following code example uses a Predicate delegate with the Array.Find method to search an array of Point structures. The example explicitly defines a Predicate delegate named predicate and assigns it a method named FindPoints that returns true if the product of the Point.X and Point.Y fields is greater than 100,000. Note that it is customary to use a lambda expression rather than to explicitly define a delegate of type Predicate, as the second example illustrates. 第1個範例明確宣告 Predicate 為 FindPoints. 第2個範例, 以 lambda expression 隱含省略了 Predicate 為 FindPoints. using System; using System.Drawing; public class Example { public static void Main() { // Create an array of Point structures. Point[] points = { new Point(100, 200), new Point(150, 250), new Point(250, 375), new Point(275, 395), new Point(295, 450) }; // Define the Predicate delegate. 明確宣告 Predicate 為 FindPoints Predicate predicate = FindPoints; // Find the first Point structure for which X times Y // is greater than 100000. Point first = Array.Find(points, predicate); // Display the first structure found. Console.WriteLine("Found: X = {0}, Y = {1}", first.X, first.Y); } private static bool FindPoints(Point obj) { return obj.X * obj.Y > 100000; } } // The example displays the following output: // Found: X = 275, Y = 395 The following example is identical to the previous example, except that it uses a lambda expression to represent the Predicate delegate. Each element of the points array is passed to the lambda expression until the expression finds an element that meets the search criteria. In this case, the lambda expression returns true if the product of the X and Y fields is greater than 100,000. using System; using System.Drawing; public class Example { public static void Main() { // Create an array of Point structures. Point[] points = { new Point(100, 200), new Point(150, 250), new Point(250, 375), new Point(275, 395), new Point(295, 450) }; // Find the first Point structure for which X times Y // is greater than 100000. Point first = Array.Find(points, x => x.X * x.Y > 100000 ); // Display the first structure found. Console.WriteLine("Found: X = {0}, Y = {1}", first.X, first.Y); } } // The example displays the following output: // Found: X = 275, Y = 395 Remarks This delegate is used by several methods of the Array and List classes to search for elements in the collection. Typically, the Predicate delegate is represented by a lambda expression. Because locally scoped variables are available to the lambda expression, it is easy to test for a condition that is not precisely known at compile time. This is simulated in the following example, which defines a HockeyTeam class that contains information about a National Hockey League team and the year in which it was founded. The example defines an array of integer values that represent years, and randomly assigns one element of the array to foundedBeforeYear, which is a variable that is locally scoped to the example's Main method. Because locally scoped variables are available to a lambda expression, the lambda expression passed to the List.FindAll method is able to return a HockeyTeam object for each team founded on or before that year. using System; using System.Collections.Generic; public class HockeyTeam { private string _name; private int _founded; public HockeyTeam(string name, int year) { _name = name; _founded = year; } public string Name { get { return _name; } } public int Founded { get { return _founded; } } } public class Example { public static void Main() { Random rnd = new Random(); List teams = new List(); teams.AddRange( new HockeyTeam[] { new HockeyTeam("Detroit Red Wings", 1926), new HockeyTeam("Chicago Blackhawks", 1926), new HockeyTeam("San Jose Sharks", 1991), new HockeyTeam("Montreal Canadiens", 1909), new HockeyTeam("St. Louis Blues", 1967) } ); int[] years = { 1920, 1930, 1980, 2000 }; int foundedBeforeYear = years[rnd.Next(0, years.Length)]; Console.WriteLine("Teams founded before {0}:", foundedBeforeYear); foreach (var team in teams.FindAll( x => x.Founded <= foundedBeforeYear)) Console.WriteLine("{0}: {1}", team.Name, team.Founded); } } // The example displays output similar to the following: // Teams founded before 1930: // Detroit Red Wings: 1926 // Chicago Blackhawks: 1926 // Montreal Canadiens: 1909 ---------- The following example demonstrates the RemoveAll method and several other methods that use the Predicate generic delegate. A List of strings is created, containing 8 dinosaur names, two of which (at positions 1 and 5) end with "saurus". The example also defines a search predicate method named EndsWithSaurus, which accepts a string parameter and returns a Boolean value indicating whether the input string ends in "saurus". The Find, FindLast, and FindAll methods are used to search the list with the search predicate method. The RemoveAll method is used to remove all entries ending with "saurus". It traverses the list from the beginning, passing each element in turn to the EndsWithSaurus method. The element is removed if the EndsWithSaurus method returns true. Note In C# and Visual Basic, it is not necessary to create the Predicate delegate (Predicate(Of String) in Visual Basic) explicitly. These languages infer the correct delegate from context, and create it automatically. Finally, the Exists method verifies that there are no strings in the list that end with "saurus". using System; using System.Collections.Generic; public class Example { public static void Main() { List dinosaurs = new List(); dinosaurs.Add("Compsognathus"); dinosaurs.Add("Amargasaurus"); dinosaurs.Add("Oviraptor"); dinosaurs.Add("Velociraptor"); dinosaurs.Add("Deinonychus"); dinosaurs.Add("Dilophosaurus"); dinosaurs.Add("Gallimimus"); dinosaurs.Add("Triceratops"); Console.WriteLine(); foreach(string dinosaur in dinosaurs) { Console.WriteLine(dinosaur); } Console.WriteLine("\nTrueForAll(EndsWithSaurus): {0}", dinosaurs.TrueForAll(EndsWithSaurus)); Console.WriteLine("\nFind(EndsWithSaurus): {0}", dinosaurs.Find(EndsWithSaurus)); Console.WriteLine("\nFindLast(EndsWithSaurus): {0}", dinosaurs.FindLast(EndsWithSaurus)); Console.WriteLine("\nFindAll(EndsWithSaurus):"); List sublist = dinosaurs.FindAll(EndsWithSaurus); foreach(string dinosaur in sublist) { Console.WriteLine(dinosaur); } Console.WriteLine( "\n{0} elements removed by RemoveAll(EndsWithSaurus).", dinosaurs.RemoveAll(EndsWithSaurus)); Console.WriteLine("\nList now contains:"); foreach(string dinosaur in dinosaurs) { Console.WriteLine(dinosaur); } Console.WriteLine("\nExists(EndsWithSaurus): {0}", dinosaurs.Exists(EndsWithSaurus)); } // Search predicate returns true if a string ends in "saurus". private static bool EndsWithSaurus(String s) { return s.ToLower().EndsWith("saurus"); } } /* This code example produces the following output: Compsognathus Amargasaurus Oviraptor Velociraptor Deinonychus Dilophosaurus Gallimimus Triceratops TrueForAll(EndsWithSaurus): False Find(EndsWithSaurus): Amargasaurus FindLast(EndsWithSaurus): Dilophosaurus FindAll(EndsWithSaurus): Amargasaurus Dilophosaurus 2 elements removed by RemoveAll(EndsWithSaurus). List now contains: Compsognathus Oviraptor Velociraptor Deinonychus Gallimimus Triceratops Exists(EndsWithSaurus): False */ Remarks The Predicate is a delegate to a method that returns true if the object passed to it matches the conditions defined in the delegate. The elements of the current List are individually passed to the Predicate delegate, and the elements that match the conditions are removed from the List. This method performs a linear search; therefore, this method is an O(n) operation, where n is Count.