At the recent Microsoft Professional Developer's Conference there was a presentation by Anders Hejlsberg about The Future of C# which unveiled some of the upcoming features of C# 4.0. Of all the features announced, the one I found most interesting was the introduction of duck typing via a new static type called dynamic.

For those who aren't familiar with Duck Typing, here is the definition from Wikipedia

In computer programming, duck typing is a style of dynamic typing in which an object's current set of methods and properties determines the valid semantics, rather than its inheritance from a particular class. The name of the concept refers to the duck test, attributed to James Whitcomb Riley (see History below), which may be phrased as follows:

If it walks like a duck and quacks like a duck, I would call it a duck.

In duck typing one is concerned with just those aspects of an object that are used, rather than with the type of the object itself. For example, in a non-duck-typed language, one can create a function that takes an object of type Duck and calls that object's walk and quack methods. In a duck-typed language, the equivalent function would take an object of any type and call that object's walk and quack methods. If the object does not have the methods that are called then the function signals a run-time error.

This is perfectly illustrated with the following IronPython program

def PrintAt(container, index):
    print "The value at [%s] is %s" % (index, container[index])

if __name__ == "__main__":
    #create dictionary

    table = {1 : "one", 2 : "two",3 : "three"}
    #create sequence

    list = ("apple", "banana", "cantalope")

    PrintAt(table, 1)
    PrintAt(list, 1)

In the above program the PrintAt() simply requires that the container object passed in supports the index operator '[]' and can accept whatever type is passed in as the index. This means I can pass both a sequence (i.e. a list) or a dictionary (i.e. a hash table) to the function and it returns results even though the semantics of using the index operator is very different for lists and dictionaries.

Proponents of static typing have long argued that features like duck typing lead to hard-to-find bugs which are only detected at runtime after the application has failed instead of during development via compiler errors. However there are many situations even in statically typed programming where the flexibility of duck typing would be beneficial. A common example is invoking JSON or SOAP web services and mapping these structures to objects. 

Recently, I had to write some code at work which spoke to a JSON-based Web service and struggled with how to deal with the fact that C# requires me to define the class of an object up front before I can use it in my application. Given the flexible and schema-less nature of JSON, this was a problem. I ended up using the JsonReaderWriterFactory to create an XmlDictionaryReader which maps JSON into an XML document which can then be processed flexibly using XML technologies. Here's what the code looked like

using System;
using System.IO; 
using System.Runtime.Serialization.Json;
using System.Text;
using System.Xml; 

namespace Test
{
    class Program
    {
        static void Main(string[] args)
        {
            string json = @"{ ""firstName"": ""John"", ""lastName"": ""Smith"", ""age"": 21 }";
            var stream = new MemoryStream(ASCIIEncoding.Default.GetBytes(json));
            var reader = JsonReaderWriterFactory.CreateJsonReader(stream, XmlDictionaryReaderQuotas.Max); 

            var doc = new XmlDocument(); 
            doc.Load(reader);
            
            //error handling omitted for brevity
            string firstName = doc.SelectSingleNode("/root/firstName").InnerText; 
            int age          = Int32.Parse(doc.SelectSingleNode("/root/age").InnerText); 
            
            Console.WriteLine("{0} will be {1} next year", firstName, age + 1);
            Console.ReadLine();  
        }
    }
}

It works but the code is definitely not as straightforward as interacting with JSON from Javascript. This is where the dynamic type from C# 4.0 would be very useful. With this type, I could rewrite the above code as follows

using System;
using System.IO; 
using System.Runtime.Serialization.Json;
using System.Text;
using System.Xml; 

namespace Test
{
    class Program
    {
        static void Main(string[] args)
        {
            string json = @"{ ""firstName"": ""John"", ""lastName"": ""Smith"", ""age"": 21 }";
            var stream = new MemoryStream(ASCIIEncoding.Default.GetBytes(json));
            dynamic person = JsonObjectReaderFactory.CreateJsonObjectReader(stream, XmlDictionaryReaderQuotas.Max); 

            Console.WriteLine("{0} will be {1} next year", person.firstName, person.age + 1);
            Console.ReadLine();  
        }
    }
}

