MVC3 - отправка байтового массива в контроллер - база данных RowVersion - PullRequest
25 голосов
/ 20 июля 2011

Я работаю над приложением MVC3. Моя клиентская ViewModel содержит свойство RowVersion SQL Server, которое является байтом []. Он отображается как массив объектов на стороне клиента. Когда я пытаюсь опубликовать свою модель представления в контроллере, свойство RowVersion всегда имеет значение null.

Я предполагаю, что сериализатор Controller (JsonValueProviderFactory) игнорирует свойство массива Object.

Я видел этот блог, но это не относится, так как я публикую JSON, а не разметку формы: http://thedatafarm.com/blog/data-access/round-tripping-a-timestamp-field-with-ef4-1-code-first-and-mvc-3/

Мой вид отображает мою модель вида следующим образом:

<script type="text/javascript">
  var viewModel = @Html.Raw( Json.Encode( this.Model ) );
</script>

Затем я отправляю viewModel в контроллер следующим образом:

    var data = {
        'contact': viewModel
    };

    $.ajax({
        type: 'POST',
        url: '/Contact/Save',
        contentType: "application/json; charset=utf-8",
        data: JSON.stringify(data),
        dataType: 'json',
        success: function (data) {
            // Success
        },
        error: function (XMLHttpRequest, textStatus, errorThrown) {
            alert(XMLHttpRequest.responseText);
        }
    });

Вот мое действие в контроллере:

  [HttpPost]
  public JsonResult Save(Contact contact) {
     return this.Json( this._contactService.Save( contact ) );
  }

ОБНОВЛЕНИЕ: на основании ответа Дарина.

Я надеялся на более чистое решение, но поскольку Дарин дал единственный ответ, мне нужно будет добавить пользовательское свойство, которое будет сериализовать мое свойство byte [] "row_version" в строку Base64. И когда для строки Base64 установлено новое пользовательское свойство, она преобразует строку обратно в байт []. Ниже приведено пользовательское свойство "RowVersion", которое я добавил в свою модель:

  public byte[] row_version {
     get;
     set;
  }

  public string RowVersion {
     get {

        if( this.row_version != null )
           return Convert.ToBase64String( this.row_version );

        return string.Empty;
     }
     set {

        if( string.IsNullOrEmpty( value ) )
           this.row_version = null;
        else
           this.row_version = Convert.FromBase64String( value );
     }
  }

Ответы [ 2 ]

37 голосов
/ 20 июля 2011

Моя клиентская модель ViewModel содержит свойство RowVersion SQL Server, которое является байтом []

Сделайте так, чтобы вместо byte[] ваша модель представления содержала свойство string, которое является base64 представлением этого byte[]. Тогда у вас не возникнет проблем с переадресацией его на клиент и обратно на сервер, где вы сможете получить исходный byte[] из строки Base64.

1 голос
/ 10 апреля 2014

Json.NET автоматически кодирует байтовые массивы как Base64.

Вы можете использовать JsonNetResult вместо JsonResult:

от https://gist.github.com/DavidDeSloovere/5689824:

using System;
using System.Web;
using System.Web.Mvc;
using Newtonsoft.Json;
using Newtonsoft.Json.Serialization;

public class JsonNetResult : JsonResult
{
    public override void ExecuteResult(ControllerContext context)
    {
        if (context == null)
        {
            throw new ArgumentNullException("context");
        }

        var response = context.HttpContext.Response;

        response.ContentType = !string.IsNullOrEmpty(this.ContentType) ? this.ContentType : "application/json";

        if (this.ContentEncoding != null)
        {
            response.ContentEncoding = this.ContentEncoding;
        }

        if (this.Data == null)
        {
            return;
        }

        var jsonSerializerSettings = new JsonSerializerSettings();
        jsonSerializerSettings.DateFormatHandling = DateFormatHandling.IsoDateFormat;
        jsonSerializerSettings.ContractResolver = new CamelCasePropertyNamesContractResolver();
        var formatting = HttpContext.Current != null && HttpContext.Current.IsDebuggingEnabled ? Formatting.Indented : Formatting.None;
        var serializedObject = JsonConvert.SerializeObject(Data, formatting, jsonSerializerSettings);
        response.Write(serializedObject);
    }
}

Использование:

[HttpPost]
public JsonResult Save(Contact contact) {
    return new JsonNetResult { Data = _contactService.Save(contact) };
}
...