【问题标题】:Delete of array of pointer: _BLOCK_TYPE_IS_VALID删除指针数组:_BLOCK_TYPE_IS_VALID
【发布时间】:2013-04-22 08:41:50
【问题描述】:

我有一个代码,我需要在其中分配一个指向指针数组的指针,但是当我想安全地删除它时,调试器给了我错误。

我知道我不应该使用指针的指针数组,而是使用向量类,但我必须这样做。 我有两个班级:A e B。在 B.h 我有这个:

private: A **aObject;

在 B.cpp 我做这个:

aObject = new A*[size];
for(int i=0; i<size; i++) {
    aObject[i] = new A();
}

当我想删除它时,我这样做:

for(int i=0; i<size; i++) {
    delete [i]aObject; // <----error debuger
}
delete []aObject;

我已经阅读了很多讨论这个问题的线程,看来我使用的方法是正确的。调试器在第 62 行返回错误delete [i]aObject;
那么,你有什么解决办法吗? 此外,当i 为1 时返回错误。A 对象(aObject) 的私有成员为CXX0030
看起来我正在从地址空间中寻址一个内存区域,aObject 是一个空指针,因此调试返回堆上的错误。
谢谢各位。

编辑:
Okei,delete aObject[i] 我不再有这个错误,但我有另一个错误:0xc0000005: Access violation reading location 0x0094e368。现在当i 为0 时返回错误。错误总是在第62 行。aObject 的私有成员是CXX0017aObject 的值是0x0094e368

编辑 2:
我没有为A 定义任何析构函数,只有默认的析构函数,但我在任何地方都不记得了。

A::~A(void){}

编辑 3: B.h,又名bci.h(对不起我的错误)

#pragma once

#include "thinkgear.h"
#include <time.h>
#include <stdlib.h>
#include <stdio.h>
#include "eeg.h"
#include <string>

class bci {

public:

    bci(float timeAcquisition);
    ~bci(void);

    // FUNZIONI BCI //
    // -> Connessione e setting della bci <- //
    void autoInit(void);
    int getDriverVersion(void);
    void setNumPortCom(void);
    void setNumPortCom(int portNumber);
    void getConnectionId(void);
    void connect(void); //Stablisce una connessione alla bci e setta connectionId
    void setStreamLog(const char *filename);
    void setStreamLog(void);
    void setDataLog(const char *filename);
    void setDataLog(void);
    void enableLowPassFilter(void);
    void disableLowPassFilter(void);
    void enableBlinkDetection(void); //Abilita il blink detection
    void disableBlinkDetection(void); //Disabilit il blink detection
    void disconnected(void); //Libera la connessione con la bci

    // -> Lettura e storing EEG <- //
    void readAllEEGValues(void);
    void storeAllEEGValues(void);
    void readStoreAllEEGValues(void);

    // -> Lettura e storing EEG (escluso segnale RAW) <- // poor signal e blink detectection inclusi
    void readEEGValues(void); //Legge i dati dalla bci e aggiorna la variabili
    void storeEEGValues(void);
    void readStoreEEGValues(void);

    // -> Lettura e storing segnale RAW <- //
    void readEEGRawValue(void);
    void storeEEGRawValues(void);
    void readStoreEEGRawValues(void);

    // -> Lettura e storing eSense Meter (attention e meditation) <- // poor signal e blink detectection inclusi
    void readESenseValues(void);
    void storeESenseValues(void);
    void readStoreESenseValues(void);

    // METODI GETTER E SETTER //
    // -> Getter e Setter per i dati estratti <- //
    time_t getTimestamp(int i);
    float getLevelAttention(int i); //Restituisce il livello attuale di attenzione
    float getLevelMeditation(int i); //Restituisce il livello attuale di meditazione
    float getLevelRaw(int i);
    float getLevelDelta(int i);
    float getLevelTheta(int i);
    float getLevelAlpha1(int i);
    float getLevelAlpha2(int i);
    float getLevelBeta1(int i);
    float getLevelBeta2(int i);
    float getLevelGamma1(int i);
    float getLevelGamma2(int i);
    float getBlinkStrengthDetection(int i); //Restitisce il livello attuale del blink detection (o dell'ultimo blink)
    time_t* getTimestamp(void);
    float* getLevelAttention(void); //Restituisce il livello attuale di attenzione
    float* getLevelMeditation(void); //Restituisce il livello attuale di meditazione
    float* getLevelRaw(void);
    float* getLevelDelta(void);
    float* getLevelTheta(void);
    float* getLevelAlpha1(void);
    float* getLevelAlpha2(void);
    float* getLevelBeta1(void);
    float* getLevelBeta2(void);
    float* getLevelGamma1(void);
    float* getLevelGamma2(void);
    float* getBlinkStrengthDetection(void); //Restitisce il livello attuale del blink detection (o dell'ultimo blink)
    bool checkBlinkDetection(void); //Controlla se è avvenuto o meno un bink
    // -> Getter e Setter per gli errori <- //
    int getIdConnection(void);
    int getErrorStreamLog(void);
    int getErrorDataLog(void);
    int getErrorLowPassFilter(void);
    int getErrorBlinkDetection(void);
    int getErrorConnect(void);
    int getPacketsRead(void);
    int getPacketsAnalize(void);

