説明してもらえますか:
- 述語デリゲートとは
- 述語はどこで使用する必要がありますか?
- 述語を使用する際のベスト プラクティスはありますか?
説明的なソースコードをいただければ幸いです。
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);
}
}
c#2 および c#3 に関する Andrew の回答から先に進むと、1 回限りの検索機能のためにインラインで実行することもできます (以下を参照)。
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;
});
}
}
お役に立てれば。
ブール値を返す単なるデリゲート。リストのフィルタリングでよく使用されますが、好きな場所で使用できます。
List<DateRangeClass> myList = new List<DateRangeClass<GetSomeDateRangeArrayToPopulate);
myList.FindAll(x => (x.StartTime <= minDateToReturn && x.EndTime >= maxDateToReturn):
.NET2 時代のものですが、述語に関する良い記事がここにあります。そのため、そこにはラムダ式についての言及はありません。
述語デリゲートとは
1) 述語は、true または false を返す機能です。この概念は、.net 2.0 フレームワークに組み込まれています。2) ラムダ式 (=>) で使用されています。ジェネリック型を引数として取ります。3) 述語関数を定義して、パラメーターとして別の関数に渡すことができます。4) これは a の特殊なケースでありFunc
、1 つのパラメーターのみを取り、常に bool を返します。
C# 名前空間:
namespace System
{
public delegate bool Predicate<in T>(T obj);
}
これは System 名前空間で定義されています。
Predicate Delegate はどこで使用する必要がありますか?
次の場合に 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、ラムダ式、およびデリゲートを使用します。
述語ベースの検索メソッドを使用すると、メソッド デリゲートまたはラムダ式で、特定の要素が「一致」するかどうかを判断できます。述語は、オブジェクトを受け取り、true または false を返す単純なデリゲートです。 public delegate 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"); }
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
または、式が 1 つだけである限り、述語をラムダとして記述できます。
Dim list As New List(Of Integer)(New Integer() {1, 2, 3})
Dim newList = list.FindAll(Function(item) item > 2)
述語は、C# のジェネリック デリゲートのカテゴリに分類されます。これは 1 つの引数で呼び出され、常にブール型を返します。基本的に、述語は条件 (真/偽) をテストするために使用されます。多くのクラスは、引数として述語をサポートしています。たとえば、list.findall はパラメーター述語を想定しています。述語の例を次に示します。
署名付きの関数ポインターを想像してください。
<modifier> bool delegate myDelegate<in T>(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();
}
}
}