При сериализации объекта типа "система" обнаружена циклическая ссылка.Отображение

у меня есть asp.net метод действия контроллера MVC 3:

public JsonResult GetRecordingRates(int Id)
{            
    List<DefaultRateChart> defaultRateCharts = new List<DefaultRateChart>();
    using (IDefaultRateChartManager defaultRateChartManager = new ManagerFactory().GetDefaultRateChartManager()) {
       defaultRateCharts = defaultRateChartManager.GetAll().Where(rc => rc.Currency.Id == Id && (!rc.NumberPrefix.StartsWith("#") || rc.NumberPrefix.StartsWith("Default")) && rc.AccountCredit == "Credit").ToList();
    }
    return Json(defaultRateCharts);
}

Я хочу отправить этот список в метод успеха jQuery ajax, но я получаю 500 Внутренняя Ошибка Сервера

мой вызов ajax выглядит так:

$.ajax({
type: "POST",
dataType: "json",
url: "/Home/GetRecordingRates",
data: {
    Id: $("#hdCurrencyId").val()                 
},
success: function (data) {
        alert(data);
   }
}); 

в firebug XHR на вкладке ответа он говорит:

при сериализации объекта типа "система" обнаружена циклическая ссылка.Отображение.RuntimeModule'.

[EDIT]

I изменен метод действия следующим образом:

public JsonResult GetRecordingRates(int Id)
{
    List<DefaultRateChart> defaultRateCharts = new List<DefaultRateChart>();
    using (IDefaultRateChartManager defaultRateChartManager = new ManagerFactory().GetDefaultRateChartManager())
    {
        defaultRateCharts = defaultRateChartManager.GetAll().Where(rc => rc.Currency.Id == Id && (!rc.NumberPrefix.StartsWith("#") || rc.NumberPrefix.StartsWith("Default")) && rc.AccountCredit == "Credit").ToList();
    }

    return this.Json(
        new
        {
            Result = (from obj in defaultRateCharts select new { Id = obj.Id, DialPrefix = obj.NumberPrefix, Rate = obj.PurchaseRates })
        }
        , JsonRequestBehavior.AllowGet
     );
}

и я не получаю эту ошибку сейчас, но как я могу разобрать коллекцию в AJAX success. Я изменил успех на следующий, но он не добавляет строки в таблицу.

success: function (data) {
            var row = $('<tr>');
            for(var i = 0; i < data.length; i++) {
                row.append($('<td>').html(data[i]));
            }   

            $('#results').append(row);                                      
            jQuery('#RecordingRates').dialog({ closeOnEscape: false });
            $(".ui-dialog-titlebar").hide();
            $("#RecordingRates").dialog({ dialogClass: 'transparent' });
            $('#RecordingRates').dialog('open');
           }
    }); 

в Firebug Net => XHR=> Json он показывает следующее JSON:

[Object { Id=

1

,  DialPrefix=

"1"

,  Rate=

2.6

}, Object { Id=

3

,  DialPrefix=

"2"

,  Rate=

2.6

}, Object { Id=

5

,  DialPrefix=

"7"

,  Rate=

3.5

}, 3 more...]


0
    Object { Id=

1

, DialPrefix=

"1"

, Rate=

2.6

}

1
    Object { Id=

3

, DialPrefix=

"2"

, Rate=

2.6

}

2
    Object { Id=

5

, DialPrefix=

"7"

, Rate=

3.5

}

3
    Object { Id=

7

, DialPrefix=

"8"

, Rate=

6

}

4
    Object { Id=

9

, DialPrefix=

"Default"

, Rate=

5

}

5
    Object { Id=

15

, DialPrefix=

"Subscription"

, Rate=

15

}

2 ответов


как я могу разобрать коллекцию в AJAX success. Я изменил успех на following, но не добавляет строки в таблицу.

вы возвращаете коллекцию со следующими полями: Id, DialPrefix, Rate, а вы не используете их.

var row = $('<tr>');
for(var i = 0; i < data.length; i++) {
    row.append($('<td>').html(data[i]));
}   

$('#results').append(row);  

вы должны добавить их в свой цикл:

row.append($('<td>').html(data[i].DialPrefix));
row.append($('<td>').html(data[i].Rate));

это происходит, когда два или более объекта в вашем JSON указывают друг на друга, сериализатор не может отразить их в JSON. Вы должны использовать сериализатор Newtonsoft (здесь объясняется, как это сделать: использование JSON.NET как сериализатор JSON по умолчанию в ASP.NET MVC 3-Возможно ли это?)

есть несколько вещей, которые вы можете сделать. Первый из них-это позволить Configuration manager игнорировать или сохранять такие ссылки:

        config.Formatters.JsonFormatter.SerializerSettings.ReferenceLoopHandling = ReferenceLoopHandling.Ignore;
        config.Formatters.JsonFormatter.SerializerSettings.PreserveReferencesHandling = PreserveReferencesHandling.None;

или

вы можете аннотировать свойства, которые вы не хотите сериализовать в объектах (поэтому круговая ссылка не будет существовать). С аннотацией [JsonIgnore] вот так:

public class Order
{
    [JsonIgnore]
    public virtual ICollection<OrderLines> OrderLines{ get; set; } 
}