    // LETTURA ERRORI //
    // -> Interpretato gli errori ricevuti <- //
    std::string getStringErrorConnectionId(void);
    std::string getStringErrorStreamLog(void);
    std::string getStringErrorDataLog(void);
    std::string getStringErrorLowPassFilter(void);
    std::string getStringErrorBlinkDetection(void);
    std::string getStringErrorConnect(void);
    /*std::string getStringPacketsRead(void);
    std::string getStringPacketsAnalize(void);*/

    // -> Printing di valori estratti e di variabili utili <- //
    void printBciInfo(void); //Stampa alcune informazioni utili della classe
    void printValues(void);
    void printDataValues(void);
    void printESenseValues(void);

    // -> Metodo setter e getter per timeAcquisition <- //
    // * Numero di campionamenti da fare, o numero di secondi di running della bci * //
    void setTimeAcquisition(int time);
    int getTimeAcquisition(void);

    // -> Salva il segnale EEG <- //
    // * num viene utilizzato nel seguente  modo: se num=0, stampa tutti tranne raw; num=1, stampa solo il raw; num=2, stampa tutto.
    void saveToCsv(const char *filename, int num);
    void saveToCsv(int num);

    // -> Altro <- //
    float* convertRawToVoltage(void); //Converte la matrice data che contiene tutti i segnali raw ai rispetti valori in Volt.
    float convertRawToVoltage(float rawValue); //Converte il valore raw e ritorna il rispettivo valore in Volt.

    // -> Prova <- //
    //eeg** 

    // -> Non implementata <- //
    //void cleanDataFile(void);

    /*void cleanCSVFile(const char *filename);
    std::vector<std::string> &split(const std::string &s, char delim, std::vector<std::string> &elems);
    std::vector<std::string> split(const std::string &s, char delim);*/

private:

    int connectionId; //Id della connessione alla bci. 0 ok, -2 no free memory to allocate connection, -1 to many connection without TG_FreeConnection()
    int dllVersion;
    int numPortCom;

    time_t *timer;
    float *attention; //Livello di attenzione attuale
    float *blinkStrength; //Livello di blink attuale
    bool blink; //E' a 'true' se si è verificato un blink, 'false' altrimenti,
    float *meditation; //Livello di meditazione attuale
    float *poorSignal; //Qualità del segnale ricevuto
    float *raw;
    float *delta;
    float *theta;
    float *alpha1;
    float *alpha2;
    float *beta1;
    float *beta2;
    float *gamma1;
    float *gamma2;

    int errorStreamLog; //-1 invalid id connection, -2 error with file, 0 success
    int errorDataLog; //-1 invalid id connection, -2 error with file, 0 success
    int errorLowPassFilter; //-1 invalid id connection, -2 rawSamplingRate is not a valid rate, 0 success
    int errorBlinkDetection; //-1 invalid id connection, 0 success
    int errorConnect; //-1 invalid id connection, -2 serialPortName could not be opened, -3 if serialBaudrate is not a valid TG_BAUD_* value, -4 if @c serialDataFormat is not a valid TG_STREAM_* type, 0 success
    int errorReadPackets; //-1 invalid id connection, -2 if there were not even any bytes available to be read, -3 if an I/O error occurs

    int packetsRead; //Handler per la lettura dei dati dalla bci
    int packetsAnalize; //Contatore per il numero totale di pacchetti letti dalla bci

    int timerAcquisition;

    eeg **data;

};

编辑 4:
也许我发现了问题。 在主函数中我这样做:

int main(void) {
        ...
    B *bObject=new B(timeAcquisition);
        ...
    bObject->~B();
    delete bObject; //<--- THIS!
}

有可能吗?如果没有delete bObject;,调试器不会返回任何错误。

编辑 5: 所以,答案是here

使用删除操作符也会导致类析构函数(如果有的话)被调用。

【问题讨论】:

  • 这是什么delete [i]aObject;
  • delete aObject[i]; 会更有意义。
  • 使用 RAII 会更有意义。
  • delete [i]aObject; 是完全有效的(尽管很少使用,而且丑陋和利诱)。我怀疑错误出在其他地方(或者编译器处理“奇数索引顺序”时存在错误)。
  • 我的水晶球告诉我保存成员变量A** aObject; 的对象类没有适当的虚拟析构函数、复制构造函数或赋值运算符,并且您隐含地至少复制了一个(浅),销毁它,随后对原始的访问是错误的,因为复制的指针不再有效。发布包含A **aObject; 的类的整个 类定义。然后阅读The Rule of Three

标签: c++ pointers heap-memory


【解决方案1】:

所以,解决方案是here

使用删除操作符也会导致类析构函数(如果有的话)被调用。

int main(void) {
        ...
    B *bObject=new B(timeAcquisition);
        ...
    delete bObject;
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2015-08-09
    • 1970-01-01
    • 1970-01-01
    • 2011-05-10
    • 2015-07-15
    • 1970-01-01
    • 2011-02-18
    相关资源
    最近更新 更多