Chinaunix

标题: 今天犯的一个愚蠢的错误 [打印本页]

作者: converse    时间: 2008-06-03 16:08
标题: 今天犯的一个愚蠢的错误
测试代码如下:


  1. #include <string.h>
  2. #include <string>
  3. using namespace std;

  4. typedef struct Test
  5. {
  6.     string name;
  7.     int    a;
  8. }SoftwareInfo;

  9. int main()
  10. {
  11.     Test tTest;
  12.     memset(&tTest, 0, sizeof(Test));

  13.     return 0;
  14. }

复制代码



作者: 思一克    时间: 2008-06-03 16:08
memset应该没有问题.

问题出在main()返回时候的DESTRUCTOR无法运行了.因为string那里被搞成0了.
作者: finixlei    时间: 2008-06-03 16:24
sizeof(Test)错了吧
string的size是16吧,不是真实的那个string的大小
作者: lipingtababa    时间: 2008-06-03 16:46
干吗要用c的memset把c++的string清零
作者: dpsuffix    时间: 2008-06-03 16:48
原帖由 converse 于 2008-6-3 16:08 发表
测试代码如下:


#include
#include
using namespace std;

typedef struct Test
{
    string name;
    int    a;
}SoftwareInfo;

int main()
{
    Test tTest;
    memset(&tTest, 0, s ...

木看懂。。。
作者: welcome008    时间: 2008-06-03 16:52
版主讲讲哪里错了。
作者: yuangong    时间: 2008-06-03 17:09
对c++不懂,不过对结构体的定义在c里面应该是编译不过的吧?
struct Test tTest;
作者: converse    时间: 2008-06-03 17:11
g++ -g参数编译程序,gdb跟进去看看就知道为什么了

作者: converse    时间: 2008-06-03 17:13
标题: 回复 #6 yuangong 的帖子
这就是一个C++问题...帖子的分类属于C++.另外别在struct Test tTest这样的问题上较劲,那不是这个问题的关键.

[ 本帖最后由 converse 于 2008-6-3 17:14 编辑 ]
作者: welcome008    时间: 2008-06-03 17:14
Breakpoint 1, main () at t1.c:11
11      int main()
Current language:  auto; currently c++
(gdb) n
main () at t1.c:13
13          Test tTest;
(gdb) n
14          memset(&tTest, 0, sizeof(Test));
(gdb) n
16          return 0;
(gdb) n

Program received signal SIGSEGV, Segmentation fault.
0x024dba19 in __gnu_cxx::__exchange_and_add () from /usr/lib/libstdc++.so.6
作者: converse    时间: 2008-06-03 17:15
标题: 回复 #9 welcome008 的帖子
你咋不bt看一下是哪儿错了
作者: welcome008    时间: 2008-06-03 17:16
楼主为啥:
#include <string.h>
#include <string>

这样呢?
作者: converse    时间: 2008-06-03 17:16
标题: 回复 #11 welcome008 的帖子
厄....看来你不懂C++...

作者: welcome008    时间: 2008-06-03 17:17
原帖由 converse 于 2008-6-3 17:15 发表
你咋不bt看一下是哪儿错了



呵呵,我对汇编很弱的

刚才还想发贴,觉得做C对汇编也要有很好的基础知识,不然没法bt调试,都看不懂,呵呵。
作者: welcome008    时间: 2008-06-03 17:18
(gdb) bt
#0  0x024dba19 in __gnu_cxx::__exchange_and_add () from /usr/lib/libstdc++.so.6
#1  0x024bfe24 in std::basic_string<char, std::char_traits<char>, std::allocator<char> >::~basic_string () from /usr/lib/libstdc++.so.6
#2  0x08048543 in ~Test (this=0xbfd82ed at t1.c:6
#3  0x08048511 in main () at t1.c:16
作者: yuangong    时间: 2008-06-03 17:19
原帖由 converse 于 2008-6-3 17:16 发表
厄....看来你不懂C++...

呵呵
作者: converse    时间: 2008-06-03 17:20
标题: 回复 #2 思一克 的帖子
bingo~~

作者: 思一克    时间: 2008-06-03 17:22
我说错了?

原帖由 converse 于 2008-6-3 17:20 发表
bingo~~

作者: 思一克    时间: 2008-06-03 17:24
哈. 我说对了.

这也是C++的要注意的问题. 明明一个没有问题的操作(memset), 最后的返回运行时候却出了错误.
作者: Edengundam    时间: 2008-06-03 17:31
析构函数时候访问了空指针?
作者: Edengundam    时间: 2008-06-03 17:33
哦, 思在前面说了... 没注意
作者: ew3j    时间: 2008-06-03 18:04
呵呵
作者: tyc611    时间: 2008-06-03 18:05
对一个别人写的类用memset,不是找死么
作者: aaaab    时间: 2008-06-03 18:15
人家已经初始化过了,嘿嘿
作者: scutan    时间: 2008-06-03 18:23
好像前两天也在本版看到一个相似的问题,不过是把虚函数指针给清空了。
作者: chrisyan    时间: 2008-06-03 21:02
写c++的根本就不会这样写,c++中不需要typedef 一个struct.
struct和class的区别就是:1.默认的访问级别为public,2继承时默认是public继承。
所以你一memset,就都破坏掉了。
这个东西还要用gdb调?根本不需要
作者: zarcoder_neu    时间: 2008-06-03 21:07
原帖由 scutan 于 2008-6-3 18:23 发表
好像前两天也在本版看到一个相似的问题,不过是把虚函数指针给清空了。

是那个最终
用memset(&a+4,0,sizeof(a)-4)解决的那个吗?
作者: voipexplore    时间: 2008-06-03 21:39
呵呵 c风格的c++。里面有string,string的内存布局是不透明的。楼主想当然了。
对带string的struct,也不能用malloc,要用new
总之,如果你想在c++里用struct,那么用class吧,所有属性public

或者 struct 里只带 基本类型。
作者: wolfchyu    时间: 2008-06-03 22:20
学些了,呵呵。看来这个贴放在C++的坛里才能容易解答哦
作者: xibaluma    时间: 2008-06-03 23:59
delete 一个空指针不是可以的么

没看懂 LZ最好把答案整理到顶楼啊 呵呵....
作者: ncoder    时间: 2008-06-04 08:53
厉害 前些天 我还遇到过这个问题 问题说段错误....

最终才发现, 是不小心把某个变量的堆栈给破坏了 ....
作者: xujg    时间: 2008-06-04 08:57
原帖由 ncoder 于 2008-6-4 08:53 发表
厉害 前些天 我还遇到过这个问题 问题说段错误....

最终才发现, 是不小心把某个变量的堆栈给破坏了 ....


什么是"某个变量的堆栈"?
作者: cvtc    时间: 2008-06-04 09:10
原帖由 tyc611 于 2008-6-3 18:05 发表
对一个别人写的类用memset,不是找死么


就是,这就是上厕所打灯笼—找屎。
C++费死劲创建了一个对象,你残忍地从头到尾给每个字节洗刷了一遍,不想活了。
这是逻辑上的愚蠢错误!!!!!!!!!!!!

[ 本帖最后由 cvtc 于 2008-6-4 09:14 编辑 ]
作者: yecheng_110    时间: 2008-06-04 09:18
C++不应该这样用(memset)吧
作者: c/unix    时间: 2008-06-04 09:30
提示: 作者被禁止或删除 内容自动屏蔽
作者: kktype    时间: 2008-06-04 09:32
C++不懂
作者: joneson119    时间: 2008-06-04 11:12
有点深
作者: cuinantrue    时间: 2008-06-04 14:02
原帖由 chrisyan 于 2008-6-3 21:02 发表
写c++的根本就不会这样写,c++中不需要typedef 一个struct.
struct和class的区别就是:1.默认的访问级别为public,2继承时默认是public继承。
所以你一memset,就都破坏掉了。
这个东西还要用gdb调?根本不需要


可移植也是要注意的吧.你也不知道你写的东西会怎么编译,在什么平台上跑.
作者: chrisyan    时间: 2008-06-04 16:11
原帖由 cuinantrue 于 2008-6-4 14:02 发表


可移植也是要注意的吧.你也不知道你写的东西会怎么编译,在什么平台上跑.


如果写了
.
.
#include <string>
using namespace std;
.
.

那肯定就是用c++编译器来编吧,我说的那是c++的标准啊,在什么平台上有啥问题吗?
作者: liguangyi    时间: 2008-06-04 16:23
呵呵,这个问题我刚开始也遇到过,而且很难找出错误,因为往往是在程序退出的时候,报指针错误
作者: cplusplusnew    时间: 2008-06-04 22:51
memset 把析构函数都给清0了
作者: whyglinux    时间: 2008-06-05 00:10
>> memset应该没有问题.

问题恰恰出现在对 memset() 的使用上。

在 C++ 中,memset() 以及 memcpy()、memmove() 等函数只能用来操作 POD 类型(可简单理解为与 C 兼容的类型)的对象。搂主程序中用它操作了非 POD 对象,所以给程序带来了问题。

了解 POD 这个概念非常重要,它是上述函数能否用来操作对象的分水岭。

>> 问题出在main()返回时候的DESTRUCTOR无法运行了.因为string那里被搞成0了.

可能出现问题,也可能不出现问题。如果出问题的话可能是这个原因,也可能是其它原因。对于无定义的情况,一般来说不会有一个确定的结论——对于同一问题,不同的实现可能带来相同或不同的表现。
作者: rjgb    时间: 2008-06-05 08:37
哦, 是这样的.
作者: decwang    时间: 2008-06-05 08:58
结构体里有类作为成员的时候,不要用memset,这是CPP常识吧
作者: tspy007    时间: 2008-06-05 09:59

作者: tyz    时间: 2008-06-05 11:21
hao  
学习
作者: qingfengjianke    时间: 2008-06-05 11:34


我在vc6.0试了下,没报一点错.....
作者: 思一克    时间: 2008-06-05 12:31
string name不在结构体中.

结构体中那里仅仅是一个指针. 将此指针清0了.
作者: welcome008    时间: 2008-06-05 14:41
谁能把这个例子给逐步gdb调试一下,给我示例一下怎么检测到内存释放错误?

谢谢!
我gdb,但看不到哪里显示说内存问题。
作者: zhangxianxiang    时间: 2008-06-05 16:02
c, c++混用的后果
作者: gateh    时间: 2008-06-05 17:51
不能用memset去初始化一个带了对象的结构. 这样会破坏对象的内存结构
作者: 勿丑于    时间: 2008-06-06 08:51
这个例子里面没有错误,很可能是虚惊一场。

我看了C++,我差点疯了,我认为string name在生成实例tTest的时候没有被分配内存,分配内存的时候,C++可以重载=然后malloc,也就是说,name的地址正好是tTest的地址。

如果生成一个不带有参数的对象的构造函数分配了内存,只要内存的地址没有变化,也不会有问题。

这里的秘密就是name正好是tTest的第一个对象,name的地址就是tTest的地址。
作者: 勿丑于    时间: 2008-06-06 09:27
另外我想了析构函数,就是前面有个~的函数,析构又怎么样,析构来析构去还是要返回对象的地址,就是把对象的空间释放,但是对象却是在main函数返回的时候,靠出栈来收回。
作者: weichongli    时间: 2008-06-06 11:43
原帖由 converse 于 2008-6-3 16:08 发表
测试代码如下:


#include
#include
using namespace std;

typedef struct Test
{
    string name;
    int    a;
}SoftwareInfo;

int main()
{
    Test tTest;
    memset(&tTest, 0, s ...


    Test tTest;
Test是一种数据类型么?
作者: 勿丑于    时间: 2008-06-06 15:16
原帖由 勿丑于 于 2008-6-6 08:51 发表
这个例子里面没有错误,很可能是虚惊一场。

我看了C++,我差点疯了,我认为string name在生成实例tTest的时候没有被分配内存,分配内存的时候,C++可以重载=然后malloc,也就是说,name的地址正好是tTest的地 ...


没人回答好没劲哪,我又想了想C++,看来对于像我这样的准黑客一级的人,C++很容易crack呀。C++出现的时候有人准保在偷着笑呢,hohohoho~~~~~~~~
作者: pythonor    时间: 2008-06-06 18:39
呵呵,楼主的错误在于用memset吧C++编译器可能放在string对象里面的任何东西就歼灭为zero了
作者: whyglinux    时间: 2008-06-06 21:11
>> 结构体里有类作为成员的时候,不要用memset,这是CPP常识吧
>> 不能用memset去初始化一个带了对象的结构. 这样会破坏对象的内存结构

类成员或者类对象并不可怕,只要它是一个 POD 的类,都可以用 memset() 来对对象清零。

>> c, c++混用的后果

明明是一个 C++ 程序,何来混用之说?

>> 这个例子里面没有错误,很可能是虚惊一场。

诚然,程序运行正确不能证明程序中不存在错误。但是象搂主的程序,在前面有的帖子中已经贴出了程序运行错误的情况。只要有一个运行错误已经可以说明程序中存在问题了,绝对不会是“虚惊一场”。
作者: nicsky    时间: 2008-06-06 22:33
学习
作者: 勿丑于    时间: 2008-06-07 14:14
原帖由 whyglinux 于 2008-6-6 21:11 发表
>> >> 这个例子里面没有错误,很可能是虚惊一场。

诚然,程序运行正确不能证明程序中不存在错误。但是象搂主的程序,在前面有的帖子中已经贴出了程序运行错误的情况。只要有一个运行错误已经可以说明程序中存在问题了,绝对不会是“虚惊一场”。


是的, 我看到了, 尽管看不懂, 但我也细想了一下例外的情况, 你说的很有道理.
作者: pythonor    时间: 2008-06-07 17:49
所以C++的magic太多了。不要用它,用C+动态语言
作者: 勿丑于    时间: 2008-06-07 18:20
呵呵, 人生苦短, 我用Python, 自由社区的人我现在老崇拜了.
我的意见和你一样, 我认为C++在发明的当天, 就是一种悲哀(原谅我的用词方式), 或者说C++是一个折衷型的语言.
我一看发明人也来自Bell实验室, 我就嗅出味道了.

我现在反正也没什么事, 另外我现在看C++和我想象的作个比较.


另: 我说悲哀的意思是它出来就是折磨人的.

[ 本帖最后由 勿丑于 于 2008-6-7 18:32 编辑 ]
作者: hejinxu    时间: 2008-06-10 12:36
不懂  但是顶一下
作者: cheng_lai_shun    时间: 2008-06-10 13:15
#include <string.h>
#include <string>
using namespace std;
//一般结构体定义我这样写
typedef struct Test
{
    string name;
    int    a;
    Test()
   {
      name = "";
      a        = -1;
   }
}SoftwareInfo;
作者: reesun    时间: 2008-06-12 00:43
将程序更改为以下代码,试试:


  1. #include <string.h>
  2. #include <string>
  3. using namespace std;

  4. typedef struct Test
  5. {
  6.     string name;
  7.     int    a;
  8. }SoftwareInfo;

  9. int main()
  10. {
  11.     SoftwareInfo tTest;
  12.     memset(&tTest, 0, sizeof(struct Test));

  13.     return 0;
  14. }
复制代码

作者: net_robber    时间: 2008-06-12 09:32
原帖由 reesun 于 2008-6-12 00:43 发表
将程序更改为以下代码,试试:


#include
#include
using namespace std;

typedef struct Test
{
    string name;
    int    a;
}SoftwareInfo;

int main()
{
    SoftwareInfo tTest; ...



不会别瞎掰,顺便也看看别人的回帖,当是学习,也算是对别人的尊重

警告一次,在有此等情况,视为恶意灌水
作者: xxandxx    时间: 2008-06-13 18:11
给个默认构造函数吧,能永久避免这种问题
作者: snow888    时间: 2008-06-13 18:25
其实问题的关键应该在结构体中的那个 string 。




欢迎光临 Chinaunix (http://bbs.chinaunix.net/) Powered by Discuz! X3.2