看似相似但行为不同的 C++ 代码 : any idea why?

标签 c++ return-value

我有一个线段树的 C++ 代码,它经常有效,但是 一个大的输入它失败了。无论如何,跟踪我发现的错误 更改代码的某些部分以获得似乎“等效”的东西 代码可以正常运行。

一些上下文:

struct state {
    int v, pos;
    state (int v, int pos) : v(v), pos(pos)  {}
};
int split(state s);
state go(state, int, int, int);

struct node{
    int link;
...
};
vector<node> t;

此代码无效:

int get_link (int v) {
    if (t[v].link != -1)  return t[v].link;
    if (t[v].par == -1)  return 0;

    int to = get_link (t[v].par);
    state aux = go (state (to,t[to].len()), t[v].l + (t[v].par==0), t[v].r, t[v].i_str);

    return t[v].link = split (aux);
}

这个有效:

int get_link (int v) {
    if (t[v].link != -1)  return t[v].link;
    if (t[v].par == -1)  return 0;

    int to = get_link (t[v].par);
    state aux = go (state (to,t[to].len()), t[v].l + (t[v].par==0), t[v].r, t[v].i_str);

    int ret = split (aux);
    t[v].link = ret;
    return ret;
}

这个有效:

int get_link (int v) {
    if (t[v].link != -1)  return t[v].link;
    if (t[v].par == -1)  return 0;

    int to = get_link (t[v].par);
    state aux = go (state (to,t[to].len()), t[v].l + (t[v].par==0), t[v].r, t[v].i_str);

    int ret = split (aux);
    t[v].link = ret;
    return t[v].link;
}

这个不行:

int get_link (int v) {

    node &w = t[v];
    if (w.link != -1)  return w.link;
    if (w.par == -1)  return 0;

    int to = get_link (w.par);
    state aux = go (state (to,t[to].len()), w.l + (w.par==0), w.r, w.i_str);

    int ret = split (aux);
    w.link = ret;
//  assert(t[v].link == ret);

    return ret;
}

此外,在最后一种情况下,断言失败了。 这很奇怪,因为在所有失败的情况下, get_line 函数运行了很多次 没有问题。

知道哪里出了问题或我误解了什么。

如果有用

$ gcc -v Using built-in specs. COLLECT_GCC=gcc COLLECT_LTO_WRAPPER=/usr/libexec/gcc/x86_64-redhat-linux/4.7.2/lto-wrapper Target: x86_64-redhat-linux Configured with: ../configure --prefix=/usr --mandir=/usr/share/man --infodir=/usr/share/info --with-bugurl=http://bugzilla.redhat.com/bugzilla --enable-bootstrap --enable-shared --enable-threads=posix --enable-checking=release --disable-build-with-cxx --disable-build-poststage1-with-cxx --with-system-zlib --enable-__cxa_atexit --disable-libunwind-exceptions --enable-gnu-unique-object --enable-linker-build-id --with-linker-hash-style=gnu --enable-languages=c,c++,objc,obj-c++,java,fortran,ada,go,lto --enable-plugin --enable-initfini-array --enable-java-awt=gtk --disable-dssi --with-java-home=/usr/lib/jvm/java-1.5.0-gcj-1.5.0.0/jre --enable-libgcj-multifile --enable-java-maintainer-mode --with-ecj-jar=/usr/share/java/eclipse-ecj.jar --disable-libjava-multilib --with-ppl --with-cloog --with-tune=generic --with-arch_32=i686 --build=x86_64-redhat-linux Thread model: posix gcc version 4.7.2 20120921 (Red Hat 4.7.2-2) (GCC)

最佳答案

很可能您的 split 函数正在调整 t vector 的大小。所以在像这样的表达式中 t[v].link = split (aux) 如果 vector::operator[] 在 split 调用之前被评估(这是可能的) 并且 split 函数正在重新分配 vector ,那么您可能正在访问对不再存在的对象的引用。

您使用临时变量的替代代码没有这个问题,因为对 vector::operator[] 的调用肯定发生在对 split 的调用之后。

关于看似相似但行为不同的 C++ 代码 : any idea why?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15807572/

相关文章:

c++ - 如何在 Rust 中为 ArmA 3 DLL 实现 RVExtension 函数?

c++ - 在 macOS 上从源代码构建 Qt 时出错

python - 如何从线程函数中获取多个返回值?

C++ 具有返回值的匿名函数

c++ - 如何访问结构中定义的变量

c++ - 如何在 Emacs 上使用 company-c-headers

c++ - 比较模板类型和具体类型的实例

java - 在 Java 中返回多个值

c - 返回状态/值的适当范围

java - 在类中返回所需的值 - Java