วิธีการโพสต์ JSON ไปยังเซิร์ฟเวอร์โดยใช้ C #?


269

นี่คือรหัสที่ฉันใช้:

// create a request
HttpWebRequest request = (HttpWebRequest)
WebRequest.Create(url); request.KeepAlive = false;
request.ProtocolVersion = HttpVersion.Version10;
request.Method = "POST";


// turn our request string into a byte stream
byte[] postBytes = Encoding.UTF8.GetBytes(json);

// this is important - make sure you specify type this way
request.ContentType = "application/json; charset=UTF-8";
request.Accept = "application/json";
request.ContentLength = postBytes.Length;
request.CookieContainer = Cookies;
request.UserAgent = currentUserAgent;
Stream requestStream = request.GetRequestStream();

// now send it
requestStream.Write(postBytes, 0, postBytes.Length);
requestStream.Close();

// grab te response and print it out to the console along with the status code
HttpWebResponse response = (HttpWebResponse)request.GetResponse();
string result;
using (StreamReader rdr = new StreamReader(response.GetResponseStream()))
{
    result = rdr.ReadToEnd();
}

return result;

เมื่อฉันเรียกใช้ฉันจะได้รับข้อผิดพลาดเซิร์ฟเวอร์ภายใน 500 ครั้ง

ผมทำอะไรผิดหรือเปล่า?


1
ขั้นแรกตรวจสอบให้แน่ใจว่าข้อมูลที่คุณโพสต์เป็นสิ่งที่เซิร์ฟเวอร์คาดหวัง
LB

ที่จริงมันดูเหมือนว่าฉันได้รับการโพสต์ข้อมูลที่ไม่ถูกต้อง ...
Arsen Zahray

เพื่อความสะดวกในการทำงานคุณสามารถเพิ่มห้องสมุดjsonในสตูดิโอภาพของคุณได้ด้วย
Alireza Tabatabaeian

@Arsen - เซิร์ฟเวอร์ไม่ควรผิดพลาดกับข้อมูลที่มีรูปแบบไม่ถูกต้อง ยื่นรายงานข้อผิดพลาด
jww

คำตอบ:


396

วิธีที่ฉันทำและทำงานคือ:

var httpWebRequest = (HttpWebRequest)WebRequest.Create("http://url");
httpWebRequest.ContentType = "application/json";
httpWebRequest.Method = "POST";

using (var streamWriter = new StreamWriter(httpWebRequest.GetRequestStream()))
{
    string json = "{\"user\":\"test\"," +
                  "\"password\":\"bla\"}";

    streamWriter.Write(json);
}

var httpResponse = (HttpWebResponse)httpWebRequest.GetResponse();
using (var streamReader = new StreamReader(httpResponse.GetResponseStream()))
{
    var result = streamReader.ReadToEnd();
}

ฉันเขียนไลบรารีเพื่อทำงานนี้ด้วยวิธีที่ง่ายกว่าอยู่ที่นี่: https://github.com/ademargomes/JsonRequest

หวังว่ามันจะช่วย


3
ฉันคิดว่าบรรทัดสตริงของ json ควรเป็น: string json = "{\" user \ ": \" ทดสอบ \ "," + "\" รหัสผ่าน \ ": \" bla \ "}"; ดูเหมือนว่าคุณกำลังหายไป \
ดรีมเลน

3
ใช้ "application / json" เสมอ (เว้นแต่มีเหตุผลอื่นที่จำเป็นต้องใช้ text / json ตัวอย่างเช่น: entwicklungsgedanken.de/2008/06/06/… ) การเชื่อถือไปที่: stackoverflow.com/questions/477816/… .
Yaniv

34
ฉันคิดว่า streamWriter.Flush (); และ streamWriter.Close (); ไม่จำเป็นเนื่องจากคุณอยู่ในบล็อกที่ใช้ ในตอนท้ายของบล็อกการใช้งานตัวเขียนสตรีมจะปิดอยู่
Ruchira

