Compartilhar via


Procedural Analogs

[Blog Map]  This blog is inactive.  New blog: EricWhite.com/blog

In the last topic, we showed an FP analog to a loop, the ForEach extension method.

There are other analogs. We can do something similar to implement something analogous to the C# switch statement. To do this, we'll write an extension method that takes as parameters, 1) a delegate that returns an integer, and 2) an array of delegates (declared using the params keyword). The extension method indexes into the array using the integer, and calls the appropriate method in the array.

Note that this is a switch statement that can be applied to every item in a collection - it is more powerful than the standard procedural language switch statement.

We can call this the FP Switch Pattern. When writing a program in the FP style, you could use this in a variety of contexts, but the cutest is a small RPN token processor. Here is the RPN token processor, including the code to exercise it, in its entirety.

// the following computes (5*2)-1

// the Token class is included in the listing at the bottom of this post

Token[] tkns = {
new Token(5),
new Token(2),
new Token(TokenType.Multiply),
new Token(1),
new Token(TokenType.Subtract)
};

 

Stack<double> st = new Stack<double>();

 

// The RPN Token Processor

tkns.Switch(
i => (int)i.Type,
s => st.Push(s.Operand),
s => st.Push(st.Pop() + st.Pop()),
s => st.Push(-st.Pop() + st.Pop()),
s => st.Push(st.Pop() * st.Pop()),
s => st.Push(1/st.Pop() * st.Pop())
);

 

Console.WriteLine(st.Pop());

The Switch extension method is almost more interesting because of its signature than the body of the method. Because of the way that lambda expressions interact with the void keyword, we have to declare two extension methods, one for a switch where each delegate returns a type, and another for a switch where each delegate returns void:

public delegate void VoidFunc<T0>(T0 a0);

 

public static class MyExtension
{
public static IEnumerable<TR> Switch<T0, TR>(
this IEnumerable<T0>source,
Func<T0, int>expr,
params Func<T0, TR>[] funcArray)
{
foreach (T0 item in source)
yield return funcArray[expr(item)](item);
}

    public static void Switch<T0>(
this IEnumerable<T0>source,
Func<T0, int>expr,
params VoidFunc<T0>[] funcArray)
{
foreach (T0 item in source)
funcArray[expr(item)](item);
}
}

Here is the program, in its entirety:

namespace MyProgram
{

    public delegate void VoidFunc<T0>(T0 a0);

 

    public static class MyExtension
{
public static IEnumerable<TR> Switch<T0, TR>(
this IEnumerable<T0>source,
Func<T0, int>expr,
params Func<T0, TR>[] funcArray)
{
foreach (T0 item in source)
yield return funcArray[expr(item)](item);
}

 

        public static void Switch<T0>(
this IEnumerable<T0>source,
Func<T0, int>expr,
params VoidFunc<T0>[] funcArray)
{
foreach (T0 item in source)
funcArray[expr(item)](item);
}
}

 

    class Program
{
public enum TokenType
{
Operand,
Add,
Subtract,
Multiply,
Divide
};

 

        public class Token {
public TokenType Type;
public double Operand;
public Token(TokenType type)
{
this.Type = type;
}
public Token(double op)
{
this.Type = TokenType.Operand;
this.Operand = op;
}
}

 

        static void Main(string[] args)
{
Token[] tkns = {
new Token(5),
new Token(2),
new Token(TokenType.Multiply),
new Token(1),
new Token(TokenType.Subtract)
};

 

            Stack<double> st = new Stack<double>();

 

            tkns.Switch(
i => (int)i.Type,
s => st.Push(s.Operand),
s => st.Push(st.Pop() + st.Pop()),
s => st.Push(-st.Pop() + st.Pop()),
s => st.Push(st.Pop() * st.Pop()),
s => st.Push(1/st.Pop() * st.Pop())
);

 

            Console.WriteLine(st.Pop());
}
}
}

When you run this, it prints 9 to the console, as expected.

Next: Pure Functions

Comments

  • Anonymous
    December 25, 2007
    In your code sample, you need to change Sequence. to Enumerable. in order to run and compile on VS2008 RTM.

  • Anonymous
    December 25, 2007
    (sorry; comment applied to previous page, this page compiles fine)

  • Anonymous
    January 18, 2008
    It seems to me that the first switch with return value is not needed for this example. It is just to see how such a switch with return value would look like then?