/* * @Author: Xiyuan Yang xiyuan_yang@outlook.com * @Date: 2025-01-22 15:06:31 * @LastEditors: Xiyuan Yang xiyuan_yang@outlook.com * @LastEditTime: 2025-01-22 17:23:51 * @FilePath: \CODE_for_Vscode\Data_structure\Class_implementation\Stack.cpp * @Description: * Do you code and make progress today? * Copyright (c) 2025 by Xiyuan Yang, All Rights Reserved. */ /* This is the implementation of the stack */
#pragma once #include<stdexcept> #include<iostream> usingnamespace std;
//Abstract class template <classT> classqueue{ public: virtual ~queue(){} virtualboolisEmpty()const=0; virtual T getHead()const=0; virtual T pop()=0; virtualvoidpush(const T& value)=0; }; //抽象基类仅仅定义接口,没有具体的数据成员
//Sequential Implementation of stack template <classT> classseqQueue:public queue<T>{ private: T* elem;//the elements int maxsize; int front,rear; voidDoubleSpace();//Tool functions for space extensions public: seqQueue(int size=10); ~seqQueue(); boolisEmpty()const; voidpush(const T& value); T getHead()const; T pop();
//Only for debugging voidprint()const; intlength()const; };
template <classT> seqQueue<T>::seqQueue(int size):maxsize(size){ elem = new T[maxsize]; //but the maximum capacity is maxsize-1 front = rear =0; }
template <classT> void seqQueue<T>::push(const T& value){ //入队操作,需要对rear操作 if((rear+1) % maxsize == front){ //judge if the queue is full DoubleSpace(); }else{ elem[rear]=value; rear=(rear+1)%maxsize; } }
template <classT> T seqQueue<T>::pop(){ if (isEmpty()) { throw std::runtime_error("Queue is empty"); } //出队操作,对front操作 T temp = elem[front];//the top element of the queue front = (front+1)%maxsize; return temp; }
template <classT> void seqQueue<T>::DoubleSpace(){ //double the space of the queue T* tmp=elem; elem = new T[2*maxsize]; for(int i=0;i<maxsize-1;i++){ elem[i] = tmp[(front+i)%maxsize]; //you only need to copy maxsize-1 elements! } front = 0; rear = maxsize-1; maxsize*=2; delete[] tmp; }
template <classT> void seqQueue<T>::print()const { //print the whole queue if(isEmpty()){ cout<<"The queue is empty!"<<endl; return; }else{ cout<<"The queue has "<<length()<<" members."<<endl; int index = front; while(index != rear){ cout<<elem[index]<<" "; index = (index+1)%maxsize; } cout<<endl; } }
template <classT> int seqQueue<T>::length() const{ if(rear >= front) return rear-front; elsereturn rear+maxsize-front; } //The following function implements some small debugging features. intmain(){ seqQueue<int> list; list.print(); list.push(2); list.push(5); cout<<list.isEmpty()<<endl; list.print(); cout<<list.pop(); cout<<list.pop(); cout<<list.isEmpty()<<endl; return0; }
template <classT> classlinkQueue : public queue<T>{ private: structNode { T data; Node* next; //parameterized constructor(default) Node(const T& value = 0, Node *p = nullptr){ data = value; next = p; } //destructor ~Node(){} }; Node *front, *rear; //two pointers point at the front or the end of the queue. public: linkQueue(); //This constructor is used to create a new linked queue. //Todo: implement copy constructor and the overload of = (assignment) ~linkQueue(); boolisEmpty()const; T pop(); T getHead()const; voidpush(const T& value);
//Implementation of linked queue template <classT> linkQueue<T>::linkQueue(){ //set nullptr for both pointers front = rear = nullptr; //it is also the judgement of an empty queue }
template <classT> linkQueue<T>::~linkQueue(){ Node * temp; while(front != nullptr){ //simulate the process of leaving the queue until the queue is empty. temp = front; front = front -> next; delete temp; } }
template <classT> T linkQueue<T>::getHead() const{ if (isEmpty()) { throw std::runtime_error("Queue is empty"); } return front -> data; }
template <classT> void linkQueue<T>::push(const T& value){ if(isEmpty()){ front = rear = newNode(value); }else{ Node *newnode = newNode(value); rear -> next = newnode; rear = rear -> next; //! ensure the rear -> next often points to the nullptr } }
template <classT> T linkQueue<T>::pop(){ if (isEmpty()) { throw std::runtime_error("Queue is empty"); }else{ T ans = front -> data; Node* tmp = front; front = front -> next; if(front == nullptr){ rear = nullptr; //now the queue is empty! //This is necessary, because before the pop process, the front and the rear pointer points at the same space //now front moves to nullptr, rear pointer becomes INVALID! } delete tmp; return ans; } }