1
อย่าสร้าง JSON ด้วยตนเอง มันง่ายที่จะทำผิดพลาดที่อนุญาตให้ฉีด JSON
Florian Winter

5
@ user3772108 ดูstackoverflow.com/a/16380064/2279059 ใช้ไลบรารี JSON เช่น Newtonsoft JSON.Net และสร้างสตริง JSON จากวัตถุหรือใช้การทำให้เป็นอนุกรม ฉันเข้าใจว่าสิ่งนี้ถูกตัดออกที่นี่เพื่อความเรียบง่าย (แม้ว่าความเรียบง่ายจะมีน้อยมาก) แต่การจัดรูปแบบสตริงข้อมูลที่มีโครงสร้าง (JSON, XML, ... ) เป็นอันตรายเกินไปที่จะทำแม้ในสถานการณ์ที่ไม่สำคัญ .
Florian Winter

149

วิธีการแก้ปัญหาของ Ademar สามารถปรับปรุงโดยการใช้ประโยชน์JavaScriptSerializerของSerializeวิธีการที่จะให้การแปลงโดยนัยของวัตถุที่จะ JSON

นอกจากนี้ก็เป็นไปได้ที่จะใช้ประโยชน์จากusingฟังก์ชั่นเริ่มต้นคำสั่งในการสั่งซื้อที่จะละเว้นอย่างชัดเจนโทรและFlushClose

var httpWebRequest = (HttpWebRequest)WebRequest.Create("http://url");
httpWebRequest.ContentType = "application/json";
httpWebRequest.Method = "POST";

using (var streamWriter = new StreamWriter(httpWebRequest.GetRequestStream()))
{
    string json = new JavaScriptSerializer().Serialize(new
                {
                    user = "Foo",
                    password = "Baz"
                });

    streamWriter.Write(json);
}

var httpResponse = (HttpWebResponse)httpWebRequest.GetResponse();
using (var streamReader = new StreamReader(httpResponse.GetResponseStream()))
{
    var result = streamReader.ReadToEnd();
}

1
อะไรคือความแตกต่างระหว่างรหัสนี้กับโค้ดด้านบนฉันทำอะไรหายไปบ้าง
JMK

16
สิ่งนี้ใช้วิธีการทำให้เป็นอนุกรมของ JavaScriptSerializer เพื่อสร้าง JSON ที่ถูกต้องแทนการสร้างมันด้วยมือ
ฌอนเดอร์สัน

ดูคำตอบของ Jean F ด้านล่าง - ควรเป็นความคิดเห็น ใช้ความระมัดระวังกับประเภทเนื้อหาที่application/jsonถูกต้อง
ลูคัส

@SeanAnderson ฉันมีข้อผิดพลาด "ไม่สามารถเชื่อมต่อกับเซิร์ฟเวอร์ระยะไกล" ได้
ralphgabb

3
@LuzanBaral คุณเพียงแค่ต้องการการชุมนุม: System.Web.Extensions
Norbrecht

60

HttpClientประเภทคือการดำเนินงานที่ใหม่กว่าและWebClientHttpWebRequest

คุณสามารถใช้บรรทัดต่อไปนี้

string myJson = "{'Username': 'myusername','Password':'pass'}";
using (var client = new HttpClient())
{
    var response = await client.PostAsync(
        "http://yourUrl", 
         new StringContent(myJson, Encoding.UTF8, "application/json"));
}

ป้อนคำอธิบายรูปภาพที่นี่

เมื่อคุณต้องการของคุณHttpClientมากกว่าหนึ่งครั้งก็แนะนำให้เพียง HttpClientFactoryแต่สร้างหนึ่งตัวอย่างและนำมาใช้หรือใช้ใหม่


