1123

Is there a way to deserialize JSON content into a C# dynamic type? It would be nice to skip creating a bunch of classes in order to use the DataContractJsonSerializer.

iliketocode
  • 6,978
  • 5
  • 46
  • 60
jswanson
  • 15,714
  • 6
  • 19
  • 15
  • 6
    If you want something 'dynamic', why not just use the get-style accessors that come with most JSON decoders that don't go to plain-old-object? (e.g. is there really a need for 'dynamic' object creation?) [json.org](http://json.org/) has a bunch of links for C# JSON implementations. –  Jun 29 '10 at 16:07
  • I'm working on a project that is trying to keep external dependencies to a minimum. So if it's possible to something with the stock .net serializers and types that would be preferred. Of course if it's not possible I'm hitting up json.org. Thanks! – jswanson Jun 29 '10 at 16:14
  • 52
    I'm really surprised the C# team added 'dynamic' but then there is no way in the CLR to convert a JSON object to a dynamic CLR class instance. – Frank Schwieterman Jul 20 '10 at 23:12
  • 2
    Unfortunately the accepted answer doesn't work in .NET 4 RTM. I posted an answer that helped me get going with this which might be useful to others. – Drew Noakes Sep 27 '10 at 17:47
  • Update October 2018, this is how the magic happens: https://stackoverflow.com/a/48023576/4180382 – Ole EH Dufour Oct 10 '18 at 18:10

30 Answers30

723

If you are happy to have a dependency upon the System.Web.Helpers assembly, then you can use the Json class:

dynamic data = Json.Decode(json);

It is included with the MVC framework as an additional download to the .NET 4 framework. Be sure to give Vlad an upvote if that's helpful! However if you cannot assume the client environment includes this DLL, then read on.


An alternative deserialisation approach is suggested here. I modified the code slightly to fix a bug and suit my coding style. All you need is this code and a reference to System.Web.Extensions from your project:

using System;
using System.Collections;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Dynamic;
using System.Linq;
using System.Text;
using System.Web.Script.Serialization;

public sealed class DynamicJsonConverter : JavaScriptConverter
{
    public override object Deserialize(IDictionary<string, object> dictionary, Type type, JavaScriptSerializer serializer)
    {
        if (dictionary == null)
            throw new ArgumentNullException("dictionary");

        return type == typeof(object) ? new DynamicJsonObject(dictionary) : null;
    }

    public override IDictionary<string, object> Serialize(object obj, JavaScriptSerializer serializer)
    {
        throw new NotImplementedException();
    }

    public override IEnumerable<Type> SupportedTypes
    {
        get { return new ReadOnlyCollection<Type>(new List<Type>(new[] { typeof(object) })); }
    }

    #region Nested type: DynamicJsonObject

    private sealed class DynamicJsonObject : DynamicObject
    {
        private readonly IDictionary<string, object> _dictionary;

        public DynamicJsonObject(IDictionary<string, object> dictionary)
        {
            if (dictionary == null)
                throw new ArgumentNullException("dictionary");
            _dictionary = dictionary;
        }

        public override string ToString()
        {
            var sb = new StringBuilder("{");
            ToString(sb);
            return sb.ToString();
        }

        private void ToString(StringBuilder sb)
        {
            var firstInDictionary = true;
            foreach (var pair in _dictionary)
            {
                if (!firstInDictionary)
                    sb.Append(",");
                firstInDictionary = false;
                var value = pair.Value;
                var name = pair.Key;
                if (value is string)
                {
                    sb.AppendFormat("{0}:\"{1}\"", name, value);
                }
                else if (value is IDictionary<string, object>)
                {
                    new DynamicJsonObject((IDictionary<string, object>)value).ToString(sb);
                }
                else if (value is ArrayList)
                {
                    sb.Append(name + ":[");
                    var firstInArray = true;
                    foreach (var arrayValue in (ArrayList)value)
                    {
                        if (!firstInArray)
                            sb.Append(",");
                        firstInArray = false;
                        if (arrayValue is IDictionary<string, object>)
                            new DynamicJsonObject((IDictionary<string, object>)arrayValue).ToString(sb);
                        else if (arrayValue is string)
                            sb.AppendFormat("\"{0}\"", arrayValue);
                        else
                            sb.AppendFormat("{0}", arrayValue);

                    }
                    sb.Append("]");
                }
                else
                {
                    sb.AppendFormat("{0}:{1}", name, value);
                }
            }
            sb.Append("}");
        }

        public override bool TryGetMember(GetMemberBinder binder, out object result)
        {
            if (!_dictionary.TryGetValue(binder.Name, out result))
            {
                // return null to avoid exception.  caller can check for null this way...
                result = null;
                return true;
            }

            result = WrapResultObject(result);
            return true;
        }

