ฉันจะส่งคืน clean 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" มาจากไหน ฉันยังไม่มีเงื่อนงำว่าทำไมตัวอักขระ escape จึงถูกแทรก เอนทิตีของฉันมีลักษณะดังนี้:

[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 endpoint เพื่อส่งคืน JSON ลองดูstackoverflow.com/questions/186631/…
Akira Yamamoto

4
โดยวิธีการที่ถ้าคนอื่นมาในนี้และสงสัยว่าทำไมคุณสมบัติ "D" อยู่ที่นั่นก็มีการแพทช์ช่องโหว่ JSON การลบจะทำให้คุณเสี่ยงอีกครั้ง
อเล็กซ์

4
@Alex - ช่องโหว่นั้นขึ้นอยู่กับการกำหนด Array object ใหม่ซึ่งไม่สามารถทำได้ในเบราว์เซอร์รุ่นใหม่ ดู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 rewriter อื่นเพื่อกำจัด. svc ใน URI


Cheeso - ฉันลองวิธีนี้ก่อนโพสต์คำถามนี้ เมื่อฉันใช้วิธีการนี้ฉันได้รับข้อผิดพลาดที่ระบุว่า "ปลายทางที่ใช้ 'UriTemplate' ไม่สามารถใช้กับ 'System.ServiceModel.Description.WebScriptEnablingBehavior' ได้ ผมทำอะไรผิดหรือเปล่า? ขอบคุณ!
user208662

28
ใช้ <webHttp /> แทน <webScriptEnablingBehavior /> ในไฟล์. config ของคุณ
Cheeso

9
ตกลงฉันแทนที่<enableWebScript />ด้วย<webHttp />และใช้งานได้
MGOwen

3
MGowen - FYI ทางออกที่ดีที่สุดเมื่อถามคำถามใหม่คือ ... เปิดคำถามใหม่แทนที่จะโพสต์คำถามเป็นความคิดเห็นในคำตอบเก่า
Cheeso

5
Favre เห็นสิ่งที่คุณทำที่นั่น
ruffin

93

หากคุณต้องการ json ที่ดีโดยไม่มีคุณลักษณะการเข้ารหัสในคลาสบริการของคุณ

ใช้<webHttp defaultOutgoingResponseFormat="Json"/>ในการกำหนดค่าพฤติกรรมของคุณ


28

สิ่งนี้สามารถทำได้ใน web.config สำหรับเว็บเซอร์ของคุณ ตั้งค่า bindingBehavior เป็น <webHttp> แล้วคุณจะเห็น JSON สะอาด พิเศษ "[d]" ถูกกำหนดโดยพฤติกรรมเริ่มต้นที่คุณต้องเขียนทับ

ดูเพิ่มเติมในบล็อกนี้: http://blog.clauskonrad.net/2010/11/how-to-expose-json-endpoint-from-wcf.html


8

ฉันประสบปัญหาเดียวกันและแก้ไขได้โดยการเปลี่ยนค่าของ 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 โดยไม่ต้องพารามิเตอร์การบูต

Aldo Flores @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
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.