5
ข้อควรทราบเล็กน้อยเกี่ยวกับ HttpClient ความเห็นทั่วไปคือคุณไม่ควรทิ้ง แม้จะใช้ IDisposable วัตถุเป็น Thread-Safe และตั้งใจที่จะนำมาใช้ใหม่ stackoverflow.com/questions/15705092/…
Jean F.

1
@JeanF เฮ้ขอบคุณสำหรับการเข้า ขณะที่ผมได้ตั้งข้อสังเกต allready HttpClientFactoryคุณควรสร้างอินสแตนซ์หรือใช้ ฉันไม่ได้อ่านคำตอบทั้งหมดในปัญหาที่เชื่อมโยง แต่ฉันคิดว่าจำเป็นต้องอัปเดตเนื่องจากไม่ได้กล่าวถึงโรงงาน
NtFreX

33

นอกเหนือจากการโพสต์ของฌอนแล้วมันไม่จำเป็นที่จะต้องซ้อนคำสั่งที่ใช้ โดยusingStreamWriter มันจะถูกฟลัชและปิดที่ส่วนท้ายของบล็อกดังนั้นจึงไม่จำเป็นต้องเรียกใช้เมธอดFlush()และClose():

var request = (HttpWebRequest)WebRequest.Create("http://url");
request.ContentType = "application/json";
request.Method = "POST";

using (var streamWriter = new StreamWriter(request.GetRequestStream()))
{
    string json = new JavaScriptSerializer().Serialize(new
                {
                    user = "Foo",
                    password = "Baz"
                });

    streamWriter.Write(json);
}

var response = (HttpWebResponse)request.GetResponse();
using (var streamReader = new StreamReader(response.GetResponseStream()))
{
        var result = streamReader.ReadToEnd();
}

1
ตอนนี้คำตอบนี้และคำตอบของ Sean Anderson ก็เหมือนกันทุกประการเหมือนกับที่ Sean ได้แก้ไขโพสต์ของเขา
faza

เฮ้นี่มันเยี่ยมมากขอบคุณ แต่เราจะส่งต่อข้อมูลอย่างไรถ้าเรามีโหนดลูกบน json ของเรา
2728409

1
serializer สามารถจัดการโหนดลูกใน json - คุณเพียงแค่ต้องให้มันเป็นวัตถุ json ที่ถูกต้อง
David Clarke

14

หากคุณต้องการโทรแบบอะซิงโครนัสให้ใช้

var request = HttpWebRequest.Create("http://www.maplegraphservices.com/tokkri/webservices/updateProfile.php?oldEmailID=" + App.currentUser.email) as HttpWebRequest;
            request.Method = "POST";
            request.ContentType = "text/json";
            request.BeginGetRequestStream(new AsyncCallback(GetRequestStreamCallback), request);

private void GetRequestStreamCallback(IAsyncResult asynchronousResult)
    {
        HttpWebRequest request = (HttpWebRequest)asynchronousResult.AsyncState;
        // End the stream request operation

        Stream postStream = request.EndGetRequestStream(asynchronousResult);


        // Create the post data
        string postData = JsonConvert.SerializeObject(edit).ToString();

        byte[] byteArray = Encoding.UTF8.GetBytes(postData);


        postStream.Write(byteArray, 0, byteArray.Length);
        postStream.Close();

        //Start the web request
        request.BeginGetResponse(new AsyncCallback(GetResponceStreamCallback), request);
    }

    void GetResponceStreamCallback(IAsyncResult callbackResult)
    {
        HttpWebRequest request = (HttpWebRequest)callbackResult.AsyncState;
        HttpWebResponse response = (HttpWebResponse)request.EndGetResponse(callbackResult);
        using (StreamReader httpWebStreamReader = new StreamReader(response.GetResponseStream()))
        {
            string result = httpWebStreamReader.ReadToEnd();
            stat.Text = result;
        }

    }