        public override bool TryGetIndex(GetIndexBinder binder, object[] indexes, out object result)
        {
            if (indexes.Length == 1 && indexes[0] != null)
            {
                if (!_dictionary.TryGetValue(indexes[0].ToString(), out result))
                {
                    // return null to avoid exception.  caller can check for null this way...
                    result = null;
                    return true;
                }

                result = WrapResultObject(result);
                return true;
            }

            return base.TryGetIndex(binder, indexes, out result);
        }

        private static object WrapResultObject(object result)
        {
            var dictionary = result as IDictionary<string, object>;
            if (dictionary != null)
                return new DynamicJsonObject(dictionary);

            var arrayList = result as ArrayList;
            if (arrayList != null && arrayList.Count > 0)
            {
                return arrayList[0] is IDictionary<string, object> 
                    ? new List<object>(arrayList.Cast<IDictionary<string, object>>().Select(x => new DynamicJsonObject(x))) 
                    : new List<object>(arrayList.Cast<object>());
            }

            return result;
        }
    }

    #endregion
}

You can use it like this:

string json = ...;

var serializer = new JavaScriptSerializer();
serializer.RegisterConverters(new[] { new DynamicJsonConverter() });

dynamic obj = serializer.Deserialize(json, typeof(object));

So, given a JSON string:

{
  "Items":[
    { "Name":"Apple", "Price":12.3 },
    { "Name":"Grape", "Price":3.21 }
  ],
  "Date":"21/11/2010"
}

The following code will work at runtime:

dynamic data = serializer.Deserialize(json, typeof(object));

