What are 'closures' in .NET?

前端 未结 12 1935
执念已碎
执念已碎 2020-11-21 13:53

What is a closure? Do we have them in .NET?

If they do exist in .NET, could you please provide a code snippet (preferably in C#) explaining it?

相关标签:
12条回答
  • 2020-11-21 14:08

    Here is a contrived example for C# which I created from similar code in JavaScript:

    public delegate T Iterator<T>() where T : class;
    
    public Iterator<T> CreateIterator<T>(IList<T> x) where T : class
    {
            var i = 0; 
            return delegate { return (i < x.Count) ? x[i++] : null; };
    }
    

    So, here is some code that shows how to use the above code...

    var iterator = CreateIterator(new string[3] { "Foo", "Bar", "Baz"});
    
    // So, although CreateIterator() has been called and returned, the variable 
    // "i" within CreateIterator() will live on because of a closure created 
    // within that method, so that every time the anonymous delegate returned 
    // from it is called (by calling iterator()) it's value will increment.
    
    string currentString;    
    currentString = iterator(); // currentString is now "Foo"
    currentString = iterator(); // currentString is now "Bar"
    currentString = iterator(); // currentString is now "Baz"
    currentString = iterator(); // currentString is now null
    

    Hope that is somewhat helpful.

    0 讨论(0)
  • 2020-11-21 14:11

    A closure is when a function is defined inside another function (or method) and it uses the variables from the parent method. This use of variables which are located in a method and wrapped in a function defined within it, is called a closure.

    Mark Seemann has some interesting examples of closures in his blog post where he does a paralel between oop and functional programming.

    And to make it more detailed

    var workingDirectory = new DirectoryInfo(Environment.CurrentDirectory);//when this variable
    Func<int, string> read = id =>
        {
            var path = Path.Combine(workingDirectory.FullName, id + ".txt");//is used inside this function
            return File.ReadAllText(path);
        };//the entire process is called a closure.
    
    0 讨论(0)
  • 2020-11-21 14:11

    If you write an inline anonymous method (C#2) or (preferably) a Lambda expression (C#3+), an actual method is still being created. If that code is using an outer-scope local variable - you still need to pass that variable to the method somehow.

    e.g. take this Linq Where clause (which is a simple extension method which passes a lambda expression):

    var i = 0;
    var items = new List<string>
    {
        "Hello","World"
    };   
    var filtered = items.Where(x =>
    // this is a predicate, i.e. a Func<T, bool> written as a lambda expression
    // which is still a method actually being created for you in compile time 
    {
        i++;
        return true;
    });
    

    if you want to use i in that lambda expression, you have to pass it to that created method.

    So the first question that arises is: should it be passed by value or reference?

    Pass by reference is (I guess) more preferable as you get read/write access to that variable (and this is what C# does; I guess the team in Microsoft weighed the pros and cons and went with by-reference; According to Jon Skeet's article, Java went with by-value).

    But then another question arises: Where to allocate that i?

    Should it actually/naturally be allocated on the stack? Well, if you allocate it on the stack and pass it by reference, there can be situations where it outlives it's own stack frame. Take this example:

    static void Main(string[] args)
    {
        Outlive();
        var list = whereItems.ToList();
        Console.ReadLine();
    }
    
    static IEnumerable<string> whereItems;
    
    static void Outlive()
    {
        var i = 0;
        var items = new List<string>
        {
            "Hello","World"
        };            
        whereItems = items.Where(x =>
        {
            i++;
            Console.WriteLine(i);
            return true;
        });            
    }
    

    The lambda expression (in the Where clause) again creates a method which refers to an i. If i is allocated on the stack of Outlive, then by the time you enumerate the whereItems, the i used in the generated method will point to the i of Outlive, i.e. to a place in the stack that is no longer accessible.

    Ok, so we need it on the heap then.

    So what the C# compiler does to support this inline anonymous/lambda, is use what is called "Closures": It creates a class on the Heap called (rather poorly) DisplayClass which has a field containing the i, and the Function that actually uses it.

    Something that would be equivalent to this (you can see the IL generated using ILSpy or ILDASM):

    class <>c_DisplayClass1
    {
        public int i;
    
        public bool <GetFunc>b__0()
        {
            this.i++;
            Console.WriteLine(i);
            return true;
        }
    }
    

    It instantiates that class in your local scope, and replaces any code relating to i or the lambda expression with that closure instance. So - anytime you are using the i in your "local scope" code where i was defined, you are actually using that DisplayClass instance field.

    So if I would change the "local" i in the main method, it will actually change _DisplayClass.i ;

    i.e.

    var i = 0;
    var items = new List<string>
    {
        "Hello","World"
    };  
    var filtered = items.Where(x =>
    {
        i++;
        return true;
    });
    filtered.ToList(); // will enumerate filtered, i = 2
    i = 10;            // i will be overwriten with 10
    filtered.ToList(); // will enumerate filtered again, i = 12
    Console.WriteLine(i); // should print out 12
    

    it will print out 12, as "i = 10" goes to that dispalyclass field and changes it just before the 2nd enumeration.

    A good source on the topic is this Bart De Smet Pluralsight module (requires registration) (also ignore his erroneous use of the term "Hoisting" - what (I think) he means is that the local variable (i.e. i) is changed to refer to the the new DisplayClass field).


    In other news, there seems to be some misconception that "Closures" are related to loops - as I understand "Closures" are NOT a concept related to loops, but rather to anonymous methods / lambda expressions use of local scoped variables - although some trick questions use loops to demonstrate it.

    0 讨论(0)
  • 2020-11-21 14:14

    I have an article on this very topic. (It has lots of examples.)

    In essence, a closure is a block of code which can be executed at a later time, but which maintains the environment in which it was first created - i.e. it can still use the local variables etc of the method which created it, even after that method has finished executing.

    The general feature of closures is implemented in C# by anonymous methods and lambda expressions.

    Here's an example using an anonymous method:

    using System;
    
    class Test
    {
        static void Main()
        {
            Action action = CreateAction();
            action();
            action();
        }
    
        static Action CreateAction()
        {
            int counter = 0;
            return delegate
            {
                // Yes, it could be done in one statement; 
                // but it is clearer like this.
                counter++;
                Console.WriteLine("counter={0}", counter);
            };
        }
    }
    

    Output:

    counter=1
    counter=2
    

    Here we can see that the action returned by CreateAction still has access to the counter variable, and can indeed increment it, even though CreateAction itself has finished.

    0 讨论(0)
  • 2020-11-21 14:21

    If you are interested in seeing how C# implements Closure read "I know the answer (its 42) blog"

    The compiler generates a class in the background to encapsulate the anoymous method and the variable j

    [CompilerGenerated]
    private sealed class <>c__DisplayClass2
    {
        public <>c__DisplayClass2();
        public void <fillFunc>b__0()
        {
           Console.Write("{0} ", this.j);
        }
        public int j;
    }
    

    for the function:

    static void fillFunc(int count) {
        for (int i = 0; i < count; i++)
        {
            int j = i;
            funcArr[i] = delegate()
                         {
                             Console.Write("{0} ", j);
                         };
        } 
    }
    

    Turning it into:

    private static void fillFunc(int count)
    {
        for (int i = 0; i < count; i++)
        {
            Program.<>c__DisplayClass1 class1 = new Program.<>c__DisplayClass1();
            class1.j = i;
            Program.funcArr[i] = new Func(class1.<fillFunc>b__0);
        }
    }
    
    0 讨论(0)
  • 2020-11-21 14:21

    Closures are functional values that hold onto variable values from their original scope. C# can use them in the form of anonymous delegates.

    For a very simple example, take this C# code:

        delegate int testDel();
    
        static void Main(string[] args)
        {
            int foo = 4;
            testDel myClosure = delegate()
            {
                return foo;
            };
            int bar = myClosure();
    
        }
    

    At the end of it, bar will be set to 4, and the myClosure delegate can be passed around to be used elsewhere in the program.

    Closures can be used for a lot of useful things, like delayed execution or to simplify interfaces - LINQ is mainly built using closures. The most immediate way it comes in handy for most developers is adding event handlers to dynamically created controls - you can use closures to add behavior when the control is instantiated, rather than storing data elsewhere.

    0 讨论(0)
提交回复
热议问题