3
ขอขอบคุณที่โพสต์ Vivek โซลูชันนี้ ในสถานการณ์ของเราเราลองใช้วิธีแก้ไขปัญหาอื่นในโพสต์นี้และดูการผิดพลาดของระบบข้อยกเว้นในแอปพลิเคชันของเราเนื่องจากสิ่งที่ฉันถือว่าเป็นโพสต์ซิงโครนัสบล็อกเธรด รหัสของคุณแก้ไขปัญหาของเรา
Ken Palmer

โปรดทราบว่าคุณไม่จำเป็นต้องแปลงเป็นไบต์ คุณควรจะสามารถที่จะทำpostStream.Write(postData);- และขึ้นอยู่กับ API ที่อาจจะต้องใช้แทนrequest.ContentType = "application/json"; text/json
vapcguy


11

ฉันเพิ่งค้นพบวิธีที่ง่ายกว่าในการโพสต์ JSON ด้วยขั้นตอนเพิ่มเติมของการแปลงจากแบบจำลองในแอปของฉัน โปรดทราบว่าคุณต้องสร้างแบบจำลอง [JsonObject] เพื่อให้ตัวควบคุมของคุณรับค่าและทำการแปลง

คำขอ:

 var model = new MyModel(); 

 using (var client = new HttpClient())
 {
     var uri = new Uri("XXXXXXXXX"); 
     var json = new JavaScriptSerializer().Serialize(model);
     var stringContent = new StringContent(json, Encoding.UTF8, "application/json");
     var response = await Client.PutAsync(uri,stringContent).Result;
     ...
     ...
  }

รุ่น:

[JsonObject]
[Serializable]
public class MyModel
{
    public Decimal Value { get; set; }
    public string Project { get; set; }
    public string FilePath { get; set; }
    public string FileName { get; set; }
}

ฝั่งเซิร์ฟเวอร์:

[HttpPut]     
public async Task<HttpResponseMessage> PutApi([FromBody]MyModel model)
{
    ...
    ... 
}

6

ตัวเลือกนี้ไม่ได้กล่าวถึง:

using (var client = new HttpClient())
{
    client.BaseAddress = new Uri("http://localhost:9000/");
    client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));

    var foo = new User
    {
        user = "Foo",
        password = "Baz"
    }

    await client.PostAsJsonAsync("users/add", foo);
}

2
ตัวเลือกนี้ไม่สามารถใช้งานได้อีกต่อไปตั้งแต่. Net 4.5.2 ดูที่นี่stackoverflow.com/a/40525794/2161568
Downhillski

Downvote ตามความคิดเห็นข้างต้น - เนื่องจากยังไม่พร้อมใช้งานควรลบคำตอบ
NovaDev

1
นั่นไม่ใช่เหตุผลที่ดีในการ downvote คำตอบนี้เนื่องจากทุกคนไม่ได้ใช้. net รุ่นล่าสุดและนี่คือคำตอบที่ถูกต้อง
Ellisan

4

วิธีที่แตกต่างและสะอาดเพื่อให้บรรลุนี้คือการใช้ HttpClient ดังนี้:

public async Task<HttpResponseMessage> PostResult(string url, ResultObject resultObject)
{
    using (var client = new HttpClient())
    {
        HttpResponseMessage response = new HttpResponseMessage();
        try
        {
            response = await client.PostAsJsonAsync(url, resultObject);
        }
        catch (Exception ex)
        {
            throw ex
        }
        return response;
     }
}

4
มีประโยชน์ แต่PostAsJsonAsyncไม่สามารถใช้งานได้อีกต่อไปตั้งแต่. NET 4.5.2 ใช้PostAsyncแทน เพิ่มเติมได้ที่นี่
Zachary Keener

HttpClient โดยทั่วไปไม่ควรใช้ในusingข้อความเช่นนี้
p3tch

ฉันคิดว่ามันใช้IDisposableอินเทอร์เฟซด้วยเหตุผล
Dima Daron

4

