MVC3 - отправка байтового массива в контроллер - База данных RowVersion
Я работаю над приложением MVC3. Моя ViewModel на стороне клиента содержит свойство RowVersion SQL Server, которое является байтом []. Он отображается как массив объектов на стороне клиента. Когда я пытаюсь опубликовать свою модель представления в контроллере, свойство RowVersion всегда равно null.
Я предполагаю, что сериализатор контроллера (JsonValueProviderFactory) игнорирует свойство массива объектов.
Я видел этот блог, но это не относится, поскольку я отправляю 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 ) );
}
ОБНОВЛЕНИЕ: на основе ответа Дарина.
Я надеялся на более чистое решение, но поскольку Дарин дал единственный ответ, мне нужно будет добавить настраиваемое свойство, которое будет сериализовать свойство my 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 );
}
}
Ответы
Ответ 1
Моя ViewModel на стороне клиента содержит свойство RowVersion SQL Server, которое является байтом []
Сделайте так, чтобы вместо byte[]
ваша модель представления содержала свойство string
, которое представляет собой base64 представление этого byte[]
. Тогда у вас не будет проблем с обращением к клиенту и обратно на сервер, где вы сможете получить исходный byte[]
из строки Base64.
Ответ 2
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) };
}