728x90

DFS와 BFS 성공

 
 
시간 제한메모리 제한제출정답맞힌 사람정답 비율
2 초 128 MB 275831 107217 63691 37.566%

문제

그래프를 DFS로 탐색한 결과와 BFS로 탐색한 결과를 출력하는 프로그램을 작성하시오. 단, 방문할 수 있는 정점이 여러 개인 경우에는 정점 번호가 작은 것을 먼저 방문하고, 더 이상 방문할 수 있는 점이 없는 경우 종료한다. 정점 번호는 1번부터 N번까지이다.

입력

첫째 줄에 정점의 개수 N(1 ≤ N ≤ 1,000), 간선의 개수 M(1 ≤ M ≤ 10,000), 탐색을 시작할 정점의 번호 V가 주어진다. 다음 M개의 줄에는 간선이 연결하는 두 정점의 번호가 주어진다. 어떤 두 정점 사이에 여러 개의 간선이 있을 수 있다. 입력으로 주어지는 간선은 양방향이다.

출력

첫째 줄에 DFS를 수행한 결과를, 그 다음 줄에는 BFS를 수행한 결과를 출력한다. V부터 방문된 점을 순서대로 출력하면 된다.

예제 입력 1 복사

4 5 1
1 2
1 3
1 4
2 4
3 4

예제 출력 1 복사

1 2 4 3
1 2 3 4

예제 입력 2 복사

5 5 3
5 4
5 2
1 2
3 4
3 1

예제 출력 2 복사

3 1 2 5 4
3 1 4 2 5

예제 입력 3 복사

1000 1 1000
999 1000

예제 출력 3 복사

1000 999
1000 999

 

 

bfs는 큐로, dfs는 재귀(스택)으로 출력할 값들을 저장하면 된다.

 

#include <iostream>
#include <queue>
using namespace std;
#define MAX_VERTEX 1001
short graph[MAX_VERTEX][MAX_VERTEX];
char check_dfs[MAX_VERTEX];
char check_bfs[MAX_VERTEX];

int n ,m, v;
int from, to;

void dfs(int node){
    if(check_dfs[node]){
        return;
    } else{
        check_dfs[node] = 1;
        cout << node << " ";
        for(int i = 1; i <= n; i++){
            if(graph[node][i] && !check_dfs[i])
            dfs(i);
        }
    }

}

void bfs(int node){
    queue<short> q;
    int temp = 0;

    q.push(node);
    check_bfs[node] = 1;

    
    while(!q.empty()){

        temp = q.front();
        q.pop();
        cout << temp << " ";


        for(int i = 1; i <= n; i++){
            if(graph[temp][i] && !check_bfs[i]){
                q.push(i);
                check_bfs[i] = 1;
            }
        }

    }
}

int main(){
    
    cin >> n >> m >> v;

    for(int i = 0 ; i < m ; i++){
        cin >> from >> to;
        graph[from][to]=1;
        graph[to][from]=1;
    }
    dfs(v);
    cout << "\n";
    bfs(v);

    return 0;
}

 

vertex(정점)과 node는 같은 개념으로 사용하였다.

 

또한, bfs에서 같은 깊이에 있는 노드들을 작은 수부터 출력하라 하였는데,  애초에 다음 노드 탐색을 for문을 이용하여 1번 노드부터 n번 노드까지 탐색하고 push하였으므로 정렬할 필요는 없다.

728x90
728x90

연결 요소의 개수 성공

 
 
시간 제한메모리 제한제출정답맞힌 사람정답 비율
3 초 512 MB 125970 56820 37348 42.054%

문제

방향 없는 그래프가 주어졌을 때, 연결 요소 (Connected Component)의 개수를 구하는 프로그램을 작성하시오.

입력

첫째 줄에 정점의 개수 N과 간선의 개수 M이 주어진다. (1 ≤ N ≤ 1,000, 0 ≤ M ≤ N×(N-1)/2) 둘째 줄부터 M개의 줄에 간선의 양 끝점 u와 v가 주어진다. (1 ≤ u, v ≤ N, u ≠ v) 같은 간선은 한 번만 주어진다.

출력

첫째 줄에 연결 요소의 개수를 출력한다.

예제 입력 1 복사

6 5
1 2
2 5
5 1
3 4
4 6

예제 출력 1 복사

2