คำเตือน! ฉันมีมุมมองที่แข็งแกร่งมากในเรื่องนี้

ไคลเอนต์เว็บที่มีอยู่ของ. NET ไม่เป็นมิตรกับผู้พัฒนา! WebRequest & WebClientเป็นตัวอย่างสำคัญของ "วิธีทำให้รอดชีวิตจากนักพัฒนา" พวกเขามีความละเอียดและซับซ้อนในการทำงานกับ; เมื่อสิ่งที่คุณต้องการทำคือการโพสต์คำของ่าย ๆ ใน C # HttpClientดำเนินการแก้ไขปัญหาเหล่านี้ได้บ้าง แต่ก็ยังสั้น ยิ่งกว่านั้นเอกสารของ Microsoft ไม่ดี…แย่จริงๆ นอกเสียจากคุณต้องการกรองหน้าและหน้าเว็บที่มีการประกาศแจ้งทางเทคนิค

โอเพ่นซอร์สเพื่อช่วยเหลือ มีตัวเลือกโอเพนซอร์สที่ยอดเยี่ยมสามไลบรารี NuGet ฟรี ขอบคุณพระเจ้า! ทั้งหมดนี้ได้รับการสนับสนุนเป็นอย่างดีจัดทำเอกสารและใช่ง่ายต่อการแก้ไข ... ใช้งานง่ายสุด ๆ

  • ServiceStack.Text - รวดเร็วสว่างและยืดหยุ่น
  • RestSharp - ไคลเอ็นต์ REST และ HTTP API แบบง่าย
  • Flurl - ไลบรารีไคลเอ็นต์ HTTP ที่คล่องแคล่วและพกพาได้และสามารถทดสอบได้

มีไม่มากระหว่างพวกเขา แต่ฉันจะให้ ServiceStack.Text ขอบเล็กน้อย ...

  • ดาว Githubนั้นเหมือนกัน
  • ปัญหาเปิดและที่สำคัญปัญหาเร่งด่วนใด ๆ ปิดตัวลง ServiceStack รับรางวัลที่นี่เพื่อการแก้ไขปัญหาที่เร็วที่สุดและไม่มีปัญหาเปิด
  • เอกสาร? ทุกคนมีเอกสารที่ยอดเยี่ยม อย่างไรก็ตาม ServiceStack นำไปสู่อีกระดับ & เป็นที่รู้จักกันในชื่อ 'มาตรฐานทองคำ' สำหรับเอกสารประกอบ

ตกลง - คำขอโพสต์ใน JSON มีลักษณะอย่างไรใน ServiceStack.Text

var response = "http://example.org/login"
    .PostJsonToUrl(new Login { Username="admin", Password="mypassword" });

นั่นคือรหัสหนึ่งบรรทัด กระชับและง่าย! เปรียบเทียบด้านบนกับไลบรารี Http ของ. NET


3

ในที่สุดฉันก็เรียกใช้โหมดซิงค์โดยรวมไฟล์. Result

HttpResponseMessage response = null;
try
{
    using (var client = new HttpClient())
    {
       response = client.PostAsync(
        "http://localhost:8000/....",
         new StringContent(myJson,Encoding.UTF8,"application/json")).Result;
    if (response.IsSuccessStatusCode)
        {
            MessageBox.Show("OK");              
        }
        else
        {
            MessageBox.Show("NOK");
        }
    }
}
catch (Exception ex)
{
    MessageBox.Show("ERROR");
}

1

var data = Encoding.ASCII.GetBytes(json);

byte[] postBytes = Encoding.UTF8.GetBytes(json);

ใช้ ASCII แทน UFT8


2
ฟังดูเหมือนความคิดที่เลวร้ายฉันหายไปบางอย่างหรือเปล่า
CyberFox

JSON สามารถมีอักขระ UTF8 ได้ดูเหมือนว่าเป็นความคิดที่แย่มาก
Adrian Smith
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.