2. 큐의 구현

2023. 3. 7. 14:35·다양한 글들/자료구조와 알고리즘
선형 큐 (linear queue)

선형 큐

: 1차원 배열을 써서 큐를 구현하는 방법

  1. 1차원 배열을 정의하고 삽입, 삭제를 위한 변수인 front와 rear를 만든다
  2. front는 큐의 첫 번째 요소를 가리키고 rear는 큐의 마지막 요소를 가리킨다
  3. front와 rear의 초기값은 -1이다. 데이터가 증가되면 rear를 하나 증가하고 그 위치에 데이터가 저장된다.
  4. 삭제할 때도 front를 하나 증가하고 front가 가리키는 위치에 있는 데이터를 삭제한다


선형큐의 구현 ft. C언어
#include <stdio.h>
#include <stdlib.h>

#define MAX_QUEUE_SIZE 5

typedef int element;
typedef struct { 				// 큐 타입
	int  front;         // 큐의 맨 앞 원소 위치
	int rear;           // 큐의 맨 뒤 원소 위치
	element  data[MAX_QUEUE_SIZE];  // 원소를 저장하는 배열
} QueueType;

// 오류 함수
void error(char* message)
{
	fprintf(stderr, "%s\n", message);
	exit(1);
}

// 큐 초기화 함수
void init_queue(QueueType* q)
{
	q->rear = -1;   // rear를 -1로 초기화
	q->front = -1;  // front를 -1로 초기화
}

// 큐의 원소 출력 함수
void queue_print(QueueType* q)
{
	for (int i = 0; i < MAX_QUEUE_SIZE; i++) {
		if (i <= q->front || i > q->rear)  // 큐가 비어있거나 원소가 없는 부분은 |로 출력
			printf("   | ");
		else
			printf("%d | ", q->data[i]);  // 큐에 저장된 원소 출력
	}
	printf("\n");
}

// 큐가 포화상태인지 검사하는 함수
int is_full(QueueType* q)
{
	if (q->rear == MAX_QUEUE_SIZE - 1)  // rear가 배열의 마지막 원소인 경우 포화상태
		return 1;
	else
		return 0;
}

// 큐가 공백상태인지 검사하는 함수
int is_empty(QueueType* q)
{
	if (q->front == q->rear)  // front와 rear가 같은 경우 공백상태
		return 1;
	else
		return 0;
}

// 큐에 원소를 삽입하는 함수
void enqueue(QueueType* q, int item)
{
	if (is_full(q)) {   // 큐가 포화상태인 경우 오류 메시지 출력
		error("큐가 포화상태입니다.");
		return;
	}
	q->data[++(q->rear)] = item;  // rear를 1 증가시키고 원소를 삽입
}

// 큐에서 원소를 삭제하고 반환하는 함수
int dequeue(QueueType* q)
{
	if (is_empty(q)) {  // 큐가 공백상태인 경우 오류 메시지 출력
		error("큐가 공백상태입니다.");
		return -1;
	}
	int item = q->data[++(q->front)];  // front를 1 증가시키고 삭제할 원소를 저장
	return item;  // 삭제한 원소 반환
}