예제 입력 2 복사

6 8
1 2
2 5
5 1
3 4
4 6
5 4
2 4
2 3

 

예제 출력 2 복사

1

 

 

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>

#define MAX_VERTEX 1001

short graph[MAX_VERTEX][MAX_VERTEX]; // adjacency matrix
char check[MAX_VERTEX]; // check visitied node


void dfs(int i,int vertex_num) {
	
	check[i] = 1;
	for (int k = 1; k <= vertex_num; k++) {
		if(graph[i][k] && !check[k])
			dfs(i, k, vertex_num);
	}
	
}

int main() {
	int vertex_num, edge_num;
	int from, to;
	scanf("%d %d", &vertex_num, &edge_num);

	for (int i = 0; i < edge_num; i++) {
		scanf("%d %d", &from, &to);
		graph[from][to] = 1;
		graph[to][from] = 1;  //no direction
	}

	int count = 0;

	for (int i = 1; i <= vertex_num; i++) {
		
		if (!check[i]) {
			count++;
			dfs(i, vertex_num);
		}
		
	}

	printf("%d", count);

}

 

그래프 유형의 문제가 처음이라 꽤 애먹었는데, 

집합의 개수를 살릴때 그래프 기준이 아니라, 노드를 기준으로 count 해야하고, bfs 혹은 dfs는 연결되어있는 노드에 방문하기 위한 알고리즘이며 어떤 순서로 방문할껀지에 따라 알고리즘이 나뉜다.

728x90
728x90

유기농 배추 성공

 
 
시간 제한메모리 제한제출정답맞힌 사람정답 비율
1 초 512 MB 180934 73195 48816 38.215%

문제

차세대 영농인 한나는 강원도 고랭지에서 유기농 배추를 재배하기로 하였다. 농약을 쓰지 않고 배추를 재배하려면 배추를 해충으로부터 보호하는 것이 중요하기 때문에, 한나는 해충 방지에 효과적인 배추흰지렁이를 구입하기로 결심한다. 이 지렁이는 배추근처에 서식하며 해충을 잡아 먹음으로써 배추를 보호한다. 특히, 어떤 배추에 배추흰지렁이가 한 마리라도 살고 있으면 이 지렁이는 인접한 다른 배추로 이동할 수 있어, 그 배추들 역시 해충으로부터 보호받을 수 있다. 한 배추의 상하좌우 네 방향에 다른 배추가 위치한 경우에 서로 인접해있는 것이다.

한나가 배추를 재배하는 땅은 고르지 못해서 배추를 군데군데 심어 놓았다. 배추들이 모여있는 곳에는 배추흰지렁이가 한 마리만 있으면 되므로 서로 인접해있는 배추들이 몇 군데에 퍼져있는지 조사하면 총 몇 마리의 지렁이가 필요한지 알 수 있다. 예를 들어 배추밭이 아래와 같이 구성되어 있으면 최소 5마리의 배추흰지렁이가 필요하다. 0은 배추가 심어져 있지 않은 땅이고, 1은 배추가 심어져 있는 땅을 나타낸다.

1 1 0 0 0 0 0 0 0 0
0 1 0 0 0 0 0 0 0 0
0 0 0 0 1 0 0 0 0 0
0 0 0 0 1 0 0 0 0 0
0 0 1 1 0 0 0 1 1 1
0 0 0 0 1 0 0 1 1 1

입력

입력의 첫 줄에는 테스트 케이스의 개수 T가 주어진다. 그 다음 줄부터 각각의 테스트 케이스에 대해 첫째 줄에는 배추를 심은 배추밭의 가로길이 M(1 ≤ M ≤ 50)과 세로길이 N(1 ≤ N ≤ 50), 그리고 배추가 심어져 있는 위치의 개수 K(1 ≤ K ≤ 2500)이 주어진다. 그 다음 K줄에는 배추의 위치 X(0 ≤ X ≤ M-1), Y(0 ≤ Y ≤ N-1)가 주어진다. 두 배추의 위치가 같은 경우는 없다.

출력

각 테스트 케이스에 대해 필요한 최소의 배추흰지렁이 마리 수를 출력한다.

예제 입력 1 복사

2
10 8 17
0 0
1 0
1 1
4 2
4 3
4 5
2 4
3 4
7 4
8 4
9 4
7 5
8 5
9 5
7 6
8 6
9 6
10 10 1
5 5

