如何解决分割错误C ++

时间:2018-09-17 03:55:00

标签: c++ arrays pointers segmentation-fault

我正在编写一个程序,该程序获取整数数组及其逻辑大小。调用时,它将创建一个仅包含数组中正数的新数组。

现在,要执行此操作,我需要编写一个具有以下参数的void类型函数:

(int* arr, int arrSize, int** outPosArrPtr, int* outPosArrSizePTR)

我应该使用指针int** outPosArrPtr来更新包含正数的数组的基地址,并使用指针outPosArrSizePtr来更新数组的逻辑大小。

现在,当我在xcode编译器上运行代码时,逻辑大小将更新为一个更大的数字。因此,当我尝试使用在线gdb编译器运行程序时,出现错误“细分错误”。

通过阅读Segmentation Fault的意思,我了解到这意味着我正在尝试访问“不属于我”的内存或不在调用堆栈或程序堆部分中的内存。

我尝试通过查看是否引用任何空指针或查看我是否引用任何悬空指针来调试代码,但似乎问题出在另一个地方。

我的代码:

#include <iostream>

typedef int* IntArrPtr;
using namespace std;


int main() {
    int arrSize;
    int *ptrSize;
    ptrSize = &arrSize;

    cout << "How many integers will this array hold:\n ";
    cin >> arrSize;

    IntArrPtr a;
    a = new int[arrSize];

    fillArr(a, arrSize);

    getPosNums4(a, arrSize,&a, ptrSize);
    cout << "The new size in main is: " << arrSize << endl;

    cout <<"The new array with positive integers is:\n";
    /*for(int i =0; i<arrSize;i++) // this runs for a large size of arrSize
        cout<< a[i] << " ";
    cout<<endl; */
    return 0;
}

void fillArr(int a[], int size){
    cout << "Please enter " << size << " Integers separated by spaces\n";
    cout << "Press enter when finished >\n";
    int i;

    for (i=0;i<size;i++)
        cin >> a[i];

}

void getPosNums4(int* arr, int arrSize, int** outPosArrPtr,int* outPosArrSizePtr){
    IntArrPtr newArr;
    newArr = new int[arrSize];
    int i;
    int newIndx = 0;
    outPosArrSizePtr = &newIndx;//initiliaze the pointer.
    for(i=0;i<arrSize;i++){
        if(arr[i] > 0){
            newArr[newIndx] =arr[i];
            newIndx++;
        }
    }
    arrSize = newIndx;
    *outPosArrSizePtr = arrSize;
    cout << "The new size is of *outPosArrSizeptr is: " << *outPosArrSizePtr << endl;

    for(int j=0;j<newIndx;j++)
        outPosArrPtr[j] = &newArr[j];
    delete []newArr;
    newArr = NULL;
    for(int i=0;i<newIndx;i++)
        arr[i] = *outPosArrPtr[i];

}

当我在Xcode上运行此程序时的示例:

How many integers will this array hold:
 6
Please enter 6 Integers separated by spaces
Press enter when finished >
3 -1 -3 0 6 4
The new size is of *outPosArrSizeptr is: 3
The new array with positive integers is:
The new size in main is: 7445512
The program ended with exit code: 0

2 个答案:

答案 0 :(得分:1)

那里有很多问题,但是最关键的问题是,将值分配给函数的自变量不会影响您将其值作为自变量传递的变量。
该参数是一个指针都没关系–指针没有什么特别的。

我认为正在发生的事情是函数中的“来回复制”循环(我不明白它应该做什么)正在输入数组之外写入,从而导致未定义的行为,在这种情况下,会覆盖main中的变量。

您使函数过于复杂了。应该

  • 创建一个新数组
  • 将正值复制到此数组
  • 使用此数组的地址及其(逻辑)大小更新输出参数

(将out参数视为返回值,并最后处理它们。)

类似这样的东西:

void getPosNums4(int* arr, int arrSize, int** outPosArrPtr,int* outPosArrSizePtr){
    int* newArr = new int[arrSize];
    int newIndx = 0;
    for (int i = 0; i < arrSize; i++){
        if (arr[i] > 0){
            newArr[newIndx] = arr[i];
            newIndx++;
        }
    }
    *outPosArrPtr = newArr;
    *outPosArrSizePtr = newIndx;
}

您也不应传递指向“原始”的指针以进行此函数的修改,而应使用新变量。

int main() {
    int arrSize = 0;
    cout << "How many integers will this array hold:\n ";
    cin >> arrSize;
    int* a = new int[arrSize];
    fillArr(a, arrSize);
    int * positives = nullptr;
    int positiveSize = 0;
    getPosNums4(a, arrSize, &positives, &positiveSize);
    cout << "The new size in main is: " << positiveSize << endl;

    delete [] a;
    delete [] positives;
}

答案 1 :(得分:0)

现代C ++使用向量,而不是手动分配数组。手动分配容易产生各种难以调试的错误。

您的getPosNums4方法中的逻辑似乎很麻烦。如果我理解此要求,它应该在输入数组中查找正整数并将其复制到新分配的输出数组。过度分配输出数组不是最优的,但不是实际的错误。

void getPosNums4(int* arr, int arrSize, int** outPosArrPtr,int* outPosArrSizePtr){
    IntArrPtr newArr;
    newArr = new int[arrSize];
    int i;
    int newIndx = 0;
    for(i=0;i<arrSize;i++){
        if(arr[i] > 0){
            newArr[newIndx] =arr[i];
            newIndx++;
        }
    }
    *outPosArrSizePtr = newIndx;
    cout << "The new size is of *outPosArrSizeptr is: " << *outPosArrSizePtr << endl;
    *outPosArrPtr = newArr;
}

请注意,新分配的数组将需要通过调用函数删除[],否则将导致内存泄漏。

这里是现代C ++中的相同程序。请注意,没有使用new / delete可以节省很多痛苦。

#include <iostream>
#include <vector>
#include <algorithm>

using namespace std;

int main() {
    vector<int> integer_vector;
    vector<int> positive_vector;

    cout << "Type in integers. Type a Q to continue:" << endl;
    int an_int;
    while(cin >> an_int)
      integer_vector.push_back(an_int);

    for_each(integer_vector.begin(),integer_vector.end(),[&](int const& n){ 
       if(n > 0)
          positive_vector.push_back(n);
    });
    cout <<"The new array with positive integers is:\n";
    for(auto const & element:positive_vector)
        cout<< element << " ";
    cout<<endl; 
    return 0;
}