Jak zwrócić czysty JSON z usługi WCF?


233

Usiłuję zwrócić trochę JSON z usługi WCF. Ta usługa po prostu zwraca część treści z mojej bazy danych. Mogę zdobyć dane. Jestem jednak zaniepokojony formatem mojego JSON. Obecnie zwracany JSON jest sformatowany w następujący sposób:

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

W rzeczywistości chciałbym, aby mój JSON został sformatowany tak czysto, jak to możliwe. Uważam (mogę się mylić), że ta sama kolekcja wyników, reprezentowana w czystym JSON, powinna wyglądać tak:

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

Nie mam pojęcia, skąd pochodzi „d”. Nie mam też pojęcia, dlaczego wstawiane są znaki specjalne. Moja jednostka wygląda następująco:

[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;
    }
}

Usługa odpowiedzialna za zwrot treści jest zdefiniowana jako:

[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;
    }
}

Jak zwrócić „czysty” JSON z usługi WCF? Dziękuję Ci!


SOAP powinien zwrócić XML. Możesz użyć punktu końcowego REST, aby zwrócić JSON. Spójrz stackoverflow.com/questions/186631/…
Akira Yamamoto

4
Nawiasem mówiąc, jeśli ktokolwiek zetknie się z tym i zastanawia się, dlaczego istnieje właściwość „d”, jest ona w stanie naprawić lukę JSON . Usunięcie go powoduje, że jesteś ponownie wrażliwy.
Alex

4
@Alex - ta usterka zależy od zmiany definicji obiektu Array, co nie jest już możliwe w nowoczesnych przeglądarkach. Zobacz stackoverflow.com/questions/16289894/…
Cheeso

Dobre. :) Połowa mojej odpowiedzi jest jednak nadal prawdą - to było tam, aby naprawić tę lukę.
Alex

Odpowiedzi:


213

Zmień typ zwracany przez GetResults na List<Person>.
Wyeliminuj kod, którego używasz do serializacji Listy do ciągu json - WCF robi to automatycznie.

Korzystając z definicji dla klasy Person, ten kod działa dla mnie:

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;
}

wyniki:

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

(Wszystko w jednym wierszu)

Użyłem również tego atrybutu w metodzie:

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

WebInvoke with Method = „GET” jest taki sam jak WebGet, ale ponieważ niektóre z moich metod to POST, używam wszystkich WebInvoke dla zachowania spójności.

UriTemplate ustawia adres URL, pod którym metoda jest dostępna. Więc mogę zrobić GET http://myserver/myvdir/JsonService.svc/playersi to po prostu działa.

Sprawdź także IIRF lub inny moduł przepisujący adresy URL, aby pozbyć się pliku .svc w URI.


Cheeso - wypróbowałem to podejście, zanim opublikowałem to pytanie. Gdy korzystam z tego podejścia, pojawia się komunikat o błędzie: „Punktów końcowych używających„ UriTemplate ”nie można używać z„ System.ServiceModel.Description.WebScriptEnablingBehavior ”.” Co ja robię źle? Dziękuję Ci!
user208662,

28
użyj <webHttp /> zamiast <webScriptEnablingBehavior /> w pliku .config.
Cheeso,

9
OK, ja zastąpić <enableWebScript /> z <webHttp /> i to działało.
MGOwen

3
MGowen - FYI, najlepiej postawić nowe pytanie, aby ... otworzyć nowe pytanie, zamiast zamieszczać je jako komentarz do starej odpowiedzi.
Cheeso

5
Favre widzi, co tam zrobiłeś.
ruffin

93

Jeśli chcesz mieć ładne json bez twardych atrybutów w swoich klasach usług,

użyj <webHttp defaultOutgoingResponseFormat="Json"/>w konfiguracji zachowania



8

Napotkałem ten sam problem i rozwiązałem go, zmieniając wartość atrybutu BodyStyle na „WebMessageBodyStyle.Bare”:

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

Zwrócony obiekt nie będzie już opakowany.


1

W przypadku korzystania z metody GET umowa musi być taka.

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

z tym mamy json bez parametru rozruchu

Aldo Flores @alduar http://alduar.blogspot.com


1

W swoim pliku IServece.cs dodaj następujący znacznik: BodyStyle = WebMessageBodyStyle.Bare

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

    List<personClass> Getperson(string id);

czy możesz również wyjaśnić, dlaczego BodyStyle może wpływać na wynik?
MBH
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.