WCFサービスからクリーンなJSONを返すにはどうすればよいですか?


233

WCFサービスからJSONを返そうとしています。このサービスは、データベースから一部のコンテンツを返すだけです。データを取得できます。ただし、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;
    }
}

WCFサービスから「クリーン」なJSONを返すにはどうすればよいですか?ありがとうございました!


SOAPはXMLを返す必要があります。RESTエンドポイントを使用してJSONを返すことができます。ルックテイクstackoverflow.com/questions/186631/...
山本

4
ちなみに、誰かがこれに遭遇し、なぜ「d」プロパティが存在するのか疑問に思った場合、JSONの脆弱性にパッチを当てるために存在します。削除すると再び脆弱になります。
Alex

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"}]

(すべて1行で)

この属性もメソッドで使用しました:

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

Method = "GET"を指定したWebInvokeはWebGetと同じですが、一部のメソッドはPOSTであるため、一貫性を保つためにすべてのWebInvokeを使用しています。

UriTemplateは、メソッドを使用できるURLを設定します。だから私はGETを実行でき、それはうまくいきます http://myserver/myvdir/JsonService.svc/players

また、IIRFまたは別のURLリライターをチェックして、URIの.svcを削除します。


Cheeso-この質問を投稿する前に、このアプローチを試しました。このアプローチを使用すると、「 'UriTemplate'を使用するエンドポイントは 'System.ServiceModel.Description.WebScriptEnablingBehavior'では使用できません」というエラーが表示されます。何が悪いのですか?ありがとうございました!
user208662 '19

28
.configファイルで<webScriptEnablingBehavior />ではなく<webHttp />を使用します。
Cheeso

9
OK、<enableWebScript /><webHttp />に置き換えたところ、問題なく動作しました。
MGOwen 2010年

3
MGowen-参考までに、新しい質問をするときに最善の策は、古い質問に対するコメントとして質問を投稿するのではなく、新しい質問を開くことです。
Cheeso

5
Favreはあなたがそこで何をしたかを見ます。
ruffin

93

サービスクラスに属性をハードコーディングせずに素敵なjsonが必要な場合は、

<webHttp defaultOutgoingResponseFormat="Json"/>あなたの行動設定で使用する


28

これは、Webサービスの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 2016
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.