예제 출력 1 복사

5
1

예제 입력 2 복사

1
5 3 6
0 2
1 2
2 2
3 2
4 2
4 0

예제 출력 2 복사

2

 

 

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>
#include <memory.h>
int global_n, global_m;

void record_history_dfs(int** arr, int x, int y) {
	int new_x, new_y;
	arr[y][x] = 0;
	// x ,y+1
	new_x = x;
	new_y = y + 1;
	if (new_y < global_n && arr[new_y][new_x]) {
		record_history_dfs(arr, new_x, new_y);
	}

	// x ,y-1
	new_x = x;
	new_y = y - 1;
	if (new_y >= 0 && arr[new_y][new_x]) {
		record_history_dfs(arr, new_x, new_y);
	}
	// x+1 ,y
	new_x = x + 1;
	new_y = y;
	if (new_x < global_m && arr[new_y][new_x]) {
		record_history_dfs(arr, new_x, new_y);
	}
	// x-1 ,y
	new_x = x - 1;
	new_y = y;
	if (new_x >= 0 && arr[new_y][new_x]) {
		record_history_dfs(arr, new_x, new_y);
	}
}

	int count(int** arr, int m, int n) {
		int total_count = 0;
		for (int i = 0; i < n; i++) {
			for (int j = 0; j < m; j++) {
				if (arr[i][j]) {
					record_history_dfs(arr, j, i);
					total_count++;
				}
			}
		}
		return total_count;
	}

	int main() {
		int size;
		int n, m, k;
		int x, y;
		int** arr = NULL;
		scanf("%d", &size);
		for (int i = 0; i < size; i++) {
			scanf("%d %d %d", &m, &n, &k);
			global_m = m;
			global_n = n;
			arr = (int**)malloc(sizeof(int*) * n);
			for (int j = 0; j < n; j++) {
				arr[j] = (int*)malloc(sizeof(int) * m);
			}
			for (int j = 0; j < n; j++) {
				for (int k = 0; k < m; k++) {
					arr[j][k] = 0;
				}
			}

			for (int j = 0; j < k; j++) {
				scanf("%d %d", &x, &y);
				arr[y][x] = 1;
			}

			printf("%d\n", count(arr, m, n));

		}
	}

재귀적으로 호출하는 dfs 알고리즘을 사용하였다.

 

우선 1을 만나면, 해당 좌표 기준 상, 하, 좌 , 우 좌표가 바운더리 내에 존재하는지 확인 후 해당 좌표도 1이라면, 모두 0으로 바꿔주는 작업을 한 후, count를 하는 방향으로 설계하였다.

728x90
728x90

피보나치 함수 성공

 
 
시간 제한메모리 제한제출정답맞힌 사람정답 비율
0.25 초 (추가 시간 없음) 128 MB 213569 65746 51877 33.114%

문제

다음 소스는 N번째 피보나치 수를 구하는 C++ 함수이다.

int fibonacci(int n) {
    if (n == 0) {
        printf("0");
        return 0;
    } else if (n == 1) {
        printf("1");
        return 1;
    } else {
        return fibonacci(n‐1) + fibonacci(n‐2);
    }
}

fibonacci(3)을 호출하면 다음과 같은 일이 일어난다.

  • fibonacci(3)은 fibonacci(2)와 fibonacci(1) (첫 번째 호출)을 호출한다.
  • fibonacci(2)는 fibonacci(1) (두 번째 호출)과 fibonacci(0)을 호출한다.
  • 두 번째 호출한 fibonacci(1)은 1을 출력하고 1을 리턴한다.
  • fibonacci(0)은 0을 출력하고, 0을 리턴한다.
  • fibonacci(2)는 fibonacci(1)과 fibonacci(0)의 결과를 얻고, 1을 리턴한다.
  • 첫 번째 호출한 fibonacci(1)은 1을 출력하고, 1을 리턴한다.
  • fibonacci(3)은 fibonacci(2)와 fibonacci(1)의 결과를 얻고, 2를 리턴한다.

1은 2번 출력되고, 0은 1번 출력된다. N이 주어졌을 때, fibonacci(N)을 호출했을 때, 0과 1이 각각 몇 번 출력되는지 구하는 프로그램을 작성하시오.

입력

