Як повернути чистий JSON з послуги WCF?


233

Я намагаюся повернути трохи JSON з послуги WCF. Ця послуга просто повертає деякий вміст з моєї бази даних. Я можу отримати дані. Однак мене хвилює формат мого JSON. В даний час повернення JSON відформатоване так:

{"d":"[{\"Age\":35,\"FirstName\":\"Peyton\",\"LastName\":\"Manning\"},{\"Age\":31,\"FirstName\":\"Drew\",\"LastName\":\"Brees\"},{\"Age\":29,\"FirstName\":\"Tony\",\"LastName\":\"Romo\"}]"} 

Насправді я хотів би, щоб мій JSON був форматований максимально чисто. Я вважаю (я можу помилятися), що той самий збірник результатів, представлений у чистому JSON, повинен виглядати так:

[{
  "Age": 35,
  "FirstName": "Peyton",
  "LastName": "Manning"
}, {
  "Age": 31,
  "FirstName": "Drew",
  "LastName": "Brees"
}, {
  "Age": 29,
  "FirstName": "Tony",
  "LastName": "Romo"
}]

Я поняття не маю, звідки походить "d". У мене також немає поняття, чому символи втечі вставляються. Моя сутність виглядає наступним чином:

[DataContract]
public class Person
{
    [DataMember]
    public string FirstName { get; set; }

    [DataMember]
    public string LastName { get; set; }

    [DataMember]
    public int Age { get; set; }

    public Person(string firstName, string lastName, int age)
    {
        this.FirstName = firstName;
        this.LastName = lastName;
        this.Age = age;
    }
}

Служба, яка відповідає за повернення вмісту, визначається як:

[ServiceContract(Namespace = "")]
[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)]
public class TestService
{
    [OperationContract]
    [WebGet(ResponseFormat = WebMessageFormat.Json)]
    public string GetResults()
    {
        List<Person> results = new List<Person>();
        results.Add(new Person("Peyton", "Manning", 35));
        results.Add(new Person("Drew", "Brees", 31));
        results.Add(new Person("Tony", "Romo", 29));

        // Serialize the results as JSON
        DataContractJsonSerializer serializer = new DataContractJsonSerializer(results.GetType());
        MemoryStream memoryStream = new MemoryStream();
        serializer.WriteObject(memoryStream, results);

        // Return the results serialized as JSON
        string json = Encoding.Default.GetString(memoryStream.ToArray());
        return json;
    }
}

Як повернути «чистий» JSON з послуги WCF? Дякую!


SOAP повинен повернути XML. Ви можете використовувати кінцеву точку REST, щоб повернути JSON. Погляньте stackoverflow.com/questions/186631/…
Акіра Ямамото

4
До речі, якщо хтось інший стикається з цим і замислюється, чому властивість "d" є, це там, щоб виправити вразливість JSON . Видалення його знову зробить вас вразливими.
Олексій

4
@Alex - що вразливість залежить від перевизначення об’єкта Array, що вже не можливо в сучасних браузерах. Див stackoverflow.com/questions/16289894 / ...
Cheeso

Добре. :) Половина моєї відповіді все-таки правдива - МАЄТЕ там виправити цю вразливість.
Олексій

Відповіді:


213

Змініть тип повернення GetResults, який буде List<Person>.
Усуньте код, який ви використовуєте для серіалізації списку до рядка json - WCF робить це для вас автоматично.

Використовуючи ваше визначення для класу Person, цей код працює для мене:

public List<Person> GetPlayers()
{
    List<Person> players = new List<Person>();
    players.Add(new  Person { FirstName="Peyton", LastName="Manning", Age=35 } );
    players.Add(new  Person { FirstName="Drew", LastName="Brees", Age=31 } );
    players.Add(new  Person { FirstName="Brett", LastName="Favre", Age=58 } );

    return players;
}

результати:

[{"Age":35,"FirstName":"Peyton","LastName":"Manning"},  
 {"Age":31,"FirstName":"Drew","LastName":"Brees"},  
 {"Age":58,"FirstName":"Brett","LastName":"Favre"}]

(Усі в одному рядку)

Я також використовував цей атрибут у методі:

[WebInvoke(Method = "GET",
           RequestFormat = WebMessageFormat.Json,
           ResponseFormat = WebMessageFormat.Json,
           UriTemplate = "players")]

WebInvoke with Method = "GET" - це те саме, що і WebGet, але оскільки деякі з моїх методів є POST, я використовую всі WebInvoke для послідовності.

UriTemplate встановлює URL-адресу, за якою доступний метод. Тож я можу зробити GET http://myserver/myvdir/JsonService.svc/playersі це просто працює.

Також перегляньте IIRF або інший переписувач URL-адрес, щоб позбутися .svc в URI.


Чесо - я спробував такий підхід, перш ніж опублікував це питання. Коли я використовую цей підхід, я отримую помилку, яка говорить: "Кінцеві точки, що використовують" UriTemplate ", не можна використовувати із" System.ServiceModel.Description.WebScriptEnablingBehavior "." Що я роблю неправильно? Дякую!
користувач208662

28
використовуйте <webHttp /> замість <webScriptEnablingBehavior /> у вашому файлі .config.
Cheeso

9
Гаразд, я замінив <enableWebScript /> на <webHttp />, і це спрацювало.
MGOwen

3
MGowen - FYI, найкраще ставити, коли задавати нове запитання - це відкрити нове запитання, а не публікувати питання як коментар до старої відповіді.
Cheeso

5
Фавр бачить, що ти там робив.
ruffin

93

Якщо ви хочете приємний json без атрибутів жорсткого кодування в свої сервісні класи,

використовувати <webHttp defaultOutgoingResponseFormat="Json"/>у своїй конфігурації поведінки


28

Це робиться в web.config для вашого веб-сервісу. Встановіть прив'язку поведінки до <webHttp>, і ви побачите чистий JSON. Додатковий "[d]" задається типовою поведінкою, яку потрібно перезаписати.

Додатково дивіться цей поштовий блог: http://blog.clauskonrad.net/2010/11/how-to-expose-json-endpoint-from-wcf.html


8

Я зіткнувся з тією ж проблемою і вирішив її, змінивши значення attribut BodyStyle на "WebMessageBodyStyle.Bare":

[OperationContract]
[WebGet(BodyStyle = WebMessageBodyStyle.Bare, RequestFormat = WebMessageFormat.Json,
        ResponseFormat = WebMessageFormat.Json, UriTemplate = "GetProjectWithGeocodings/{projectId}")]
GeoCod_Project GetProjectWithGeocodings(string projectId);

Повернутий об’єкт більше не буде загортатися.


1

Коли ви використовуєте метод GET, договір повинен бути таким.

[WebGet(UriTemplate = "/", BodyStyle = WebMessageBodyStyle.Bare, ResponseFormat = WebMessageFormat.Json)]
List<User> Get();

з цим у нас є json без параметра завантаження

Альдо Флорес @alduar http://alduar.blogspot.com


1

У свій IServece.cs додайте такий тег: BodyStyle = WebMessageBodyStyle.Bare

 [WebInvoke(Method = "GET", ResponseFormat = WebMessageFormat.Json, BodyStyle = WebMessageBodyStyle.Bare, UriTemplate = "Getperson/{id}")]

    List<personClass> Getperson(string id);

Ви можете також пояснити, чому BodyStyle може впливати на результат?
MBH
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.