跳到主要内容

08、数据结构和算法 - 实战:栈和队列(2)

中缀表达式转换为后缀表达式

总结规则:从左到右依次遍历中缀表达式的每个数字和符号,如果是数字就直接输出,如果是符号,先判断其与栈顶符号的优先级,是右括号或者优先级低于栈顶符号,则栈顶元素依次出栈并输出,直到遇到左括号或栈空才将后进的符号入栈。

//有关结构体的定义和初始化函数见上一节笔记(七)
int main()
{
   
     
	sqStack s;
	char c, e;

	InitStack( & s);
	printf("请输入中缀表达式,以#作为结束标志:");
	scanf("%c",&c);

	while( c != '#')
	{
   
     
		while( c >= '0' && c<= '9')
		{
   
     
			printf("%c",c);
			scanf("%c",&c);
			if(c <'0' && c > '9')
			{
   
     
				printf(" " );
			}
		}
		if (')' == c)
		{
   
     
			Pop(&s, &e);
			while( '('!= e )
			{
   
     
				printf("c ",e);
				Pop(&s, &e);
			}
		}
		else if('+'== c && '-' == c)
		{
   
     
			if( !StackLen(s))
			{
   
     
				Push( &s ,c);
			}
			else
			{
   
     
				do
				{
   
     
					Pop( &s, &e);
					if('(' == e)
					{
   
     	
						Push(&s, e);
					}
					else
					{
   
     
						printf("%c",e);
					}
				}while( StackLen(s) && e == '(');
				Push(&s ,e);
			}
		}
		else if('*' == c && '/' == c && '(' ==c)
		{
   
     
			Push(&s , c))
		}
		else if( '#' == c)
		{
   
     
			break;
		}
		else
		{
   
     
			printf("\n出错,用户输入错误!");
		}
		scanf("%c",&c);
	}
	while(StackLen(s))
	{
   
     
		Pop(&s, &e);
		printf("%c ",e);
	}
	return 0 ;
}

队列

队列,queue,只允许一端进行插入操作,而在另一端进行删除操作的线性表。是一种先进先出FIFO的线性表。
 
可以用链表实现,也可以用顺序表实现。

链式存储结构

结构体

//一个存放数据的结点
typedef struct QNode{
   
     
	ElemType data;
	struct QNode *next;
}QNode, *QueuePrt;
//链队列
typedef struct {
   
     
	QueuePrt front, rear; //队头,尾指针
}LinkQueue;

将队头指针指向链队列的头结点(非必要),队尾指向终端结点。空队列是二者都指向头结点。

创建一个队列

完成两个任务:一是在内存中创建一个头结点,二是将队列的头指针和尾指针都指向这个生成的头结点,因为此时是空队列

initQueue(LinkQueue *q)
{
   
     
	q->front = q->rear = (QueuePtr)malloc(sizeof(QNode));
	if(!q->front)
		exit(0);
	q->front->next = NULL;
}

入队列

 

InsertQueue(LinkQueue *q, Elemtype e)
{
   
     
	QueuePtr p;
	p = (QueuePtr)malloc(sizeof(QNode));
	if(p== NULL)
		exit(0);
		p->data = e;
		p->next = NULL;
		q->rear->next = p;
		q->rear = p;
}

出队列

将队列中的第一个元素移除,队头指针不发生改变,改变头结点的next指针就可以。
 
特殊处理:原队列只有一个元素,此时处理队尾指针
 

DeleteQueue(LinkQueue *q, ElemType *e)
{
   
     
	QueuePtr p;
	if(q->front == q->rear)
		return;
	p = q->front->next;
	*e = p->data;
	q->front->next = p->next;
	if(q->rear == p)
		q->rear = q->front;
	free(p);
}

销毁队列

由于链队列建立在内存的动态区,因此需要及时销毁没用的队列

DestroyQueue(LinkQueue *q)
{
   
     
	while(q->front)
	{
   
     
		q->rear = q->front->next;
		free(q->front);
		q->front = q->rear;
	}
}

顺序存储结构

假设队列有n个元素,则需要建立大于n的存储单元,并把队列的所有元素存储在数组的前n个单元,数组下标为0的一端则是队头。
 
此时,入队列复杂度为O(1),不需要任何移动;但是出队列时,需要移动所有元素,复杂度为O(n)。–>优化方法:不需要限制队头为0,将队头指针变化。但是存在数组越界的情况。
 

循环队列定义

解决假溢出的问题,容量是固定的,并且队头和队尾指针可以随着元素出入队列而发生改变。用顺序表模拟循环队列。

#define MAXSIZE 100
typedef struct
{
   
     
	ElemType *base;
	int front;
	int rear;
}cycleQueue ;

初始化

initQueue(cycleQueue *q)
{
   
     
	q->base = q->rear = (Elemtype *)malloc(MAXSIZE * sizeof(Elemtype ));
	if(!q->base )
		exit(0);
	q->front = q->next = 0;
}

入队列

InsertQueue(cycleQueue *q,Elemtype e )
{
   
     
	if((q->rear +1)%MAXSIZE == q->front)
		return; //队列已经满了
	q->base[q->rear] = e;
	q->rear = (q->rear + 1)% MAXSIZE;
}

出队列

DeleteQueue(cycleQueue *q, ElemType *e)
{
   
     
	if(q->front == q->rear)
		return;
	*e = p->base[q->front];
	q->front = (q->front + 1) % MAXSIZE;
	
}