In the C# 4.0 version I can declare a person object whose class/type I don't have to define at compile time. Instead the property accesses are converted to dynamic calls to the named properties using reflection by the compiler. So at runtime when my [imaginary] JsonObjectReader dynamically creates the person objects from the input JSON, my application works as expected with a lot less lines of code.

It's amazing how Python-like C# gets each passing year.

FURTHER READING:


 

Saturday, 08 November 2008 17:28:19 (GMT Standard Time, UTC+00:00)
Or, rather than waiting for C# 4, you could just use the JsonObject type available today in Silverlight 2 (in System.Json.dll):

JsonObject person = (JsonObject) JsonValue.Parse(json);
int age = person["age"];
Console.WriteLine("{0} will be {1} next year", person["firstName"], age + 1);

That's less code than the dynamically-typed version. And you can use it today.
Saturday, 08 November 2008 17:47:44 (GMT Standard Time, UTC+00:00)
Something has gone horribly wrong with your code samples, when rendered within Google Reader, at least for me, viewing with Safari.
Sunday, 09 November 2008 00:39:12 (GMT Standard Time, UTC+00:00)
+1 on JsonObject. I use this with LINQ and anonymous types and it's a beautiful thing. Duck typing will improve things though, by making it possible to pass such results between functions easier.
Sunday, 09 November 2008 09:55:29 (GMT Standard Time, UTC+00:00)
I love your title "C# is the Next Python: Duck Typing and C# 4.0" It gives me warm feelings about the current status of Python vs C#.

Python has come a long way from the automatic derisory attacks on its use of whitespace as block delimeter, and the automatic (and wrong), cry that it was weakly typed.

- Paddy.
Sunday, 09 November 2008 20:21:36 (GMT Standard Time, UTC+00:00)
Patrick, as best as I can tell the problem is in the underlying feed. The newlines appear to be missing from there too.

Dare, if the JSON-outputting API that you allude to is Reader's, you can also get XML out of it by adding an output=xml query parameter to most of our URLs (if it's easier to go straight to XML that way).

Mihai Parparita
Google Reader Engineer
Monday, 10 November 2008 14:56:50 (GMT Standard Time, UTC+00:00)
What they've announced at PDC isn't duck typing. But it could be.

What has been announced is some compiler magic to treat the "dynamic" keyword to mean "dispatch via the DLR's IDynamicObject interface". This means late bound support for COM and DLR based languages. In fact, it's just like VB.NETs late bound COM support.

What they haven't announced is treating traditional C# classes as dynamic objects. So, there's only limited support for duck typing - only really in consuming objects that are already defined as being dynamic. I can't treat 2 C# classes the same because they support the one method I need to call.

Of course, if there's an interface to implement, then there's fun to be had. The imaginary JsonObjectRead could implement IDynamicObject and dynamically query an internal representation of a JSON stream when dynamically queried for properties.

Or, I could create an implementation of IDynamicObject that would use reflection to delegate to a wrapped static class. My consuming method would end up taking in an object, so I'd actually be very loosely typed, but hey ho - it's still duck typing.

Another option would be some form of "duck-casting", such as proposed by Jon Skeet (check out the comments, too) http://msmvps.com/blogs/jon_skeet/archive/2008/10/30/c-4-0-dynamic-lt-t-gt.aspx, giving me strongly typed methods, but the ability to cast on the fly. But that's not too different to a project that's been around for quite a while: http://www.deftflux.net/blog/page/Duck-Typing-Project.aspx

I'm not too bothered with the COM and DLR interop (which does seem to be the main story), but the hacks that could come out of this could be very nice (dynamic mock objects?)

Anyway, some more info at Chris Burrow's blog:

http://blogs.msdn.com/cburrows/archive/2008/10/27/c-dynamic.aspx
http://blogs.msdn.com/cburrows/archive/2008/10/28/c-dynamic-part-ii.aspx

Cheers
Matt
Comments are closed.