2017-05-16 52 views
2

我有一个使用OWIN自支持的web API,而且我似乎很难在并发连接上。当我使用50个并发获取事件进行压力测试时,响应时间正在增加A LOT,即使操作非常简单,使用在uri中键入的id返回200 OK。Web Api 2 OWIN Selfhost处理多个并发请求

如何我的网页api OWIN和api控制器每个请求运行simultanious?

这里是我的项目的一个样本:

启动类

public class Startup 
{ 
    public void Configuration(IAppBuilder appBuilder) 
    { 
     try 
     { 
      appBuilder.UseWelcomePage("/welcome.html"); 
      appBuilder.UseErrorPage(new ErrorPageOptions 
      { 
       ShowExceptionDetails = 
    #if DEBUG 
       true 
    #else 
       false 
    #endif 
      }); 

      Debug.WriteLine($"Setting Web Api Configuration up..."); 
      var apiConfig = ConfigureWebApi(); 
      appBuilder.UseWebApi(new HttpServer(apiConfig)); 
      appBuilder.UseCors(CorsOptions.AllowAll); 

      Debug.WriteLine($"Completed Web Api configuration."); 
     } 
     catch (Exception ex) 
     { 
      Debug.WriteLine($"Failed to complete Web Api setup.", ex); 
      throw; 
     } 
    } 
} 

控制器

public class TestController : ApiController 
{ 
    public async Task<IHttpActionResult> Get(int id) 
    { 
     return Task<IHttpActionResult>.Factory.StartNew(() => { 
      try 
      { 
       return Ok(id); 
      } 
      catch (Exception ex) 
      { 
       Debug.WriteLine(ex.Message, nameof(Get)); 
       return InternalServerError(); 
      } 
     }); 
    } 
} 

我的控制台Web API的启动:

public static void Open() 
{ 
    _webApp = WebApp.Start<Startup>(HostAddress); 
    Console.WriteLine($"Host open -> {HostAddress}"); 
} 
+0

你最近怎么样做压力测试?你可以发布该代码吗? – MichaelDotKnox

+0

另外,不要使用Factory.StartNew,而是返回不太可能阻塞的Task.FromResult(Ok(Id))。不知道这是否会导致你的问题。 – MichaelDotKnox

+0

使用Paessler的Webserver Stress工具来执行压力测试。思考如何创建一个简单的客户端,看看它是如何发展的。感谢与Task.FromResult提示。 – grmihel

回答

1

你的所有的API CON troller方法将同时运行,显然受托管api的服务器的功能限制。这是Web服务器的功能,无论是IIS还是自己托管。然而,你可以做些事情来帮助这一点。

如果您一直保持一切异步,您将看到最佳性能。例如,如果你的控制器方法是这样的:

public async Task<IHttpActionResult> Get(int id) 
{ 
    var testObject = await _repository.GetAsync(id); 

    return Ok(testObject); 
} 

,你有一个看起来像这样的库方法:

public async Task<TestObject> GetAsync(int id) 
{ 
    using(var connection = new SqlConnection(_connectionString) 
    { 
     // using Dapper 
     return await connection.QuerySingleAsync<TestObject>("sp_GetTestObject", new {Id = id}, commandType = CommandType.StoredProcedure); 
    } 
} 

这样做可以让计算机使用工作线程数据库I/O,释放处理能力来处理其他任务,例如接受对API端点的其他请求。像@StephenCleary这样的人可能可以阐述我说的和/或纠正它。 @ StephenCleary的blog和book绝对值得一读,以了解异步编程。

希望有所帮助。

+0

谢谢,这正是我期待的东西。似乎现在尽可能流畅地运行。 – grmihel