Mongodb C ++ 11 API错误干扰

时间:2015-09-15 19:26:31

标签: c++ mongodb api c++11 mongodb-c

似乎很多mongodb c ++ 11函数将系统错误代码更改为11(EWOULDBLOCK / EAGAIN)。这目前正在干扰我的其他程序。我有几个问题:

  • mongodb将errno更改为11的原因是什么?
  • 有没有办法在每次调用各种mongodb函数后重置errno?

下面是一个示例,显示了errno的变化是多么普遍。该示例改编自:https://www.mongodb.com/blog/post/introducing-new-c-driver?jmp=docs&_ga=1.90709144.367237569.1438109079

#include <iostream>

#include <bsoncxx/builder/stream/document.hpp>
#include <bsoncxx/json.hpp>

#include <mongocxx/client.hpp>
#include <mongocxx/instance.hpp>

int main(int, char**) {
    errno = 0;
    int counter(0);
    std::string str;

    mongocxx::instance inst{};
    mongocxx::client conn{};

    bsoncxx::builder::stream::document document{};

    auto collection = conn["testdb"]["testcollection"];
    document << "hello" << "world";

    printf("errno %i ... %s:%i\n", errno, __FILE__, __LINE__);
    collection.insert_one(document.view());
    printf("errno %i ... %s:%i\n", errno, __FILE__, __LINE__);
    errno = 0;

    printf("errno %i ... %s:%i\n", errno, __FILE__, __LINE__);
    collection.insert_one(document.view());
    printf("errno %i ... %s:%i\n", errno, __FILE__, __LINE__);
    errno = 0;

    auto cursor = collection.find({});

    printf("errno %i ... %s:%i\n", errno, __FILE__, __LINE__);
    for (auto&& doc : cursor) {
        if (errno) {
            printf("errno %i ... %s:%i\n", errno, __FILE__, __LINE__);
            errno = 0;
        }

        str = bsoncxx::to_json(doc);
        //std::cout << str << std::endl;
        printf("counter: %i\n",counter++);
    }

    printf("errno %i ... %s:%i\n", errno, __FILE__, __LINE__);
    collection.drop();
    printf("errno %i ... %s:%i\n", errno, __FILE__, __LINE__);
}

结果如下:

errno 0 ... hellomongo.cpp:22
errno 11 ... hellomongo.cpp:24
errno 0 ... hellomongo.cpp:27
errno 11 ... hellomongo.cpp:29
errno 0 ... hellomongo.cpp:34
errno 11 ... hellomongo.cpp:37
counter: 0
counter: 1
errno 0 ... hellomongo.cpp:46
errno 11 ... hellomongo.cpp:48

1 个答案:

答案 0 :(得分:0)

一般情况下,几乎所有对任何库的调用都会导致errno发生变化。

在内部,C ++ 11驱动程序构建在C驱动程序之上,这使得调用socket(7)subsytem,它设置了errno。特别是,C驱动程序库进行非阻塞套接字IO调用,这解释了您在errno中观察到的EWOULDBLOCK / EAGAIN值。

通常,使用errno的良好做法要求您在调用要提取详细错误信息的函数后立即捕获errno值 。干预电话可能会重置它。