data.Date; // "21/11/2010"
data.Items.Count; // 2
data.Items[0].Name; // "Apple"
data.Items[0].Price; // 12.3 (as a decimal)
data.Items[1].Name; // "Grape"
data.Items[1].Price; // 3.21 (as a decimal)
Drew Noakes
  • 284,599
  • 158
  • 653
  • 723
  • Thanks Drew, your TryGetMember sorted my issue around recursing into nested collections, but could you perhaps let us know what it is that makes it work. Is it in the fact that array lists are cast into IDictionary and then projected as DynamicJsonObjects, rather than being projected as arrayList members? Hope this isn't a too dumb question. Thanks for the answer :) – Mark Dickinson Feb 09 '11 at 13:05
  • @Mark, it's been a while since I looked at this but from memory what you're describing sounds right. – Drew Noakes Feb 09 '11 at 14:01
  • 3
    I get an error in dynamic obj = serializer.Deserialize(json, typeof(object)); saying that no overload for method with 2 arguments..wrong dll or what? – Stewie Griffin Jun 18 '11 at 20:17
  • @Stewie, the type I'm using is `System.Web.Script.Serialization.JavaScriptSerializer` which is in version 4.0.0.0 of `System.Web.dll`. – Drew Noakes Jun 19 '11 at 08:36
  • System.Web.Script.Serialization.JavaScriptSerializer is in System.Web.Extensions.dll - http://goo.gl/8zRrj – Rush Frisby Dec 12 '11 at 16:43
  • 2
    I found that your ToString method wasn't working for me, so I rewrote it. It might have some bugs, but it's working over my dataset, so I'll provide it here for anyone else who might be having trouble with this: http://pastebin.com/BiRmQZdz – Quantumplation Dec 18 '11 at 13:22
  • @Tim, I can't see why not, but I'm not familiar enough with VB.NET to convert the code for you. You could compile the C# code and use it via an assembly reference, or try to convert it yourself. I don't think it would be too hard to convert. – Drew Noakes Jan 09 '12 at 09:24
  • Yes, I already used the c# assembly, but it is still the same. This "dynamic" object detection didnt work. In my case, "data" is an array of Objects that each of them includes key-value pair for json data. – Timuçin Jan 09 '12 at 17:36
  • 37
    You can use System.Web.Helpers.Json - it offers a Decode method that returns a dynamic object. I've also posted this info as an answer. – Vlad Iliescu Feb 29 '12 at 07:30
  • 2
    sometimes in js you have fields with special chars like "background-color". To access such fields in js you do obj["background-color"]. How can I access such fields from c# after deserializing to dynamic object? I can't do obj.background-color, of course, and obj["background-color"] doesn't seem to work. It would be nice if the dynamic object could also be accessed as a dictionary, at the same time, exactly like in js. – Radu Simionescu Sep 28 '12 at 11:59
  • @RaduSimionescu, have you tried swapping the hyphen for an underscore? I haven't tried it myself, but I recall something somewhere about this. – Drew Noakes Oct 01 '12 at 12:16
  • 2
    @RaduSimionescu I am probably a bit late, but maybe this helps future visitors. I had the same problem, just with the field-name `params` (which is a keyword in C#). In addition to `TryGetMember` you can override `TryGetIndex`, which gives you exactly the same behavior as in JS. Then you can do `obj["params"]` or `obj["background-color"]` for awkward field names. – Martin Ender Jan 20 '13 at 14:30
  • 1
    System.Web.Helpers could be downloaded in VS2015 using Package Manage Console with this command: Install-Package System-Web-Helpers.dll Also, Json in System.Web.Mvc conflicts with Json in System.Web.Helpers. So you can use dynamic data = System.Web.Helpers.Json.Decode(myJSONString); – Mahesh Dec 15 '16 at 18:10
  • 1
    FYI If you're using .NET Core, they moved this thing into the "microsoft-web-helpers" package. – starmandeluxe Feb 14 '18 at 08:42
  • 1
    to use System.Web.Helpers you need to add a nuget package: https://www.nuget.org/packages/microsoft-web-helpers – syonip Mar 05 '19 at 12:37
  • nuget.org/packages/microsoft-web-helpers cause error: System.IO.FileNotFoundException File name: 'System.Web.Extensions, Version=4.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35' at System.Web.Helpers.Json.Decode(String value) – zolty13 May 25 '20 at 14:59
  • Last link in answer is dead - *"403 Site Disabled | This site is stopped"*. – Pang Oct 23 '20 at 03:18
  • `data.Items.Count` didn't work for me - `data.Items.Length` did. – vt100 Mar 11 '21 at 11:41
  • visual studio didn't recognize Json class even after added using System.Web.Helpers; – RainCast Apr 21 '21 at 09:22
706

It's pretty simple using Json.NET:

dynamic stuff = JsonConvert.DeserializeObject("{ 'Name': 'Jon Smith', 'Address': { 'City': 'New York', 'State': 'NY' }, 'Age': 42 }");

string name = stuff.Name;
string address = stuff.Address.City;

Also using Newtonsoft.Json.Linq:

dynamic stuff = JObject.Parse("{ 'Name': 'Jon Smith', 'Address': { 'City': 'New York', 'State': 'NY' }, 'Age': 42 }");

string name = stuff.Name;
string address = stuff.Address.City;

Documentation: Querying JSON with dynamic

Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
Tom Peplow
  • 7,181
  • 1
  • 14
  • 8
  • 1
    Interesting, using MVC, how do I bind this to the view? I read through a few posts about dynamic or anonymous type bind, but I'm not seeing an example of how to use it. Say I have json that may have a lot of layers. – Vyache Mar 23 '14 at 02:16
  • How can you introspect the dynamic `stuff`? – Hot Licks Jun 04 '14 at 17:18
  • 15
    @HotLicks: To introspect the dynamic `stuff` do something like: `foreach (Newtonsoft.Json.Linq.JProperty jproperty in stuff) { Console.WriteLine("jproperty.Name = {0}", jproperty.Name);}` – Matthias Jun 15 '14 at 20:48
  • 11
    What's the difference between JsonConvert.DeserializeObject and JObject.Parse ? The answer is using them both in the same way to do the same thing but doesn't explain the difference. – cja Oct 08 '14 at 16:04
  • 10
    @TomPeplow Tried this. It didn't work for me. It says that "JObject doesn't implement 'Name'". – Lee Louviere Feb 18 '15 at 17:27
  • 5
    @cja no difference: http://stackoverflow.com/questions/23645034/jobject-parse-vs-jsonconvert-deserializeobject – nawfal Jun 15 '15 at 09:16
  • 10
    I can't get this to work. I've narrowed the issue down to being inside an `async` method. If I make the method synchronous it works as expected. However, make the method `async` and I _can't_ get a `dynamic`, I just get an `object`. Explicit casting does nothing, still just gives me an `object`. Is anyone else experiencing this? – codeConcussion Dec 11 '15 at 19:57
  • @codeConcussion It is possible some people having issues are missing a reference to `Microsoft.CSharp`? `dynamic` requires this reference (as does async -- I might be wrong there) – Michael Blackburn Aug 05 '16 at 15:23
  • 1
    I would use Newtonsoft Json but documentation is very poor. Lacks a lot in examples. I would need to know how to deal with arrays, how to deal with subobjects as properties, how to check if a property exists, how the json value types (integers, decimals, boolean, null) are converted in c#... One simple sample would be enough, but really looks like I am looking for the weirdest thing. – davidthegrey Mar 09 '19 at 17:15
  • If you can't make it work, try this: `dynamic v = JsonConvert.DeserializeObject(someJSONstring);` – Lucio M. Tato Jul 30 '19 at 07:10
  • 3
    @codeConcussion had the same issue with the async method that retrieves an object rather than a dynamic. to fix it simply do an explicit cast to dynamic and you're OK (myresult as dynamic). – Andrei U Aug 17 '19 at 22:08
  • I think the VS debugger Watch window fails to retrieve properties when called inside an async method. Works fine from a non-async method – stuzor Aug 30 '21 at 02:55
308

You can do this using System.Web.Helpers.Json - its Decode method returns a dynamic object which you can traverse as you like.

It's included in the System.Web.Helpers assembly (.NET 4.0).

var dynamicObject = Json.Decode(jsonString);
Vlad Iliescu
  • 7,887
  • 5
  • 24
  • 23
  • 26
    FYI System.Web.Helpers.dll requires .net 4.0 but is not included in .net 4.0. It can be installed with ASP.NET MVC 3 – jbtule Mar 30 '12 at 20:44
  • 7
    You will find this assembly in the Extensions group under Assemblies in Visual Studio 2012 – W3Max Jan 17 '13 at 13:55
  • 1
    Any Issues with using dynamic ? How can we handle exceptions efficiently if input JSON does not contain the properties.. – Usama Khalil Apr 15 '13 at 10:24
  • 5
    If you're wanting to strongly type the model then be sure to use the Json.Decode(string) method. – Mike Apr 15 '13 at 20:47
  • 2
    To add this library to your project: http://stackoverflow.com/questions/8037895/where-can-i-find-system-web-helpers-system-web-webpages-and-system-web-razor –  Jul 10 '14 at 17:50
86

.NET 4.0 has a built-in library to do this:

using System.Web.Script.Serialization;
JavaScriptSerializer jss = new JavaScriptSerializer();
var d = jss.Deserialize<dynamic>(str);

This is the simplest way.

Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
Peter Long
  • 3,914
  • 2
  • 21
  • 17
  • Most of the previous answers came before .NET 4.0 RTM. – jswanson Jun 13 '11 at 14:08
  • 29
    have you tried this? It returns `Dictionary`. Unless I'm missing something, your example does not return a dynamic object. – sergiopereira Jun 13 '11 at 15:15
  • @sergiopereira yes I did. you can cast any type to dynamic type. so why not just do a cast? – Peter Long Jun 14 '11 at 08:05
  • 19
    This doesn't work, it just return a dict in the form of a dynamic – mattmanser Jun 30 '11 at 15:56
  • @mattmanser, please learn what is dynamic. every dynamic type would be resolved to a static type finally. – Peter Long Jun 30 '11 at 23:51
  • 56
    @Peter Long I believe I have failed to state my case clearly, dear fellow. Let me attempt to rectify my error. I know what a dynamic is. This doesn't allow you to pass in a JSON object and use d.code, you'd have to do d["code"].Value, which isn't what most people finding this answer want, we already know how to get the dictionary and casting it to a dynamic is a total waste of time. I respectfully disagree, sir. – mattmanser Jul 01 '11 at 09:22
  • 4
    @mattmanser, `we already know how to get the dictionary and casting it to a dynamic`. It does not have to be a dictionay. Json also have lists besides dictionary. And also lists and dictionaries could be nested. My code could handle all of these situations. BUT your method can NOT. – Peter Long Jul 14 '11 at 02:44
  • @mattmanser, instead of `d["code"].Value`, you want `d.code`. As far as I know, it is impossible(or meaningless). Think of dynamic as deferring part of the compiler’s job to runtime. That's what dynamic is useful for. Dynamic is not mean to solve your problem. – Peter Long Jul 14 '11 at 02:58
  • this is simple and direct-to-the-point answer...**and it works**. While there are 3rd party libraries that are more full-featured and more efficient (ServiceStack comes to mind, NOT the overrated and super-bloated JSON.NET libraries), sometimes it's better to have the native option (which is available since .NET 4.x) if you don't necessarily need the speed...also, @mattmanser is wrong on all counts; he doesn't seem to grasp what dynamic types are at the time he posted his comments. – Dexter Legaspi Feb 22 '14 at 16:39
  • 4
    @mattmanser is right; it is possible to implement `IDynamicMetaObjectProvider` (or use e.g. `ExpandoObject`) that is able to intercept properties and look them up in an internal dictionary. This combined with the use of `dynamic` allows code such as `d.code` to be used. It's kind of pointless to cast a dictionary to a dynamic. – Stephen Drew Oct 23 '14 at 10:36
  • 1
    I upvoted this answer even though the answer doesnt strictly return a `IDynamicMetaObjectProvider` object. I mean it's useful for people who dont strictly want it to be dynamic. The OP states he wants to avoid creating custom classes and attributes and this answer helps. Peter Long could have been more clear about that in his answer, though. Can I edit this answer? – nawfal Jun 15 '15 at 07:04
  • The discussion above is a result of confusion between a 'dynamic' variable declaration (@PeterLong's topic) and the 'DynamicObject' type (@mattmanser's desired result). For details on both see the documentation (as always): https://docs.microsoft.com/en-us/dotnet/csharp/programming-guide/types/walkthrough-creating-and-using-dynamic-objects https://docs.microsoft.com/en-us/dotnet/csharp/programming-guide/types/using-type-dynamic – LanchPad Aug 31 '18 at 15:11
85

Simple "string JSON data" to object without any third-party DLL file:

WebClient client = new WebClient();
string getString = client.DownloadString("https://graph.facebook.com/zuck");

JavaScriptSerializer serializer = new JavaScriptSerializer();
dynamic item = serializer.Deserialize<object>(getString);
string name = item["name"];

//note: JavaScriptSerializer in this namespaces
//System.Web.Script.Serialization.JavaScriptSerializer

Note: You can also using your custom object.

Personel item = serializer.Deserialize<Personel>(getString);
Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
İbrahim Özbölük
  • 3,082
  • 22
  • 20
36

You can achieve that with the help of Newtonsoft.Json. Install it from NuGet and then:

using Newtonsoft.Json;

dynamic results = JsonConvert.DeserializeObject<dynamic>(YOUR_JSON);
Pang
  • 9,073
  • 146
  • 84
  • 117
Waleed Naveed
  • 1,915
  • 1
  • 21
  • 42
29

JsonFx can deserialize JSON content into dynamic objects.

Serialize to/from dynamic types (default for .NET 4.0):

var reader = new JsonReader(); var writer = new JsonWriter();

string input = @"{ ""foo"": true, ""array"": [ 42, false, ""Hello!"", null ] }";
dynamic output = reader.Read(input);
Console.WriteLine(output.array[0]); // 42
string json = writer.Write(output);
Console.WriteLine(json); // {"foo":true,"array":[42,false,"Hello!",null]}
Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
jbtule
  • 30,814
  • 12
  • 94
  • 127
23

Another way using Newtonsoft.Json:

dynamic stuff = Newtonsoft.Json.JsonConvert.DeserializeObject("{ color: 'red', value: 5 }");
string color = stuff.color;
int value = stuff.value;
Jonas Lundgren
  • 1,001
  • 10
  • 16
20

I made a new version of the DynamicJsonConverter that uses Expando Objects. I used expando objects, because I wanted to Serialize the dynamic back into JSON using Json.NET.

using System;
using System.Collections;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Dynamic;
using System.Web.Script.Serialization;

public static class DynamicJson
{
    public static dynamic Parse(string json)
    {
        JavaScriptSerializer jss = new JavaScriptSerializer();
        jss.RegisterConverters(new JavaScriptConverter[] { new DynamicJsonConverter() });

        dynamic glossaryEntry = jss.Deserialize(json, typeof(object)) as dynamic;
        return glossaryEntry;
    }

    class DynamicJsonConverter : JavaScriptConverter
    {
        public override object Deserialize(IDictionary<string, object> dictionary, Type type, JavaScriptSerializer serializer)
        {
            if (dictionary == null)
                throw new ArgumentNullException("dictionary");

            var result = ToExpando(dictionary);

            return type == typeof(object) ? result : null;
        }

        private static ExpandoObject ToExpando(IDictionary<string, object> dictionary)
        {
            var result = new ExpandoObject();
            var dic = result as IDictionary<String, object>;

            foreach (var item in dictionary)
            {
                var valueAsDic = item.Value as IDictionary<string, object>;
                if (valueAsDic != null)
                {
                    dic.Add(item.Key, ToExpando(valueAsDic));
                    continue;
                }
                var arrayList = item.Value as ArrayList;
                if (arrayList != null && arrayList.Count > 0)
                {
                    dic.Add(item.Key, ToExpando(arrayList));
                    continue;
                }

                dic.Add(item.Key, item.Value);
            }
            return result;
        }

        private static ArrayList ToExpando(ArrayList obj)
        {
            ArrayList result = new ArrayList();

            foreach (var item in obj)
            {
                var valueAsDic = item as IDictionary<string, object>;
                if (valueAsDic != null)
                {
                    result.Add(ToExpando(valueAsDic));
                    continue;
                }

                var arrayList = item as ArrayList;
                if (arrayList != null && arrayList.Count > 0)
                {
                    result.Add(ToExpando(arrayList));
                    continue;
                }

                result.Add(item);
            }
            return result;
        }

        public override IDictionary<string, object> Serialize(object obj, JavaScriptSerializer serializer)
        {
            throw new NotImplementedException();
        }

        public override IEnumerable<Type> SupportedTypes
        {
            get { return new ReadOnlyCollection<Type>(new List<Type>(new[] { typeof(object) })); }
        }
    }
}
Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
Jason Bolton
  • 201
  • 2
  • 3
14

Creating dynamic objects with Newtonsoft.Json works really great.

//json is your string containing the JSON value
dynamic data = JsonConvert.DeserializeObject<dynamic>(json);

Now you can access the data object just like if it was a regular object. This is the JSON object we currently have as an example:

{ "ID":123,"Name":"Jack","Numbers":[1, 2, 3] }

This is how you access it after deserialization:

data.ID //Retrieve the int
data.Name //Retrieve the string
data.Numbers[0] //Retrieve the first element in the array
akac
  • 161
  • 3
  • 9
10

I use http://json2csharp.com/ to get a class representing the JSON object.

Input:

{
   "name":"John",
   "age":31,
   "city":"New York",
   "Childs":[
      {
         "name":"Jim",
         "age":11
      },
      {
         "name":"Tim",
         "age":9
      }
   ]
}

Output:

public class Child
{
    public string name { get; set; }
    public int age { get; set; }
}

public class Person
{
    public string name { get; set; }
    public int age { get; set; }
    public string city { get; set; }
    public List<Child> Childs { get; set; }
}

After that I use Newtonsoft.Json to fill the class:

using Newtonsoft.Json;

namespace GitRepositoryCreator.Common
{
    class JObjects
    {
        public static string Get(object p_object)
        {
            return JsonConvert.SerializeObject(p_object);
        }
        internal static T Get<T>(string p_object)
        {
            return JsonConvert.DeserializeObject<T>(p_object);
        }
    }
}

You can call it like this:

Person jsonClass = JObjects.Get<Person>(stringJson);

string stringJson = JObjects.Get(jsonClass);

PS:

If your JSON variable name is not a valid C# name (name starts with $) you can fix that like this:

public class Exception
{
   [JsonProperty(PropertyName = "$id")]
   public string id { get; set; }
   public object innerException { get; set; }
   public string message { get; set; }
   public string typeName { get; set; }
   public string typeKey { get; set; }
   public int errorCode { get; set; }
   public int eventId { get; set; }
}
Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
RoJaIt
  • 432
  • 3
  • 10
10

I came here to find an answer for .NET Core, without any third-party or additional references. It works fine if you use ExpandoObject with the standard JsonSerializer class. Here is the example that worked for me:

using System.Text.Json;
using System.Dynamic;

dynamic json = JsonSerializer.Deserialize<ExpandoObject>(jsonText);
Console.WriteLine(json.name);

This code prints out the string value of a name property that exists within the JSON text passed into the Deserialize method. Voila - no additional libraries, no nothing. Just .NET core.

Edit: May have a problem for several levels of json with nested elements. Worked for a single-level flat object.

Tengiz
  • 7,426
  • 26
  • 39
  • It's not working in .net 6, Any idea?. I want to read property which have array of elements. – Anirudha Gupta Feb 12 '22 at 15:45
  • It will only work for the primitive type properties because the expando object handles the property read by name and returns the value as-is. The thing is that the Console.WriteLine converts the value to the string by calling ToString which for the primitive types will give the correct value. For the array, you will perhaps see not the actual value but the object type in the output. – Tengiz Feb 14 '22 at 13:36
8

The simplest way is:

Just include this DLL file.

Use the code like this:

dynamic json = new JDynamic("{a:'abc'}");
// json.a is a string "abc"

dynamic json = new JDynamic("{a:3.1416}");
// json.a is 3.1416m

dynamic json = new JDynamic("{a:1}");
// json.a is

dynamic json = new JDynamic("[1,2,3]");
/json.Length/json.Count is 3
// And you can use json[0]/ json[2] to get the elements

dynamic json = new JDynamic("{a:[1,2,3]}");
//json.a.Length /json.a.Count is 3.
// And you can use  json.a[0]/ json.a[2] to get the elements

dynamic json = new JDynamic("[{b:1},{c:1}]");
// json.Length/json.Count is 2.
// And you can use the  json[0].b/json[1].c to get the num.
Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
user1006544
  • 1,444
  • 2
  • 16
  • 26
7

Another option is to "Paste JSON as classes" so it can be deserialised quick and easy.

  1. Simply copy your entire JSON
  2. In Visual Studio: Click EditPaste SpecialPaste JSON as classes

Here is a better explanation n piccas... ‘Paste JSON As Classes’ in ASP.NET and Web Tools 2012.2 RC

Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
nitsram
  • 658
  • 5
  • 5
6

You can use using Newtonsoft.Json

var jRoot = 
 JsonConvert.DeserializeObject<dynamic>(Encoding.UTF8.GetString(resolvedEvent.Event.Data));

resolvedEvent.Event.Data is my response getting from calling core Event .

ΩmegaMan
  • 26,526
  • 10
  • 91
  • 107
Vivek Shukla
  • 1,445
  • 1
  • 12
  • 13
6

You can extend the JavaScriptSerializer to recursively copy the dictionary it created to expando object(s) and then use them dynamically:

static class JavaScriptSerializerExtensions
{
    public static dynamic DeserializeDynamic(this JavaScriptSerializer serializer, string value)
    {
        var dictionary = serializer.Deserialize<IDictionary<string, object>>(value);
        return GetExpando(dictionary);
    }

    private static ExpandoObject GetExpando(IDictionary<string, object> dictionary)
    {
        var expando = (IDictionary<string, object>)new ExpandoObject();

        foreach (var item in dictionary)
        {
            var innerDictionary = item.Value as IDictionary<string, object>;
            if (innerDictionary != null)
            {
                expando.Add(item.Key, GetExpando(innerDictionary));
            }
            else
            {
                expando.Add(item.Key, item.Value);
            }
        }

        return (ExpandoObject)expando;
    }
}

Then you just need to having a using statement for the namespace you defined the extension in (consider just defining them in System.Web.Script.Serialization... another trick is to not use a namespace, then you don't need the using statement at all) and you can consume them like so:

var serializer = new JavaScriptSerializer();
var value = serializer.DeserializeDynamic("{ 'Name': 'Jon Smith', 'Address': { 'City': 'New York', 'State': 'NY' }, 'Age': 42 }");

var name = (string)value.Name; // Jon Smith
var age = (int)value.Age;      // 42

var address = value.Address;
var city = (string)address.City;   // New York
var state = (string)address.State; // NY
alonzofox
  • 91
  • 1
  • 4
5

Look at the article I wrote on CodeProject, one that answers the question precisely:

Dynamic types with JSON.NET

There is way too much for re-posting it all here, and even less point since that article has an attachment with the key/required source file.

vitaly-t
  • 22,286
  • 10
  • 106
  • 127
5

Try this:

  var units = new { Name = "Phone", Color= "White" };
    var jsonResponse = JsonConvert.DeserializeAnonymousType(json, units);
Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
Nirupam
  • 122
  • 2
  • 4
5

I am using like this in my code and it's working fine

using System.Web.Script.Serialization;
JavaScriptSerializer oJS = new JavaScriptSerializer();
RootObject oRootObject = new RootObject();
oRootObject = oJS.Deserialize<RootObject>(Your JSon String);
Vasim Shaikh
  • 4,385
  • 2
  • 20
  • 50
  • 1
    but that's not what the question is asking about. there's a different when you have to specify the type for every json string and working with dynamic type. – Illuminati Jun 07 '16 at 12:31
4

Use DataSet(C#) with JavaScript. A simple function for creating a JSON stream with DataSet input. Create JSON content like (multi table dataset):

[[{a:1,b:2,c:3},{a:3,b:5,c:6}],[{a:23,b:45,c:35},{a:58,b:59,c:45}]]

Just client side, use eval. For example,

var d = eval('[[{a:1,b:2,c:3},{a:3,b:5,c:6}],[{a:23,b:45,c:35},{a:58,b:59,c:45}]]')

Then use:

d[0][0].a // out 1 from table 0 row 0

d[1][1].b // out 59 from table 1 row 1

// Created by Behnam Mohammadi And Saeed Ahmadian
public string jsonMini(DataSet ds)
{
    int t = 0, r = 0, c = 0;
    string stream = "[";

    for (t = 0; t < ds.Tables.Count; t++)
    {
        stream += "[";
        for (r = 0; r < ds.Tables[t].Rows.Count; r++)
        {
            stream += "{";
            for (c = 0; c < ds.Tables[t].Columns.Count; c++)
            {
                stream += ds.Tables[t].Columns[c].ToString() + ":'" +
                          ds.Tables[t].Rows[r][c].ToString() + "',";
            }
            if (c>0)
                stream = stream.Substring(0, stream.Length - 1);
            stream += "},";
        }
        if (r>0)
            stream = stream.Substring(0, stream.Length - 1);
        stream += "],";
    }
    if (t>0)
        stream = stream.Substring(0, stream.Length - 1);
    stream += "];";
    return stream;
}
Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
Behnam Mohammadi
  • 5,808
  • 38
  • 33
4

To get an ExpandoObject:

using Newtonsoft.Json;
using Newtonsoft.Json.Converters;

Container container = JsonConvert.Deserialize<Container>(jsonAsString, new ExpandoObjectConverter());
Ryan Norbauer
  • 1,698
  • 2
  • 17
  • 24
4

For that I would use JSON.NET to do the low-level parsing of the JSON stream and then build up the object hierarchy out of instances of the ExpandoObject class.

Daniel Earwicker
  • 111,938
  • 35
  • 201
  • 280
4

Deserializing in JSON.NET can be dynamic using the JObject class, which is included in that library. My JSON string represents these classes:

public class Foo {
   public int Age {get;set;}
   public Bar Bar {get;set;}
}

public class Bar {
   public DateTime BDay {get;set;}
}

Now we deserialize the string WITHOUT referencing the above classes:

var dyn = JsonConvert.DeserializeObject<JObject>(jsonAsFooString);

JProperty propAge = dyn.Properties().FirstOrDefault(i=>i.Name == "Age");
if(propAge != null) {
    int age = int.Parse(propAge.Value.ToString());
    Console.WriteLine("age=" + age);
}

//or as a one-liner:
int myage = int.Parse(dyn.Properties().First(i=>i.Name == "Age").Value.ToString());

Or if you want to go deeper:

var propBar = dyn.Properties().FirstOrDefault(i=>i.Name == "Bar");
if(propBar != null) {
    JObject o = (JObject)propBar.First();
    var propBDay = o.Properties().FirstOrDefault (i => i.Name=="BDay");
    if(propBDay != null) {
        DateTime bday = DateTime.Parse(propBDay.Value.ToString());
        Console.WriteLine("birthday=" + bday.ToString("MM/dd/yyyy"));
    }
}

//or as a one-liner:
DateTime mybday = DateTime.Parse(((JObject)dyn.Properties().First(i=>i.Name == "Bar").First()).Properties().First(i=>i.Name == "BDay").Value.ToString());

See post for a complete example.

Chad Kuehn
  • 3,801
  • 33
  • 30
  • This approach allow to "traverse" the jSON document, so that you can manage situation where the JSON structure is unknown or variable (for example, many API returns a completely different JSON document when an error occurs). There are other libraries that permits to do that, apart from Newtonsoft.JSON (aka JSON.NET) ? – Alex 75 Dec 20 '19 at 22:42
4

The object you want DynamicJSONObject is included in the System.Web.Helpers.dll from the ASP.NET Web Pages package, which is part of WebMatrix.

Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
Nick Daniels
  • 902
  • 8
  • 13
4

How to parse easy JSON content with dynamic & JavaScriptSerializer

Please add reference of System.Web.Extensions and add this namespace using System.Web.Script.Serialization; at top:

public static void EasyJson()
{
    var jsonText = @"{
        ""some_number"": 108.541,
        ""date_time"": ""2011-04-13T15:34:09Z"",
        ""serial_number"": ""SN1234""
    }";

    var jss = new JavaScriptSerializer();
    var dict = jss.Deserialize<dynamic>(jsonText);

    Console.WriteLine(dict["some_number"]);
    Console.ReadLine();
}

How to parse nested & complex json with dynamic & JavaScriptSerializer

Please add reference of System.Web.Extensions and add this namespace using System.Web.Script.Serialization; at top:

public static void ComplexJson()
{
    var jsonText = @"{
        ""some_number"": 108.541,
        ""date_time"": ""2011-04-13T15:34:09Z"",
        ""serial_number"": ""SN1234"",
        ""more_data"": {
            ""field1"": 1.0,
            ""field2"": ""hello""
        }
    }";

    var jss = new JavaScriptSerializer();
    var dict = jss.Deserialize<dynamic>(jsonText);

    Console.WriteLine(dict["some_number"]);
    Console.WriteLine(dict["more_data"]["field2"]);
    Console.ReadLine();
}
Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
Mist
  • 658
  • 7
  • 27
4

There is a lightweight JSON library for C# called SimpleJson.

It supports .NET 3.5+, Silverlight and Windows Phone 7.

It supports dynamic for .NET 4.0

It can also be installed as a NuGet package

Install-Package SimpleJson
Peter Mortensen
  • 30,030
  • 21
  • 100
  • 124
prabir
  • 7,484
  • 4
  • 29
  • 43
4

I want to do this programmatically in unit tests, I do have the luxury of typing it out.

My solution is:

var dict = JsonConvert.DeserializeObject<ExpandoObject>(json) as IDictionary<string, object>;

Now I can assert that

dict.ContainsKey("ExpectedProperty");
3

With Cinchoo ETL - an open source library available to parse JSON into a dynamic object:

string json = @"{
    ""key1"": [
        {
            ""action"": ""open"",
            ""timestamp"": ""2018-09-05 20:46:00"",
            ""url"": null,
            ""ip"": ""66.102.6.98""
        }
    ]
}";
using (var p = ChoJSONReader.LoadText(json)
    .WithJSONPath("$..key1")
    )
{
    foreach (var rec in p)
    {
        Console.WriteLine("Action: " + rec.action);
        Console.WriteLine("Timestamp: " + rec.timestamp);
        Console.WriteLine("URL: " + rec.url);
        Console.WriteLine("IP address: " + rec.ip);
    }
}

