In this article, we are going to talk about Action
and Func
delegates in C#, We would learn what they are and how to implement them.
Let’s Begin.
Action
and Func
delegates are function objects that allow us to treat functions or methods as data just like int and string. As a result, we can use the functions anywhere. This includes passing them as an argument to other functions.
Action Delegates
Action
delegates are built-in generic delegates in the System namespace
that takes in zero or more input parameters and no out parameter. So, in other words, we can only pass methods with void
return types to Action
delegates.
A base syntax for how to implement an Action
delegate that takes a single parameter is:
public delegate void Action<in T>(T arg);
There are different ways to use an Action
delegate.
Let’s first inspect how, to assigning a method to a Action
delegate:
//Assigning methods to Action Delegates
static void SayHello(string name)
{
Console.WriteLine($"Hello {name}");
}
Action<string> Greet = SayHello;
Greet("John Doe");
Here, we create a method SayHello
that receives a name parameter and writes to the console. Now we can pass the method to the Greet
Action delegate.
Next, we can use an Action
delegate with an Anonymous method:
//Assigning anonymous methods to Action Delegates
Action<int, int> AddNumbers = delegate (int num1, int num2)
{
Console.WriteLine($"Total of {num1} + {num2} is {num1 + num2}");
};
AddNumbers(20, 32);
Now, we create an anonymous method and reference the AddNumbers
Action delegate, this delegate takes in two input parameters and returns the sum of the integers.
Finally, Let’s see how we can do the same with Lambda expressions:
//Using Action Delegates with Lambda expressions
Action<string> Greetings = (name) => Console.WriteLine($"Good day, {name}.");
Greetings("John Doe");
Lambda expressions help simplify the implementation of Action delegates. In the example, we create a function that accepts a name
parameter and writes to the console. Action delegates can take up to 16 input parameters.
Func Delegates
Func
delegate is very similar to Action
delegates in that it takes zero or more input parameters. However, Func
delegates take one out parameter:
public delegate TResult Func<in T, out TResult>(T arg);
This means that we can pass methods with return types to Func
delegates. We can instantiate a Func
delegate that takes in two input parameters with this syntax:
public delegate TResult Func<in T1, in T2, out TResult>(T1 arg1, T2 arg2);
Comparing both Func
syntax, we noticed that the out parameter is always the last parameter. Therefore, Func
delegates with no input parameters would have just an out parameter:
public delegate TResult Func<out TResult>();
Like Action
delegates, we can implement Func
delegates in different ways.
Let’s implement a Func
delegate using a method:
//Assigning methods to Func Delegates
static int AddNumber(int num1, int num2)
{
return num1 + num2;
}
Func<int, int, int> Sum = AddNumber;
Console.WriteLine(Sum(5, 8));
The AddNumber
method takes two integers and returns their sum. It is then passed to the Sum
Func delegate which accepts two input parameters and returns an integer.
Next, Let’s inspect how we can use a Func
delegate with anonymous methods:
//Assigning anonymous methods to Func Delegates
Func<int, bool> OlderThanEighteen = delegate (int age)
{
return age > 18;
};
Console.WriteLine(OlderThanEighteen(19));
The OlderThanEighteen
method receives an anonymous method that checks if the age is greater than 18 and then returns true or false. This Func
delegate receives just an integer input parameter and returns a boolean.
Finally, Let’s use a Func
delegate with a Lambda expression:
//Using Action Delegates with Lambda expressions
Func<string, string, string> GetFullName = (firstName, lastName) => $"Your full name is - {firstName} {lastName}";
Console.WriteLine(GetFullName("John", "Doe"));
This Func
delegate accepts two input parameters of FirstName and LastName then returns FullName. Like Action Delegates, Func Delegates can take a maximum of 16 parameters.
Implementation
double SquareRoot(int number)
{
return Math.Sqrt(number);
}
Func<int, double> GetSquareRoot = GetSquareRoot;
double result = GetSquareRootFunc(20);
Console.WriteLine(result);
The SquareRoot
method receives an integer, then calculates the square root. We then assign the method to the GetSquareRoot
Func delegate. The Func
delegate takes an integer and returns a double as the result of the calculation.
To help decide which of these built-in delegates to use, the most important factor to put into consideration is the return type of the method we wish to assign to our delegate. Therefore, the question would be, does the method have a return type? If yes, go ahead to use a Func
delegate. However, if no, then an Action
delegate is the one for us.
Conclusion
In this article, we have learned:
What an Action delegate is
What a Func delegate is
How to use Action and Func delegates