첫째 줄에 테스트 케이스의 개수 T가 주어진다.

각 테스트 케이스는 한 줄로 이루어져 있고, N이 주어진다. N은 40보다 작거나 같은 자연수 또는 0이다.

출력

각 테스트 케이스마다 0이 출력되는 횟수와 1이 출력되는 횟수를 공백으로 구분해서 출력한다.

예제 입력 1 복사

3
0
1
3

예제 출력 1 복사

1 0
0 1
1 2

예제 입력 2 복사

2
6
22

예제 출력 2 복사

5 8
10946 17711

 

해당 문제는 다이나믹 프로그래밍 알고리즘 문제로, 탑다운 혹은 바텀업 방식을 선택할 수 있다.

특징으론, 시간이 0.25초로 매우 빠듯하게 설정되어 있으므로, 효율적이로 구현해야 한다.

문제에서 n의 범위가 40까지밖에 되지 않으므로, 바텀업 방식으로 간단하게 구현하였다.

 

#include <iostream>
#include <stdlib.h>
using namespace std;
int main(){
    int one[41] = {0,};
    int zero[41] = {0,};

    zero[0] = 1;
    one[0] = 0;

    zero[1] = 0;
    one[1] = 1;

    zero[2] = 1;
    one[2] = 1;

    for(int i = 3; i <= 40; i++){
        zero[i] = zero[i-1] + zero[i-2];
        one[i] = one[i-1] + one[i-2];
    }

    int size = 0;
    cin >> size;
    int* user_input = (int*)malloc(sizeof(int)*size);
    for(int i = 0; i < size; i++)
        cin >> user_input[i];
    for(int i = 0; i < size; i++)
        cout << zero[user_input[i]] << " " << one[user_input[i]] << "\n";
}

 

인덱스가 2인 경우 또한 for문에 넣어도 되지만, 이해를 돕고자 위처럼 분리하여 선언하였다.

 

결국 n도 n-1, n-2로 분리되므로 1의 개수와 0의 개수는 수마다 고유하다고 생각하면 된다.

728x90
728x90

통계학 

 
 
시간 제한메모리 제한제출정답맞힌 사람정답 비율
2 초 256 MB 160120 38733 31053 26.358%

문제

수를 처리하는 것은 통계학에서 상당히 중요한 일이다. 통계학에서 N개의 수를 대표하는 기본 통계값에는 다음과 같은 것들이 있다. 단, N은 홀수라고 가정하자.

  1. 산술평균 : N개의 수들의 합을 N으로 나눈 값
  2. 중앙값 : N개의 수들을 증가하는 순서로 나열했을 경우 그 중앙에 위치하는 값
  3. 최빈값 : N개의 수들 중 가장 많이 나타나는 값
  4. 범위 : N개의 수들 중 최댓값과 최솟값의 차이

N개의 수가 주어졌을 때, 네 가지 기본 통계값을 구하는 프로그램을 작성하시오.

입력

첫째 줄에 수의 개수 N(1 ≤ N ≤ 500,000)이 주어진다. 단, N은 홀수이다. 그 다음 N개의 줄에는 정수들이 주어진다. 입력되는 정수의 절댓값은 4,000을 넘지 않는다.

출력

첫째 줄에는 산술평균을 출력한다. 소수점 이하 첫째 자리에서 반올림한 값을 출력한다.

둘째 줄에는 중앙값을 출력한다.

셋째 줄에는 최빈값을 출력한다. 여러 개 있을 때에는 최빈값 중 두 번째로 작은 값을 출력한다.

넷째 줄에는 범위를 출력한다.

예제 입력 1 복사

5
1
3
8
-2
2

예제 출력 1 복사

2
2
1
10

예제 입력 2 복사

1
4000

예제 출력 2 복사

4000
4000
4000
0

예제 입력 3 복사

5
-1
-2
-3
-1
-2

예제 출력 3 복사

-2
-2
-1
2

예제 입력 4 복사

3
0
0
-1

예제 출력 4 복사

0
0
0
1

(0 + 0 + (-1)) / 3 = -0.333333... 이고 이를 첫째 자리에서 반올림하면 0이다. -0으로 출력하면 안된다.

 

 

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <math.h>


int compare(const void* a, const void* b) {
	int int_a = *(int*)a;
	int int_b = *(int*)b;

	if (int_a < int_b) return -1;
	else if (int_a > int_b) return 1;
	else return 0;
}