// main 함수
int main(void)
{
	int item = 0;
	QueueType q;

	init_queue(&q);  // 큐 초기화

	enqueue(&q, 10); queue_print(&q);
	enqueue(&q, 20); queue_print(&q);
	enqueue(&q, 30); queue_print(&q);

	item = dequeue(&q); queue_print(&q);
	item = dequeue(&q); queue_print(&q);
	item = dequeue(&q); queue_print(&q);
	return 0;

선형 큐의 응용 : 작업 스케줄링
  • 큐는 운영 체제에서도 사용된다.
  • 운영체제는 많은 작업들을 동시에 실행해야 한다.
  • 만약 CPU가 하나뿐이고 모든 작업들은 우선순위를 가지지 않는다고 가정하면 작업들은 운영 체제에 들어간 순서대로 처리될 것이다.
  • 이럴 때는 큐를 사용하여서 작업들을 처리할 수 있다


원형 큐
  • 선형큐는 이해하기는 쉽지만 문제점이 있다.
  • front와 rear의 값이 계속 증가만 하기 때문에 언젠가는 배열의 끝에 도달하게 되고 배열의 앞부분이 비어 있더라도 사용하지를 못한다는 점이다.
  • 따라서 주기적으로 모든 요소들을 왼쪽으로 이동시켜야 한다.

 

  • 큐는 다음과 같은 상태에 있을 수 있고 오른쪽에 삽입을 위한 공간을 만들기 위해서는 모든 요소들을 왼쪽으로 이동시켜야 한다.
  • 이런 식으로 요소들을 이동시키면 해결은 되지만 매번 이동시키려면 상당한 시간이 걸리고 또한 프로그램 코딩이 복잡하다.

 

  • 이 문제는 배열을 선형이 아닌 원형으로 생각하면 쉽게 해결된다.
  • 즉 front와 rear의 값이 배열의 끝인 (MAX_QUEUE_SIZE-1)에 도달하면 다음에 증가되는 값은 0이 되도록 하는 것이다.
  • 즉 배열이 원형으로 처음과 끝이 연결되어 있다고 생각하는 것이다.
  • 여기서 실제 배열이 원형으로 변화되는 것은 아니다.
  • 그냥 개념상으로 원형으로 배열의 인덱스를 변화시켜주는 것뿐이다.

 

  • 원형큐에서는 front와 rear의 개념이 약간 변경된다.
  • 먼저 초기값은 -1이 아닌 0이다
  • front는 항상 큐의 첫 번째 요소의 하나 앞을, rear는 마지막 요소를 가리킨다
  • 처음에 front, rear는 모두 0이고 삽입 시에는 rear가 먼저 증가되고, 증가된 위치에 새로운 데이터가 삽입된다.
  • 또 삭제 시에도 front가 먼저 증가된 다음, 증가된 위치에서 데이터를 삭제한다.

 

  • front와 rear의 값이 같으면 원형 큐가 비어 있음을 나타낸다
  • 포화 상태와 공백 상태를 구별하기 위해서 원형큐에서는 하나의 자리는 항상 비워둔다(자리를 비워두지 않으면 오류 상태)
  • 공백 상태 : front==rear
  • 포화 상태 : front가 rear보다 하나 앞에 있는 상태

*만약 요소들의 개수를 저장하고 있는 추가적인 변수 count 변수를 사용할 수 있다면 한자리를 비워두지 않아도 된다.


원형 큐의 삽입, 삭제 알고리즘

삽입이나 삭제를 하기 전에 front나 rear를 원형 회전시켜서 하나 증가시키고 증가된 위치에 데이터를 삽입 또는 삭제한다.

원형큐의 구현에 있어서 중요한 것은 front와 rear를 원형으로 회전시켜야 한다는 것이고 이는 나머지 연산자 %를 이용하여 쉽게 구현이 가능하다

front <- (front+1) % MAX_QUEUE_SIZE
rear <- (rear+1) % MAX_QUEUE_SIZE

 front와 rear 값은 (MAX_QUEUE_SIZE-1)에서 하나 증가되면 0으로 된다.

즉 MAX_QUEUE_SIZE를 5로 정의하면, front와 rear 값은 0, 1, 2, 3, 4, 0과 같이 변화된다.


원형큐의 구현 ft. C언어
#include <stdio.h>
#include <stdlib.h>

#define MAX_QUEUE_SIZE 5
typedef int element;
typedef struct { // 큐 타입
    element data[MAX_QUEUE_SIZE];
    int front, rear;
} QueueType;

void error(char* message) // 오류 메시지 출력 함수
{
    fprintf(stderr, "%s\n", message);
    exit(1);
}

void init_queue(QueueType* q) // 큐 초기화 함수
{
    q->front = q->rear = 0;
}

int is_empty(QueueType* q) // 큐 공백 상태 검출 함수
{
    return (q->front == q->rear);
}

int is_full(QueueType* q) // 큐 포화 상태 검출 함수
{
    return ((q->rear + 1) % MAX_QUEUE_SIZE == q->front);
}

void queue_print(QueueType* q) // 큐 출력 함수
{
    printf("QUEUE(front=%d rear=%d) = ", q->front, q->rear);
    if (!is_empty(q)) {
        int i = q->front;
        do {
            i = (i + 1) % (MAX_QUEUE_SIZE);
            printf("%d | ", q->data[i]);
            if (i == q->rear)
                break;
        } while (i != q->front);
    }
    printf("\n");
}

void enqueue(QueueType* q, element item) // 큐 삽입 함수
{
    if (is_full(q))
        error("큐가 포화상태입니다");
    q->rear = (q->rear + 1) % MAX_QUEUE_SIZE;
    q->data[q->rear] = item;
}

element dequeue(QueueType* q) // 큐 삭제 함수
{
    if (is_empty(q))
        error("큐가 공백상태입니다");
    q->front = (q->front + 1) % MAX_QUEUE_SIZE;
    return q->data[q->front];
}

element peek(QueueType* q) // 큐에서 가장 먼저 들어온 데이터 반환 함수
{
    if (is_empty(q))
        error("큐가 공백상태입니다");
    return q->data[(q->front + 1) % MAX_QUEUE_SIZE];
}

int main(void)
{
    QueueType queue;
    int element;

    init_queue(&queue); // 큐 초기화
    printf("--데이터 추가 단계--\n");
    while (!is_full(&queue)) // 큐가 포화 상태가 될 때까지 데이터 삽입
    {
        printf("정수를 입력하시오: ");
        scanf("%d", &element);
        enqueue(&queue, element);
        queue_print(&queue);
    }
    printf("큐는 포화상태입니다.\n\n");

    printf("--데이터 삭제 단계--\n");
    while (!is_empty(&queue)) // 큐가 공백 상태가 될 때까지 데이터 삭제
    {
        element = dequeue(&queue);
        printf("꺼내진 정수: %d \n", element);
        queue_print(&queue);
    }
    printf("큐는 공백상태입니다.\n");
    return 0;
}


원형큐의 구현 ft. C++

CircularQueue.h

#pragma once
#include <iostream>
#include <cstdio>
#include <cstdlib>

const int MAX_QUEUE_SIZE = 100;

inline void error(const char* message) {
	printf("%s\n", message);
	exit(1);
}

template <typename T>
class CircularQueue {
protected:
	int _front;
	int _rear;
	T data[MAX_QUEUE_SIZE];
public:
	CircularQueue() : _front(0), _rear(0) {}
	void enqueue(const T& value);
	T dequeue();
	T peek();
	void display();
	bool isEmpty() { return _front == _rear; }
	bool isFull() { return (_rear + 1) % MAX_QUEUE_SIZE == _front; }
};

template <typename T>
void CircularQueue<T>::enqueue(const T& value) {
	if (isFull()) error("큐가 가득참!\n");
	else {
		_rear = (_rear + 1) % MAX_QUEUE_SIZE;
		data[_rear] = value;
	}
}

template <typename T>
T CircularQueue<T>::dequeue() {
	if (isEmpty()) error("큐가 비어있음!\n");
	else {
		_front = (_front + 1) % MAX_QUEUE_SIZE;
		return data[_front];
	}
}

template <typename T>
T CircularQueue<T>::peek() {
	if (isEmpty()) error("큐가 비어있음!\n");
	else {
		return data[(_front + 1) % MAX_QUEUE_SIZE];
	}
}

template <typename T>
void CircularQueue<T>::display() {
	std::cout << "[queue]: ";
	int max = (_front < _rear) ? _rear : _rear + MAX_QUEUE_SIZE;
	for (int i = _front + 1; i < max; i++)
	{
		std::cout << data[i] << " ";
	}
	std::cout << std::endl;
}

 

main.cpp

#include "CircularQueue.h"
using namespace std;

int main()
{
	CircularQueue<double> q;
	for (double i = 0.5; i < 15; i++)
		q.enqueue(i);
	q.display();

	q.dequeue();
	q.dequeue();
	q.dequeue();
	q.dequeue();

	q.display();

	return 0;
}
저작자표시 (새창열림)

'다양한 글들 > 자료구조와 알고리즘' 카테고리의 다른 글

1. 트리  (0) 2023.03.07
3. 덱 (deque)  (0) 2023.03.07
1. 큐 ADT  (0) 2023.03.07
3. 스택의 응용  (0) 2023.03.07
2. 스택의 구현  (0) 2023.03.07
'다양한 글들/자료구조와 알고리즘' 카테고리의 다른 글
  • 1. 트리
  • 3. 덱 (deque)
  • 1. 큐 ADT
  • 3. 스택의 응용
smile blog
smile blog
건국대 첨단바이오공학부 & 컴퓨터공학부 BT & IT 기술로 희망을 꿈 꿉니당
  • smile blog
    스마일 블로그
    smile blog
  • 전체
    오늘
    어제
    • 분류 전체보기 (822)
      • 일상 생각들 (3)
      • 학과에 대해 (4)
        • 첨단바이오공학부 (4)
        • 컴퓨터공학부 (0)
      • -------- 프로젝트 -------- (0)
      • [DS] 토이 프로젝트 (1)
      • [Web, Game, XR] 토이 프로젝트 (11)
      • 경진대회 (1)
      • -------- 진로 -------- (0)
      • 생물정보학자 (18)
        • 데이터 과학이란? (0)
        • 되는 방법 (8)
        • 책 추천 (2)
        • 인강 (1)
        • 대학 (2)
        • 회사 (1)
        • 학원 (2)
        • 학회 (2)
      • 디지털 헬스케어 (72)
        • 방법 (8)
        • 생각들 (10)
        • 공부법 (4)
        • 책 추천 (2)
        • 학원 (2)
        • 참고 (2)
        • 대학 (3)
        • 회사 (3)
        • 인강 (2)
        • 게임 엔진들 (1)
        • 게임 프로그래머 개론 (2)
        • 게임 프로그래머 취업 전략 가이드 (7)
        • 취업 서류 (1)
        • 애정하는 게임들 (4)
        • XR 테크니컬 아티스트 (9)
        • 영화, 애니메이션 테크니컬 디렉터 (12)
      • -------- 기초 학문 -------- (0)
      • 생명과학 이야기 (2)
        • 대학 강의 (2)
      • 화학 이야기 (0)
      • 컴퓨터과학 이야기 (0)
      • 통계학 이야기 (0)
      • 수학 이야기 (1)
        • 공학 수학 (1)
      • 영어 이야기 (1)
      • 심리학 이야기 (7)
        • 현대인과 정신건강 (7)
      • -------- 컴퓨터 언어 -------- (0)
      • Python (3)
        • 나도코딩의 파이썬 입문 (1)
        • 파이썬 관련 정보 (1)
      • SQL (0)
      • C 언어 (32)
        • 혼자 공부하는 C언어 요약 (1)
        • [책 정리] 혼자 공부하는 C언어 (31)
      • C++ (33)
        • 명품 C++ 프로그래밍 요약 (1)
        • [책 정리] 명품 C++ 프로그래밍 (27)
        • C++ STL (0)
        • 뇌를 자극하는 C++ STL (5)
      • -------- 생명과학 -------- (0)
      • 생화학 (5)
        • 대학 강의 (5)
      • 분자세포생물학 (3)
        • 대학 강의 (3)
      • 유전자치료공학 (2)
        • 대학 강의 (2)
      • 생명정보학 (5)
        • 대학 강의 (5)
      • 약리학 (2)
        • 대학 강의 (2)
      • -------- 컴퓨터과학 -------- (0)
      • 자료구조와 알고리즘 (8)
        • 자료구조와 알고리즘의 정의 (3)
        • [책 정리] C언어로 쉽게 풀어쓴 자료구조 요약 (1)
        • [인강] 자료구조와 알고리즘 (2)
        • 코딩 테스트 대비하기! (1)
      • 컴퓨터 회로 (0)
      • 컴퓨터 구조 (43)
        • 컴퓨터 구조와 운영체제 요약 (1)
        • ---------------------------------------- (0)
        • [전공 책 정리] 컴퓨터 구조 및 설계 (1)
        • Ch1. 컴퓨터 추상화 및 관련 기술 (8)
        • Ch2. 명령어 : 컴퓨터 언어 (11)
        • Ch3. 컴퓨터 연산 (8)
        • Ch4. 프로세서 (11)
        • Ch5. 메모리 계층구조 (3)
        • Ch6. 병렬 프로세서 : 클라이언트에서 클라우드까지 (0)
      • 시스템 프로그래밍 (15)
        • [책 정리] 시스템 프로그래밍 유닉스 & 리눅스 (0)
        • [인강] 리눅스 시스템 프로그래밍 (2)
        • 리눅스에서 코딩이란? (8)
        • 대학교 강의 정리 (5)
      • 운영체제 (0)
      • 컴퓨터 네트워크 (37)
        • 모두의 네트워크 요약 (1)
        • [책 정리] 모두의 네트워크 (10)
        • ---------------------------------------- (0)
        • [전공 책 정리] 컴퓨터 네트워킹 하향식 접근 8판 (1)
        • Ch1. 컴퓨터 네트워크와 인터넷 (7)
        • Ch2. 애플리케이션 계층 (7)
        • Ch3. 트랜스포트 계층 (8)
        • Ch4. 네트워크 계층 : 데이터 평면 (3)
        • Ch5. 네트워크 계층 : 제어 평면 (0)
        • Ch6. 링크 계층과 근거리 네트워크 (0)
        • Ch7. 무선 및 이동 네트워크 (0)
        • Ch8. 컴퓨터 네트워크 보안 (0)
      • 데이터베이스 (1)
      • -------- 데이터과학 -------- (0)
      • 데이터 사이언스 (8)
        • 인강 (8)
      • 데이터 분석 (2)
        • 인강 (2)
      • 머신러닝 (2)
        • 대학 수업 (2)
      • 인공지능 (11)
        • 대학교 강의 정리 (10)
        • 인공지능 관련 정보 (1)
      • -------- +a -------- (0)
      • Visual Studio Community (7)
        • 설치법 (1)
        • 단축키 (1)
        • 오류 (5)
      • Visual Studio Code (0)
      • 노션 (1)
      • 깃허브 (7)
        • 깃허브 사용법 (5)
        • 유니티, 언리얼 & 깃허브 (1)
        • 깃허브 주의사항 (1)
      • 챗GPT 활용법 (0)
      • 기타 feat. 프로그래밍 (7)
        • 프로그래머로 살아남기 (5)
        • 코딩 vs 프로그래밍 (1)
        • 애플 비전 프로 (1)
      • 메타버스 (5)
      • -------- 예술 -------- (0)
      • 음악 (1)
      • 미술 (0)
      • -------- XR -------- (0)
      • 유니티 이야기 (23)
        • 레트로의 유니티 게임 프로그래밍 에센스 요약 (4)
        • 유니티 관련 정보 (1)
        • 유니티 디버깅 (13)
        • 유니티 인강 (3)
        • 대학교 게임 프로그래밍 강의 (2)
      • 언리얼 이야기 (0)
        • 인생 언리얼 교과서 요약 (0)
      • 컴퓨터 그래픽스 (6)
        • OpenGL (6)
      • 가상현실 & 증강현실 (4)
        • 유니티 vr (4)
      • HCI 와 UI UX (7)
        • [책 정리] HCI 개론 (6)
      • -------- Design -------- (0)
      • 캐릭터 (1)
        • 모델링 (0)
        • 리깅 (1)
      • 포토샵 (3)
      • 3ds Max (7)
      • Maya (9)
        • 블로그 (1)
        • 인강 (6)
        • 대학교 (2)
      • Blender (14)
        • 책 (1)
        • 인강 (7)
        • 기타 (3)
        • 대학교 (3)
      • 아트 작업물들 (2)
      • 에셋 사이트 (1)
      • -------- 건강관리 -------- (0)
      • 건강관리 ft. 정현 (12)
        • 목 디스크 (2)
        • 눈 관리 (2)
        • 일상생활 습관 (6)
        • 일상생활 꿀팁 (2)
        • 사무직 꿀팁 (0)
      • 헬스의 정석 ft. 정현 (28)
        • 헬스와 건강 (8)
        • 헬스 구체화 정보 (6)
        • 헬스 유튜버 (1)
        • 헬스 서적 (1)
        • 도전 바디프로필! (11)
        • 헬스장 패션 (1)
      • -------- etc -------- (0)
      • 진로 관련 잡다한 글들 (34)
        • 진도율 (9)
        • 진로 관련 글들 (15)
        • 학교 강의 관련 글들 (10)
      • 인생 꿀 Tip (23)
        • 컴퓨터 초기 설정 (9)
        • 원격 데스크톱 (1)
        • 노트북 발열 (1)
        • 전자기기 (2)
        • 중고기기 팔기 (1)
        • 아이패드 필기 어플 (1)
        • 에어팟 (1)
        • 커피 (1)
        • 맥북 (1)
        • lg 그램 (1)
        • 검색엔진에서 내 티스토리 검색 (1)
        • hELLO 다크 모드 없애기 (1)
        • 인터넷 연결 문제 (1)
        • 키보드 문제 해결 (1)
      • 유튜브 (3)
      • 청춘 그리고 추억 (1)
      • 인생 계획표 (2)
        • 2024년 2학기 (1)
        • 2024년 여름방학 (0)
        • 2024년 1학기 (0)
        • 2023년 겨울방학 (1)
      • 다양한 글들 (98)
        • C++ STL (6)
        • Win32 API (24)
        • PushPush 게임 (13)
        • 컴퓨터구조 (1)
        • 자료구조와 알고리즘 (50)
        • 게임의 정의 (3)
        • 영상 회사 (1)
  • 블로그 메뉴

    • 홈
    • 태그
    • 방명록
  • 링크

  • 공지사항

    • Dream
    • 코딩을 시작한 이유
    • 나를 소개합니다!
    • 블로그 공부법
    • IT & 가치 있는 일들
  • 인기 글

  • 태그

    유니티
    블렌더
    건국대
    C++
    데이터과학
    컴퓨터구조
    첨단바이오공학부
    연산자
    인공지능
    리눅스 터미널
    코드잇
    의생명공학
    생명과학
    함수
    리눅스
    의생명공학과
    C언어
    AI
    자료구조
    생명공학
    생물정보학
    데이터사이언스
    포인터
    스택
    명령어
    알고리즘
    배열
    unity
    심리학
    컴퓨터 네트워크
  • 최근 댓글

  • hELLO· Designed By정상우.v4.10.3
smile blog
2. 큐의 구현
상단으로

티스토리툴바