MongoClient的生命周期应该如何工作?

时间:2015-10-09 14:22:01

标签: c# asp.net mongodb asp.net-mvc-4 c#-4.0

我有一个使用MongoDB作为数据库的ASP.Net MVC应用程序。网站和数据库位于不同的服务器上。

目前,我有一个看起来像这样的课程:

public class Mongo
{
    private IMongoDatabase database;

    public Mongo()
    {
        var client = new MongoClient("mongodb://username:password@ipaddress:port");
        database = client.GetDatabase("MyDatabase");
    }

    public IMongoCollection<ApplicationUser> Users() { return database.GetCollection<ApplicationUser>("Users"); }
}

所以要使用它......(主要是在每个请求中)

Mongo mon = new Mongo();
mon.Users.Find(........); // etc.

这很好用 - 大多数时候。在奇怪的情况下,事情似乎锁定,数据库只是超时,直到我重新启动网站。我在尝试连接数据库服务器时遇到超时错误。

当这种情况发生时,数据库服务器已经确认,我可以毫无问题地从另一个mongo客户端连接到MongoDB。

我意识到这里没有“紧密联系”的事情 - 但似乎没有一个人甚至打电话。所以我想知道是否有太多的连接被打开而没有正确清理。

我是否错误地使用MongoClient?是否有一些显而易见的事情可以阻止这些超时?

这是我得到的错误的痕迹:

   [SocketException (0x274c): A connection attempt failed because the connected party did not properly respond after a period of time, or established connection failed because connected host has failed to respond ipaddress:port]
       System.Net.Sockets.Socket.EndConnect(IAsyncResult asyncResult) +6768957
       System.Threading.Tasks.TaskFactory`1.FromAsyncCoreLogic(IAsyncResult iar, Func`2 endFunction, Action`1 endAction, Task`1 promise, Boolean requiresSynchronization) +57
       System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) +99
       System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) +58
       System.Runtime.CompilerServices.TaskAwaiter.ValidateEnd(Task task) +25
       MongoDB.Driver.Core.Connections.<ConnectAsync>d__7.MoveNext() +1542
       System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) +99
       System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) +58
       MongoDB.Driver.Core.Connections.<CreateStreamAsync>d__0.MoveNext() +345
       System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) +99
       System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) +58
       MongoDB.Driver.Core.Connections.<OpenAsyncHelper>d__1.MoveNext() +553

[MongoConnectionException: An exception occurred while opening a connection to the server.]
   MongoDB.Driver.Core.Connections.<OpenAsyncHelper>d__1.MoveNext() +1372
   System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) +99
   System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) +58
   MongoDB.Driver.Core.Servers.<GetChannelAsync>d__0.MoveNext() +548
   System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) +99
   System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) +58
   System.Runtime.CompilerServices.TaskAwaiter.ValidateEnd(Task task) +25
   MongoDB.Driver.Core.Operations.<ExecuteAsync>d__2.MoveNext() +639
   System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) +99
   System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) +58
   MongoDB.Driver.<ExecuteReadOperationAsync>d__0`1.MoveNext() +272
   System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) +99
   System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) +58
   MongoDB.Driver.<ExecuteReadOperation>d__35`1.MoveNext() +396
   System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) +99
   System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) +58
   MongoDB.Driver.<ToListAsync>d__14`1.MoveNext() +294

1 个答案:

答案 0 :(得分:8)

您错误地使用了MongoClient。如the docs

中所示
  

建议在全球范围内存储MongoClient个实例,   要么作为静态变量,要么在具有单例的IoC容器中   寿命。

因此,您应该重新编写代码以创建单个MongoClient实例,然后在整个程序中使用它。

原因是MongoClient维护了一个可以共享和重用的开放连接池,以获得最佳性能。