int mean(int* arr, int size) {
	double sum = 0; 
	for (int i = 0; i < size; i++) {
		sum += arr[i];
	}
	return (int)round(sum/size);
}

int median(int* arr, int size) {
	int indexA = 0, indexB = 0;
	int temp[2] = { 0 };
	//sort first

	if (size % 2 == 0) { //even case
		indexA = size / 2 - 1;
		indexB = size / 2;
		temp[0] = arr[indexA];
		temp[1] = arr[indexB];
		return mean(temp, 2);

	}
	else {//odd case
		indexA = size / 2;
		return arr[indexA];
	}
}



int most_num(int* arr, int size) {
	int maxCount = 0, mode = arr[0], count = 1, modeCount = 0; //maxCount : 현재 최고 빈도수, mode = 현재 최빈값, count = 현재 반복수, modeCount = 현재 최빈값 개수

	for (int i = 1; i < size; i++) {
		if (arr[i] ==  arr[i - 1]) { //같은 원소를 발견하면
			count++;  //count + 1
		}
		else { //다른 원소를 발견하면
			if (count > maxCount) { //만약, 빈도수가 더 큰값이면
				maxCount = count; //최고 빈도수를 업데이트
				mode = arr[i - 1]; //최고 빈도수를 가지는 값을 업데이트
				modeCount = 1; //최고 빈도수를 1개로 업데이트 
			}
			else if (count == maxCount) { //같은 빈도수를 가지는 값일 경우, 특히 빈도수가 현재 최고 빈도수값이랑 같을 경우
				if (modeCount == 1) { //최고 빈도수의 개수가 하나일 경우에만
					mode = arr[i - 1]; //최고 빈도수 값을 업데이트, 왜냐하면 두번째로 작은 수를 기록해야기 때문
					modeCount++; //최고 빈도수를 가지는 값의 개수를 업데이트, 2개 이상이므로 이제 기록 안됨
				}
			}
			count = 1; // else 문 마지막 문장으로, 다른 원소를 발견했으므로 count값 초기화
		}
	}
	// Check last element
	if (count > maxCount) { //마지막에 최빈값이 나온 경우
		mode = arr[size - 1];
	}
	else if (count == maxCount && modeCount == 1) { //배열의 끝에 최빈값이 나오는 경우. 끝에 나오는 경우가 아닌 이상 modeCount는 1이 아니다. 왜냐? 위 for문의 else-elseif문에서 해당 경우를 처리했기 때문
		mode = arr[size - 1]; //그 값 리턴
	}
	return mode;
}

int range(int* arr, int size) {
		return arr[size - 1] - arr[0];
}


int main() {
	int size = 0;
	int* arr = NULL;
	
	scanf("%d", &size);
	arr = (int*)malloc(sizeof(int) * size);
	for (int i = 0; i < size; i++) {
		scanf("%d", &arr[i]);
	}

	qsort(arr, size, sizeof(int), compare);

	printf("%d\n",mean(arr, size));
	printf("%d\n",median(arr, size)); //n이 홀수이므로,  arr[size/2] 로 처리하여도 된다
	printf("%d\n",most_num(arr, size));
	printf("%d\n",range(arr, size));

}

 

최빈값을 구할 때, for문에서 i를 1부터 시작하여 size전까지 돈다.

그리고, 우측값과 좌측값을 비교하여 같은지 여부를 확인하고, (정렬되어 있다) 같은 경우엔 count++, 다를 경우 값을 업데이트 할지 여부를 결정한다. 하지만! 위 for문 로직으로는 배열의 끝에 최빈값이 등장한 경우를 처리하지 못한다.

(i를 기준으로 arr[i]와 arr[i-1]과 다를 경우 변수를 업데이트하는데, 위 경우에는 다른 경우가 없기 때문) 따라서, 해당 경우return하기 전에 조건문으로 한번 더 확인한다.

 

생각보다 많이 까다로웠던 문제다.

728x90
728x90

부녀회장이 될테야 

 

 

시간 제한메모리 제한제출정답맞힌 사람정답 비율
0.5 초 (추가 시간 없음) 128 MB 100353 56482 47987 57.340%

문제

평소 반상회에 참석하는 것을 좋아하는 주희는 이번 기회에 부녀회장이 되고 싶어 각 층의 사람들을 불러 모아 반상회를 주최하려고 한다.

