0

我有一个字符串流,其中包含多行文本。

例如,

f.html                            Sat Oct 19 22:59:47 2013         23675
folder                            Mon Nov  4 19:36:14 2013         4096
readdirpractice.cpp               Tue Nov  5 03:00:10 2013         1203
server.cpp                        Mon Nov  4 21:22:27 2013         11369
photo.jpg                         Wed Oct 23 01:45:04 2013         4360
qq                                Sun Nov  3 01:54:36 2013         66031
server.cpp~                       Mon Nov  4 21:22:25 2013         11368
myhttp.cpp                        Sun Nov  3 01:43:09 2013         1816
getoptpractice.cpp~               Sun Nov  3 01:15:25 2013         1324

这就是字符串流通常的样子。

当我将此字符串流转换为字符串时,信息是相同的。

但是,当我将其转换为 C 字符串时,它只占用第一行,而所有其他行都丢失了。

它只有

f.html                            Sat Oct 19 22:59:47 2013         23675

转换后。

为什么会这样,我该如何解决?

我附上了我的代码的一部分。

if(is_dir) {
    char dirname[1024];
    strncpy(dirname, requests[1].c_str(), sizeof(dirname));
    dirname[sizeof(dirname)-1] = 0;

    DIR           *d;
    struct dirent *dir;
    d = opendir(dirname);
    stringstream ss;
    if (d) {
        while ((dir = readdir(d)) != NULL) {
            char* file = dir->d_name;
            if(file[0] != '.') {
                struct stat sb;

                if (stat(file, &sb) == -1) {
                    cerr << "stat error" << endl;
                    exit(EXIT_FAILURE);
                }

                char* lm = ctime(&sb.st_mtime);
                string lastmod(lm);
                lastmod.at(lastmod.size()-1) = '\0';
                string spacing = "                                  ";
                ss << file << spacing.substr(0, spacing.size() - strlen(file)) << lastmod << spacing.substr(0, spacing.size() - lastmod.size()) << sb.st_size << '\n';
            }
        }
        closedir(d);
    }
    //cout << ss.str() << endl; // for testing
    char msg2[10000];
    strncpy(msg2, ss.str().c_str(), sizeof(msg2));
    msg2[sizeof(msg2)-1] = 0;
    msg = msg2;
4

3 回答 3

2

std::string可能嵌入了(NUL) 字符,这些字符在被解释为 C 字符串时会'\0'提前终止该序列。char

尝试使用擦除 NUL 字节std::remove

编辑 - 这可能是你的问题:

             lastmod.at(lastmod.size()-1) = '\0';

这只是自找麻烦。std::string忠实地存储您的 NUL 字节并将其复制到输出中。

于 2013-11-05T08:52:39.660 回答
1

根据评论,您指出“msg”是返回值。

char msg2[10000];
strncpy(msg2, ss.str().c_str(), sizeof(msg2));
msg2[sizeof(msg2)-1] = 0;
msg = msg2;

“msg2”被创建为局部变量,即在堆栈上。然后,您将 stringstream 复制到此堆栈区域,获取字符串的地址(在堆栈上),然后离开您的函数,此时堆栈的该区域被释放并由应用程序的其他部分使用。

您要么需要在堆上为字符串分配内存,要么需要从调用函数传入存储空间。

由于您已经在进行字符串操作,因此使用 std::string 封装管理开销将是最合乎逻辑的。将标准字符串的引用作为参数:

void getDirListing(std::string& msg) {
}

然后打电话给

std::string dirListing = "";
getDirListing(dirListing);
于 2013-11-05T09:04:02.053 回答
1

这些行看起来很可疑:

string lastmod(lm);
lastmod.at(lastmod.size()-1) = '\0';

将空字符插入字符串类实例的中间可能会产生意外行为(例如打印时字符串过早终止)。

我认为你不需要这样做。只需尝试删除该lastmod.at行。

于 2013-11-05T09:05:22.877 回答