Output:

Action: open
Timestamp: 2018-09-05 20:46:00
URL: http://www.google.com
IP address: 66.102.6.98

Sample fiddle: https://dotnetfiddle.net/S0ehSV

For more information, please visit codeproject articles

Disclaimer: I'm the author of this library.

Cinchoo
  • 5,777
  • 2
  • 17
  • 33
2

try this way!

JSON example:

[{
    "id": 140,
    "group": 1,
    "text": "xxx",
    "creation_date": 123456,
    "created_by": "xxx@gmail.co",
    "tags": ["xxxxx"]
  }, {
    "id": 141,
    "group": 1,
    "text": "xxxx",
    "creation_date": 123456,
    "created_by": "xxx@gmail.com",
    "tags": ["xxxxx"]
}]

C# code:

var jsonString = (File.ReadAllText(Path.Combine(Directory.GetCurrentDirectory(),"delete_result.json")));
var objects = JsonConvert.DeserializeObject<dynamic>(jsonString);
foreach(var o in objects)
{
    Console.WriteLine($"{o.id.ToString()}");
}
np_6
  • 504
  • 1
  • 6
  • 19
LoremIpsum
  • 1,256
  • 13
  • 24
0

If you want to skip creating a class while deserializing JSON, you can do it using NewtonSoft.Json's DeserializeAnonymousType method.

The below example can even deserialize JSON to a list of anonymous objects.

var json = System.IO.File.ReadAllText(@"C:\TestJSONFiles\yourJSONFile.json");
var fooDefinition = new { a = "", b = 0 }; // type with fields of string, int
var fooListDefinition = Enumerable.Range(0, 0).Select(e => fooDefinition).ToList();

var foos = JsonConvert.DeserializeAnonymousType(json, fooListDefinition);
Ash K
  • 851
  • 9
  • 20