1

我正在从 Winforms 客户端成功调用 Web API 应用程序中的 POST 方法,该客户端为存储过程传递一些参数。

不过,如果可能的话,我更愿意通过 FromBody 功能将存储过程的结果(我必须首先在客户端上运行)传递给 POST 方法。

通过网络发送大量数据,但我现在这样做的方式是我必须运行 SP 两次 - 首先在客户端 Winforms 应用程序上,然后在 Web API 服务器应用程序上,并同时调用此 SP似乎有时会引起一些问题。

因此,如果可行,我想通过“FromBody”发送 DataTable,或者如果更可取,发送数据的 XML 化或 json 化版本(然后在另一端解压,将其转换为 html 以进行检索当调用相应的 GET 方法时。

有没有人有任何可以显示的代码?

可以在这里看到我现有的仅通过参数的代码。

更新

好的,根据 Amit Kumar Ghosh 的回答,我将代码更改为:

WebApiConfig.cs

public static class WebApiConfig
{
    public static void Register(HttpConfiguration config)
    {
        // Web API configuration and services
        // Configure Web API to use only bearer token authentication.
        config.SuppressDefaultHostAuthentication();
        config.Filters.Add(new    
HostAuthenticationFilter(OAuthDefaults.AuthenticationType));

    // Web API routes
    config.MapHttpAttributeRoutes();

    config.Routes.MapHttpRoute(
        name: "DefaultApi",
        routeTemplate: "api/{controller}/{id}",
        defaults: new { id = RouteParameter.Optional }
    );


    config.Formatters.Add(new DataTableMediaTypeFormatter());
}

public class DataTableMediaTypeFormatter : BufferedMediaTypeFormatter
{
    public DataTableMediaTypeFormatter()
        : base()
    {
        SupportedMediaTypes.Add(new System.Net.Http.Headers.MediaTypeHeaderValue("test/dt"));
    }

    public override object ReadFromStream(Type type, Stream readStream,
        HttpContent content, IFormatterLogger formatterLogger, System.Threading.CancellationToken cancellationToken)
    {
        var data = new StreamReader(readStream).ReadToEnd();
        var obj = JsonConvert.DeserializeObject<DataTable>(data);
        return obj;
    }

    public override bool CanReadType(Type type)
    {
        return true;
    }

    public override bool CanWriteType(Type type)
    {
        return true;
    }
}

控制器

[Route("{unit}/{begindate}/{enddate}/{stringifiedjsondata}")]
[HttpPost]
public void Post(string unit, string begindate, string enddate, DataTable stringifiedjsondata)
{
    DataTable dt = stringifiedjsondata;
    . . .

客户

private async Task SaveProduceUsageFileOnServer(string beginMonth, string beginYear, string endMonth, string endYear)
{
    string beginRange = String.Format("{0}{1}", beginYear, beginMonth);
    string endRange = String.Format("{0}{1}", endYear, endMonth);
    HttpClient client = new HttpClient();
    client.BaseAddress = new Uri("http://localhost:52194");
    string dataAsJson = JsonConvert.SerializeObject(_rawAndCalcdDataAmalgamatedList, Formatting.Indented);
    String uriToCall = String.Format("/api/produceusage/{0}/{1}/{2}/{3}", _unit, beginRange, endRange, @dataAsJson);
    HttpResponseMessage response = await client.PostAsync(uriToCall, null);
}

...但仍未到达控制器;具体来说,“DataTable dt = dtPassedAsJson;”中的断点 永远达不到。

实际上,它没有崩溃,这让我感到惊讶,因为正在传递一个字符串,但那里声明的数据类型是“DataTable”

更新 2

我也试过这个,在意识到它不是我从客户端传递的真正的字符串化/jsonized DataTable,而是一个字符串化/jsonized 通用列表之后:

网页 API 控制器

[Route("{unit}/{begindate}/{enddate}/{stringifiedjsondata}")]
[HttpPost]
public void Post(string unit, string begindate, string enddate, List<ProduceUsage> stringifiedjsondata)
{
    List<ProduceUsage> _produceUsageList = stringifiedjsondata;

WebApiConfig.cs

我将此添加到 Register 方法中:

config.Formatters.Add(new GenericProduceUsageListMediaTypeFormatter());

...和这个新课程:

// adapted from DataTableMediaTypeFormatter above
public class GenericProduceUsageListMediaTypeFormatter : BufferedMediaTypeFormatter
{
    public GenericProduceUsageListMediaTypeFormatter()
        : base()
    {
        SupportedMediaTypes.Add(new System.Net.Http.Headers.MediaTypeHeaderValue("test/dt"));
    }

    public override object ReadFromStream(Type type, Stream readStream,
        HttpContent content, IFormatterLogger formatterLogger, System.Threading.CancellationToken cancellationToken)
    {
        var data = new StreamReader(readStream).ReadToEnd();
        var obj = JsonConvert.DeserializeObject<List<ProduceUsage>>(data);
        return obj;
    }

    public override bool CanReadType(Type type)
    {
        return true;
    }

    public override bool CanWriteType(Type type)
    {
        return true;
    }
}

尽管如此,Controller 中的主要断点行还是:

List<ProduceUsage> _produceUsageList = stringifiedjsondata;

...未达到。

4

4 回答 4

1

或数据的 jsonized 版本(然后在另一端解压

我结束了这个 -

public class ParentController : ApiController
{
    public string Post(DataTable id)
    {
        return "hello world";
    }
}

在配置中

config.Formatters.Add(new DataTableMediaTypeFormatter());

和 -

public class DataTableMediaTypeFormatter : BufferedMediaTypeFormatter
{
    public DataTableMediaTypeFormatter()
        : base()
    {
        SupportedMediaTypes.Add(new System.Net.Http.Headers.MediaTypeHeaderValue("test/dt"));
    }

    public override object ReadFromStream(Type type, System.IO.Stream readStream,
        System.Net.Http.HttpContent content, IFormatterLogger formatterLogger, System.Threading.CancellationToken cancellationToken)
    {
        var data = new StreamReader(readStream).ReadToEnd();
        var obj = JsonConvert.DeserializeObject<DataTable>(data);
        return obj;
    }

    public override bool CanReadType(Type type)
    {
        return true;
    }

    public override bool CanWriteType(Type type)
    {
        return true;
    }
}

header of my request -
User-Agent: Fiddler
Host: localhost:60957
Content-Type : test/dt
Content-Length: 28

身体 -

[{"Name":"Amit","Age":"27"}]
于 2016-01-06T06:31:55.653 回答
1

虽然代码现在已被取代,但我以前做过一次,所以我只能从我的 TFS 历史中获取零碎的信息。

从我的控制台应用程序中,我将发布数据(是一个我转换为 POCO 的 DataTable),如下所示;

            using (HttpClient httpClient = new HttpClient())
            {
                MyDataType data = BogusMethodToPopulateData();

                httpClient.BaseAddress = new Uri(Properties.Settings.Default.MyService);
                httpClient.DefaultRequestHeaders.Accept.Clear();
                httpClient.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));

                HttpResponseMessage response;

                // Add reference to System.Net.Http.Formatting.dll
                response = await httpClient.PostAsJsonAsync("api/revise", data);

                if (response.IsSuccessStatusCode)
                {
                    Console.WriteLine("File generation process completed successfully.");
                }
            }

在服务器端,我有以下内容。这里的概念主要基于链接帖子的发送复杂类型部分。我知道您专门在查看 DataTables,但我相信您可能会弄乱这些示例或将数据提取到 POCO 中;

    // https://damienbod.wordpress.com/2014/08/22/web-api-2-exploring-parameter-binding/
    // http://www.asp.net/web-api/overview/advanced/sending-html-form-data,-part-1
    // http://www.asp.net/web-api/overview/formats-and-model-binding/parameter-binding-in-aspnet-web-api
    [POST("revise")]
    public IEnumerable<Revised_Data> Revise(MyDataType data)
    {
        if (ModelState.IsValid && data != null)
        {
            return ProcessData(data.year, data.period, data.DataToProcess).AsEnumerable();
        }
        return null;
    }
于 2016-01-06T07:41:50.453 回答
1

客户端实际上是在传递一个json形式的数据表,然后webapi运行时根据特殊的媒体类型在服务器端再次将json转换为数据表。

于 2016-01-06T18:39:49.333 回答
0

在此处查看 Hernan Guzman 的回答。

基本上,您必须将“[FromBody]”添加到服务器上的方法中,然后从客户端传递数据,将其添加到 URL 参数之后。

于 2016-01-21T00:39:55.120 回答