C++读取访问权限冲突引发异常问题的原因分析

 更新时间:2021年07月01日 12:46:55   作者:Gabriel17  
C语言是一门通用计算机编程语言,广泛应用于底层开发,最近在用C++写代码时经常会遇到“引发了异常: 读取访问权限冲突,所以这篇文章主要给大家介绍了关于C++读取访问权限冲突引发异常问题的相关资料,需要的朋友可以参考下

用C++写代码时经常会遇到“引发了异常: 读取访问权限冲突。”这样的错误提示,这种情况产生原因主要有两点:

一、访问数组越界

当采用线性表的顺序结构,例如顺序表、队列、栈等,用数组存储数据时,若将要读取数据的位置超出了当前数组的长度,就会发生数组访问越界的状况。

可这并不会造成编译错误,也就是说,编译器并不会在你编译的时候就指出你访问数组越界了,这个时候可能还是“0 errors,0 warnings”

你还在暗暗庆幸自己的代码没有bug,但是当你运行之后就会抛出访问权限冲突的异常。

例如:下面这段对队列和栈的操作

struct Stack              //这里定义了一个栈的结构
{
	int data[maxSize];    //定义存储栈中元素的数组
	int top;              //定义栈顶指针
};
struct Queue              //这里定义了一个队列的结构
{
	int elements[2000];   //定义一个长度为2000的数组存放队列中的元素
	int front;            //定义队头指针
	int rear;             //定义队尾指针
};
while (queueA->front < queueA->rear || queueB->front < queueB->rear) 
{
	queueA->front++;
	inStack(stack, queueA->elements[queueA->front]);
	outStack(stack, queueA);
	queueB->front++;
	inStack(stack, queueB->elements[queueB->front]);
	outStack(stack, queueB);
}
//inStack函数实现出队后进栈操作
//outStack函数实现出栈后入队操作,入队后队尾指针后移

编译成功没有报错,但是运行后抛出了读取访问权限冲突的异常

我们查看此时局部变量的状态可以发现,队列A 的队头指针和队尾指针都指向了异常的下标

这显然发生了访问越界的情况,因为长度为2000的数组,下标的取值应该是0~1999。

后来,发现错误出在while循环判定的条件。本意上是当A、B之间任意一个队列中的元素为空的时候就跳出循环,但是错将判定条件中的逻辑且写成了逻辑或,导致迟迟不能跳出循环,最终队列A的队头指针和队尾指针在不断的循环操作中超出了数组边界,发生了读取访问权限冲突的异常。

将while的判定条件修改后,程序可正常运行。

所以遇到读取访问权限冲突的提示,如果异常发生在读取数组中数据的时候,可以优先考虑是否是因为某种操作不当导致访问数组越界了。

二、空指针异常

这主要发生在通过指针读取数据时,比如在使用链表的过程中。

示例:链表中的操作

struct Node              //这里定义了一个结点
{
	int data;
	Node* next;
};
LinkListInvert::LinkListInvert(int arg[],int n)
{                            //重载构造函数对链表进行初始化
	Node* r = first;
	Node* s = nullptr;
	for (int i = 0; i < n; i++)
	{
		s = new Node;
		s->data = arg[i];
		r->next = s;
		r = s;
	}
	r->next = nullptr;
}

代码编译无错误,运行后发生读取访问权限异常

这里不难看出是因为没有初始化first指针,使得first指针成为了一个空指针,指向了一个不确定的值,随后进行的操作

	Node* r = first;

使r指针也成为了一个空指针,在进行接下来的操作时就抛出了读取权限冲突的异常。

LinkListInvert::LinkListInvert(int arg[],int n)
{
	first = new Node;           //此处为修改的地方,对first初始化
	Node* r = first;
	Node* s = nullptr;
	for (int i = 0; i < n; i++)
	{
		s = new Node;
		s->data = arg[i];
		r->next = s;
		r = s;
	}
	r->next = nullptr;
}

而修改的方式也很简单,对first指针进行初始化后即可运行成功。

类似的,由指针未初始化而引发的读取访问权限异常,还会给出诸如 0XCCCCCCCC、0xCDCDCDCD的异常提示。

当然,不止链表,空指针异常还会出现在很多其他情况下,比如在数据库查询,指针未初始化时也会产生空指针异常。

总结

到此这篇关于C++读取访问权限冲突引发异常问题的文章就介绍到这了,更多相关C++读取访问权限冲突内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 对比C语言中execv相关的执行文件的三个函数

    对比C语言中execv相关的执行文件的三个函数

    这篇文章主要介绍了对比C语言中execv相关的执行文件的三个函数,分别为execv()函数和execve()函数以及execvp()函数,需要的朋友可以参考下
    2015-08-08
  • C语言实现简单扫雷游戏

    C语言实现简单扫雷游戏

    这篇文章主要为大家详细介绍了C语言实现简单扫雷游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-02-02
  • C++结合QT实现带有优先级的计算器功能

    C++结合QT实现带有优先级的计算器功能

    这篇文章主要介绍了C++结合QT实现带有优先级的计算器,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-01-01
  • c++将vector迭代器转换为指针的实现方式

    c++将vector迭代器转换为指针的实现方式

    这篇文章主要介绍了c++将vector迭代器转换为指针的实现方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-11-11
  • C语言的运算符你了解吗

    C语言的运算符你了解吗

    这篇文章主要介绍了C语言中的运算符,本文给大家介绍的非常详细,具有参考借鉴价值,需要的朋友可以参考下,希望能给你带来帮助
    2021-08-08
  • C语言实现随机发扑克牌

    C语言实现随机发扑克牌

    这篇文章主要为大家详细介绍了C语言实现随机发扑克牌,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-04-04
  • C++实现一个简单消息队列的示例详解

    C++实现一个简单消息队列的示例详解

    消息队列在多线程的场景有时会用到,尤其是线程通信跨线程调用的时候,就可以使用消息队列进行通信。本文将利用C++实现一个简单的消息队列,感兴趣的可以了解一下
    2022-12-12
  • C++ Boost Optional示例超详细讲解

    C++ Boost Optional示例超详细讲解

    Boost是为C++语言标准库提供扩展的一些C++程序库的总称。Boost库是一个可移植、提供源代码的C++库,作为标准库的后备,是C++标准化进程的开发引擎之一,是为C++语言标准库提供扩展的一些C++程序库的总称
    2022-11-11
  • 利用C++模拟实现STL容器:list

    利用C++模拟实现STL容器:list

    列表是一种顺序容器,它允许在序列中的任何位置执行常量时间插入和删除操作,并允许在两个方向上进行迭代。本文将利用C++模拟实现list,希望对大家有所帮助
    2022-12-12
  • 详解VS2010实现创建并生成动态链接库dll的方法

    详解VS2010实现创建并生成动态链接库dll的方法

    在某些应用程序场景下,需要将一些类或者方法编译成动态链接库dll,以便别的.exe或者.dll文件可以通过第三方库的方式进行调用,下面就简单介绍一下如何通过VS2010来创建动态链接库
    2022-12-12

最新评论