이 아파트에 거주를 하려면 조건이 있는데, “a층의 b호에 살려면 자신의 아래(a-1)층의 1호부터 b호까지 사람들의 수의 합만큼 사람들을 데려와 살아야 한다” 는 계약 조항을 꼭 지키고 들어와야 한다.

아파트에 비어있는 집은 없고 모든 거주민들이 이 계약 조건을 지키고 왔다고 가정했을 때, 주어지는 양의 정수 k와 n에 대해 k층에 n호에는 몇 명이 살고 있는지 출력하라. 단, 아파트에는 0층부터 있고 각층에는 1호부터 있으며, 0층의 i호에는 i명이 산다.

입력

첫 번째 줄에 Test case의 수 T가 주어진다. 그리고 각각의 케이스마다 입력으로 첫 번째 줄에 정수 k, 두 번째 줄에 정수 n이 주어진다

출력

각각의 Test case에 대해서 해당 집에 거주민 수를 출력하라.

제한

  • 1 ≤ k, n ≤ 14

예제 입력 1 복사

2
1
3
2
3

예제 출력 1 복사

6
10

 

 

해당 문제는 다이나믹 프로그래밍 문제이다.

 

다이나믹 프로그래밍이란, 

복잡한 문제를 여러 개의 작은 하위 문제로 나누어 해결하는 방식을 말한다.

 

수열과 같이 특정한 규칙이 반복되는 경우, 해당 방법으로 해결할수 있다.

다이나믹 프로그래밍 문제를 푸는 방식은 1.바텀업, 2.탑다운 방식으로 총 2가지가 있는데, 

1. 바텀업 방식 -> 반복문을 활용하여, 필요한 정답을 모두 데이터에 저장한 후, 불러옴

2. 탑다운 방식-> 재귀함수를 이용하여, 필요한 값을 기준으로 하나하나 내려가면서(재귀함수를 통해) 더이상 내려갈 필요 없는 부분까지 내려간 후, 위 값을 이용해 다시 올라옴 (재귀함수 원리)

 

바텀업 방식은, 필요하지 않은 정답까지 모두 불러옴으로 시간과 자원을 낭비하는 단점이 있고

탑다운 방식은 스택 오버플로우 위험성과 재귀 호출의 오버헤드로 인한 단점이 있다. (흔히 재귀함수의 단점) 

 

 

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>

int cal(int k, int n) {
	if (k == 0) {
		return n;
	}
	int temp = 0;
	for (int i = 1; i <= n; i++)
		temp += cal(k - 1, i);
	return temp;
}

int main() {
	int size;
	int n, k;
	scanf("%d", &size);
	for (int i = 0; i < size; i++) {
		scanf("%d %d", &n, &k);
		printf("%d\n", cal(n, k));
	}
	
	return 0;
}

 

재귀함수는 직관적이라 이해/작성 하기 쉬움으로, 탑다운 방식으로 작성하였다.

728x90

'알고리즘 > 백준' 카테고리의 다른 글

[백준] 1003 피보나치 함수  (0) 2024.03.04
[백준] 2108 통계학  (2) 2024.02.27
[백준] 2292 벌집  (0) 2024.02.26
[백준] 10814 나이순 정렬  (0) 2024.02.25
[백준] 9012 괄호  (0) 2024.02.25
728x90

벌집 

 
시간 제한메모리 제한제출정답맞힌 사람정답 비율
2 초 128 MB 173331 78473 66792 44.798%

문제

위의 그림과 같이 육각형으로 이루어진 벌집이 있다. 그림에서 보는 바와 같이 중앙의 방 1부터 시작해서 이웃하는 방에 돌아가면서 1씩 증가하는 번호를 주소로 매길 수 있다. 숫자 N이 주어졌을 때, 벌집의 중앙 1에서 N번 방까지 최소 개수의 방을 지나서 갈 때 몇 개의 방을 지나가는지(시작과 끝을 포함하여)를 계산하는 프로그램을 작성하시오. 예를 들면, 13까지는 3개, 58까지는 5개를 지난다.

입력

첫째 줄에 N(1 ≤ N ≤ 1,000,000,000)이 주어진다.

출력

입력으로 주어진 방까지 최소 개수의 방을 지나서 갈 때 몇 개의 방을 지나는지 출력한다.

