0

我是 C++ 的初学者。我以前用解释语言编写过代码。

我正在使用套接字编程在 C++ 中编写客户端服务器代码来执行一些文件操作。客户端将文件名发送到服务器,服务器打开该文件以执行操作。

以下片段显示了从客户端接受文件名的过程:

char buf[512];
char *name;
connectionID = accept(socketID, (sockaddr *) NULL, NULL);
cout << "Connection created" << endl;
int bytes = recv(connectionID, buf, sizeof(buf), 0);
buf[bytes] = '\0';
strcpy(name, buf);
cout << name << endl;

此代码随机运行。我将它移植到另一个文件,它突然停止工作。当我尝试使用 GDB 查看发生分段错误的位置时,GDB 显示空行或代码中的任何行,似乎无关紧要。

>(gdb) where
>#0  0xb74a8979 in ?? ()
>#1  0x00000000 in ?? ()
>(gdb) up
>#1  0x00000000 in ?? ()
>(gdb) down
>#0  0xb74a8979 in ?? ()
>(gdb) down
>Bottom (innermost) frame selected; you cannot go down.

但是,valgrind 输出显示了一些信息,但我仍然无法弄清楚这里出了什么问题。

>==11855== Invalid write of size 1
>==11855==    at 0x402C6C3: strcpy (in /usr/lib/valgrind/vgpreload_memcheck-x86-    linux.so)
>==11855==    by 0x804D38A: main (server.cpp:100)
>==11855==  Address 0x0 is not stack'd, malloc'd or (recently) free'd
>==11855== 
>==11855== 
>==11855== Process terminating with default action of signal 11 (SIGSEGV)
>==11855==  Access not within mapped region at address 0x0
>==11855==    at 0x402C6C3: strcpy (in /usr/lib/valgrind/vgpreload_memcheck-x86-l

Valgrind 还展示了

>==11871== Use of uninitialised value of size 4
>==11871==    at 0x402C6C3: strcpy (in /usr/lib/valgrind/vgpreload_memcheck-x86-    linux.so)
>==11871==    by 0x8048DF8: main (in /media/sf_programs/project/a.out)

但是,如果我将名称指针初始化为 NULL,则不会发生此错误。然后,只有无效的大小写入发生。

我看到了无效写入大小的帖子,在这些帖子中,当指针超出范围访问时会发生段错误。但我无法弄清楚它在哪里超出范围。

你有什么有用的信息吗?似乎程序在接收字符串输入并将其从客户端存储时崩溃。

4

1 回答 1

1
char *name;

你做了一个指针,很棒,但你忘了让它指向任何东西。例如,内存缓冲区。

不要在这里沉迷于特定的valgrind诊断;他们都在同一个主题上:您正在尝试使用不属于您的完全随机内存。

无论您初始化还是未初始化或喂它奶油奶酪,name这都是正确的。NULL在您为其分配您拥有的有效记忆的位置之前,您还不如在泰坦尼克号的甲板上唱马卡雷娜,因为这艘船可能会在几个场景中漂浮一点(对于一些戏剧性的电影来说足够时间)但是,在从长远来看,它不会再航行了!

于 2013-11-10T02:46:35.893 回答