c++ - 如果使用 Makefile 编译,代码可以正常工作,如果使用 XCode 编译,代码会崩溃

标签 c++ c++11 xcode4 makefile ofstream

我在我的一个项目中遇到了一个奇怪的问题。我的代码库依赖于一个外部库,其中包含一个名为 Dataset 的类. Dataset类私有(private)继承自 std::vector<Sample> (其中 Sample 是库中定义的另一个自定义类)。

此外,这样的类公开了一个 Save成员函数,以便将构成数据集的数据序列化为文本文件。 Save成员函数定义如下:

inline void Dataset::Save(string filename, ModalityType modality)
{
    ofstream log_file;
    if (modality == OVERWRITE) {
        log_file.open(filename.c_str());
    } else {
        log_file.open(filename.c_str(), ios::out | ios::app);
    }
    if (log_file.is_open()) {
        log_file << *(this);
    }
    log_file.close();
}

ofstream& operator<< (ofstream& out, Dataset& ds)
{
    unsigned int size = ds.size();
    unsigned int input_size = ds.GetInputSize();
    unsigned int output_size = ds.GetOutputSize();
    out << input_size << " " << output_size << endl;
    for (unsigned int i = 0; i < size; i++) {
        Sample* s = ds[i];
        for (unsigned int j = 0; j < input_size; j++) {
            out << s->GetInput(j) << " ";
        }
        for (unsigned int j = 0; j < output_size; j++) {
            out << s->GetOutput(j) << " ";
        }
        out << endl;
    }

    return out;
}

我的代码和外部库都是在 OS X 10.8.2 下编译的,带有 clang 4.2 和标志 -std=c++11 -stdlib=libc++ .我需要这样做,因为我的代码库使用了多个 C++11 工具(例如 random )。此外,该库本身依赖于 boost,而 boost 又使用 clang 和 C++11 支持进行编译。

使用以下 Makefile 一切都按预期编译和工作:

CXX = clang++
CXXDIALECT = -std=c++11 -stdlib=libc++
DEFS = -DBOOST_NO_CXX11_NUMERIC_LIMITS
INCLUDE_DIRS = -I. -I/usr/local/include -I/usr/include -I/opt/local/include
LIB_DIRS = -L/usr/local/lib -L/opt/local/lib 
LIBS = -lfitted -lgsl -lgslcblas -lboost_thread -lboost_program_options -lboost_regex -lboost_system
CINCLUDE = $(INCLUDE_DIRS)
CXXFLAGS = -Os $(CXXDIALECT) $(CINCLUDE) $(DEFS)

tests := main.cpp
sources := $(filter-out $(tests), $(wildcard *.cpp))
objects :=  $(patsubst %.cpp,%.o,$(sources))

main: $(objects)
    $(CXX) $(CXXFLAGS) -o $@ $@.cpp $(objects) $(LIBS) $(LIB_DIRS) -v

%.o : %.cpp
    $(CXX) -c $(CPPFLAGS) $(CXXFLAGS) $< -o $@

哪里libfitted是外部库的名称。

尽管如此,我还是在 XCode 4.6.2 下开发我的代码。问题是每次我尝试运行/调试 XCode 中的代码时,Dataset.save成员函数触发以下错误:

EXC_BAD_ACCESS

和一个空的 dataset.txt文件在磁盘上创建。这是堆栈跟踪的几个屏幕截图:

stack trace1 stack trace2

点击herehere以全尺寸查看它们。

正如屏幕截图中所报告的,问题似乎ofstream.flush()中。成员函数。 最后,我报告了 make 的输出和 xcodebuild

我真的不明白为什么同样的代码,同样的编译器和库,如果用前面提到的 Makefile 编译,可以正确执行,而如果在 XCode 中执行,却不能正常工作。

更新 #1: 我刚刚注意到,如果我使用 GDB 而不是 LLDB 作为调试器,代码在 XCode 中是可调试的(尽管不可运行),即 Dataset.save不会触发 EXC_BAD_ACCESS 错误。

更新 #2: 我用 -g -O0 重新编译了库标志以便保留调试符号。问题是每次 ofstream对象在 Dataset 中初始化的成员函数,this Dataset 的指针实例变为 NULL ,即 Dataset对象被取消。因此,每次尝试访问成员函数内的任何数据成员都会导致 EXC_BAD_ACCESS。 .这是我见过的最奇怪的事情之一。知道为什么会这样吗?

谢谢。

最佳答案

std::vector<...> 类并非旨在用作基类。如此使用它会导致未定义的行为(至少是因为缺少虚拟析构函数)。

C++ 编码标准(Sutter、Alexandrescu)中的第 35 项,已命名

Avoid inheriting from classes that were not designed to be base classes.

在这里可能会有帮助。

Effective C++ (Meyers) 的第 7 条讨论了这个问题:

class SpecialString: public std::string{
...
}

At first glance, this may look innocuous, but if anywhere in an application you somehow convert a pointer-to-SpecialString into a pointer-to-string and you then use delete on the string pointer, you are instantly transported to the realm of undefined behavior.

关于c++ - 如果使用 Makefile 编译,代码可以正常工作,如果使用 XCode 编译,代码会崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16882160/

相关文章:

objective-c - 如何仅根据名称在应用程序中设置图标

c++ - C++中默认复制构造函数的奇怪行为

c++ - 在 vscode linux 中使用 cpp 扩展设置 lldb 调试

c++ - 如果未明确说明大小信息,模板函数如何知道大小?

c++ - 如何在与定义分离的 void 上实现部分专用模板?

c++ - 如何在 C++11 中的映射中找到所有等于或高于 KEY_1 且低于 KEY_2 的键?

ios - 构建 App Store 和 Adhoc Distribution 时出错(使用 three20)

objective-c - 在 Xcode 中为无法识别的选择器创建断点

C++ 帮助 STL - sort() 函数

c++ - 制作 boost::ptr_vector 容器类 push_back 函数