예제 입력 1 복사

13

예제 출력 1 복사

3

 

 

경계값을 기준으로 보면, 

1, 1+6, 1+6+12, ... {1+6+12 .. + 6*(n)}와 같은 형식으로 이루어진다.

 

수가 주어지면, 위 형식의 경계값 중 주어진 수보다 작으면서 가장 가까운 경계값을 찾은 후 이를 이용해 방의 수를 구할 수 있다.

 

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
int main() {
	int num;
	scanf("%d", &num);
    
	if (num == 1) {
		printf("1");
		return 0;
	}
	
	int index = 0;
	int boundary = 1;
	
	while (boundary < num) {
		boundary = boundary + 6 * index++;
	}
	printf("%d", index);

	return 0;
}

 

1일때는 예외처리가 필요하다.

728x90

'알고리즘 > 백준' 카테고리의 다른 글

[백준] 2108 통계학  (2) 2024.02.27
[백준] 2775 부녀회장이 될테야 (다이나믹 프로그래밍)  (0) 2024.02.26
[백준] 10814 나이순 정렬  (0) 2024.02.25
[백준] 9012 괄호  (0) 2024.02.25
[백준] 2798 블랙잭  (1) 2024.02.25
728x90

나이순 정렬 

시간 제한메모리 제한제출정답맞힌 사람정답 비율
3 초 256 MB 138713 62404 47788 43.578%

문제

온라인 저지에 가입한 사람들의 나이와 이름이 가입한 순서대로 주어진다. 이때, 회원들을 나이가 증가하는 순으로, 나이가 같으면 먼저 가입한 사람이 앞에 오는 순서로 정렬하는 프로그램을 작성하시오.

입력

첫째 줄에 온라인 저지 회원의 수 N이 주어진다. (1 ≤ N ≤ 100,000)

둘째 줄부터 N개의 줄에는 각 회원의 나이와 이름이 공백으로 구분되어 주어진다. 나이는 1보다 크거나 같으며, 200보다 작거나 같은 정수이고, 이름은 알파벳 대소문자로 이루어져 있고, 길이가 100보다 작거나 같은 문자열이다. 입력은 가입한 순서로 주어진다.

출력

첫째 줄부터 총 N개의 줄에 걸쳐 온라인 저지 회원을 나이 순, 나이가 같으면 가입한 순으로 한 줄에 한 명씩 나이와 이름을 공백으로 구분해 출력한다.

예제 입력 1 복사

3
21 Junkyu
21 Dohyun
20 Sunyoung

예제 출력 1 복사

20 Sunyoung
21 Junkyu
21 Dohyun

 

 

구조체를 이용하여 데이터를 저장해야하는데, 멤버로는 가입 순서, 나이, 이름 이 필요하다.

STL을 이용하여 정렬을 수행하면 된다.

 

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>

typedef struct Info{
	int order_of_registration;
	int age;
	char name[101];
} UserInfo;

int compare_UserInfo_age_and_order(const void* a, const void* b) {
	UserInfo* userA = (UserInfo*)a;
	UserInfo* userB = (UserInfo*)b;
	if(userA->age != userB->age)
		return userA->age - userB->age;
	return userA->order_of_registration - userB->order_of_registration;
}


int main() {
	int size = 0;
	scanf("%d", &size);
	
	UserInfo* info_array = (UserInfo*)malloc(sizeof(UserInfo) * size);
	
	for (int i = 0; i < size; i++) {
		info_array[i].order_of_registration = i;
		scanf("%d %s", &(info_array[i].age), info_array[i].name);
	}

	qsort(info_array, size, sizeof(UserInfo), compare_UserInfo_age_and_order);

	for (int i = 0; i < size; i++) {
		printf("%d %s\n", info_array[i].age, info_array[i].name);
	}

	free(info_array);

	return 0;
}
728x90

'알고리즘 > 백준' 카테고리의 다른 글

[백준] 2775 부녀회장이 될테야 (다이나믹 프로그래밍)  (0) 2024.02.26
[백준] 2292 벌집  (0) 2024.02.26
[백준] 9012 괄호  (0) 2024.02.25
[백준] 2798 블랙잭  (1) 2024.02.25
[백준] 1929 소수 구하기  (0) 2024.02.22

+ Recent posts