[정보처리기사/예상문제] - 2023 정보처리기사 실기 예상 문제 모음
C언어 기출문제 모음
비전공자용 C언어 요약 1탄
비전공자용 C언어 요약 2탄(조건문, 반복문, 배열)
비전공자용 C언어 요약 3탄(함수, 포인터)
비전공자용 C언어 요약 4탄(포인터 심화, 구조체)


목차
1. 포인터와 1차원 배열
2. 포인터와 2차원 배열
3. 포인터와 문자 배열
4. 포인터와 문자열 배열
5. 함수 심화(Call by Value, Call by Reference)
6. 구조체


1. 포인터와 1차원 배열
우선 배열의 이름은 배열의 시작주소이다.
 
아래 예문을 보자.

#include <stdio.h>

int main() {
	int array[3] = { 10,20,30 };
    printf("%x %x %x \n", array, array + 0, &array[0]);  //배열 0번째 요소의 주소 출력
	printf("%d %d %d \n", *array, *(array + 0), *&array[0]);  //배열 0번째 요소의 값 10 출력
}

*array는 array(배열 이름) 주소가 가리키는 값인 10이 출력된다
*(array+0)은 array주소에 0칸 이동한 주소가 가리키는 값인 10이 출력된다
*&가 동시에 있으면 생략이 가능하다.
즉 *&arr[0]=array[0] 따라서 역시 배열의 0번째 인덱스인 10이 출력된다
 
결국 array==array+0==&array[0]이라는 의미이다
 
다만 여기서 주의점은 int형 배열이라 요소 1개의 크기는 4바이트이고 array,array+0, &array[0]으로 출력되는 값은 동일하지만 array만 0번째 요소가 아니라 배열 전체를 가리키기 때문에 12바이트(int형이 3칸있으니 12바이트)가 된다
 
다음과 같이 포인터변수 p에 array(배열이름)으로 배열의 시작주소를 저장하면 포인터변수를 배열처럼 사용할 수 있다
아래 예문은 위의 예문과 결과값이 100% 동일하다

#include <stdio.h>

int main() {
	int array[3] = { 10,20,30 };
	int* p = array; //포인터 변수에 배열의 시작 주소를 저장

	printf("%x %x %x \n",p,p+0,&p[0]); //배열 0번째 요소의 주소 출력
	printf("%d %d %d \n", *p,*(p+0),*&p[0]); //배열 0번째 요소의 값 출력
}

 
다 필요없고 위 같은 경우 포인터변수 p랑 array랑 똑같다고 생각해주면 된다


2. 포인터와 2차원 배열
1차원배열에서는 *(array+i)==array[i]==*&array[i]는 값이라고 배웠는데
2차원 배열에서는  *(array+i)==array[i]==*&array[i]가 주소를 가리킨다
 
int array[2][3] = { 10,20,30,40,50,60 }라는 2차원 배열이 있을 때 테이블 같은 구조를 연상하지만 실제는 일직선으로 나란히 돼있는 것을 일단 상기시키고 가보도록 하자.

2차원배열에서도 1차원배열에서 처럼 &array[0][0] 이런식으로 하면 각 배열의 요소의 주소, *&array[0][0] 하면 각 배열의 요소의 값이 잘 출력된다.
 
그렇다면 아래 예문은 어떤 결과가 나올까? 

#include <stdio.h>

int main() {
	int array[2][3] = { 10,20,30,40,50,60 };

	(1)printf("%x\n", array);  
	(2)printf("%x\n", array[0]);
	(3)printf("%x\n", *(array + 0));
	(4)printf("%x\n", &array[0][0]);
    
	(5)printf("%x\n", array+1);
	(6)printf("%x\n", array[1]);
	(7)printf("%x\n", *(array + 1));
	(8)printf("%x\n", &array[1][0]);
    
    
    (9)printf("%d\n",sizeof(&array[0][0]));  //4 출력(32비트 운영체제의 경우)
    //64비트 운영체제의 경우 8로 출력된다
    
	(10)printf("%d\n", sizeof(array[0]));  //12 출력
    (11)printf("%d\n", sizeof(array));  //24 출력
}

(1)~(4)까지는 array[0][0]의 주소를 출력하고
(5)~(8)까지는 array[1][0]의 주소를 출력한다
 
결과값은 동일하지만 4번과 8번의 경우는 arr[0][0], array[1][0] 처럼 특정한 요소의 주소를 가리키지만 나머지는 아래 그림과 같이 특정행을 가리킨다. 
따라서 (9)는 특정요소를 가리키므로 4가 출력되고 (10)은 행을 가리키므로 12가 출력되고 (11)은 배열 전체를 가리켜서 24가 출력된다 

 
*(array + 0)이 형태만 조금 더 살펴 보자
먼저 array는 배열 전체를 가리키지만 위치는 array[0][0] 시작주소에 가 있다

두번째 *(array+0)는 array에서 +0한 곳의 값을 읽는 것인데 2차원배열에서는 이 값이 행의 주소를 가리킨다
배열에서 특정 행을 가리키지만 위치는 array[0][0] 시작주소에 가 있다

세번째 array[0]+0은 배열에서 특정 행의 특정 요소 가리키지만 역시 위치는 array[0][0] 시작주소에 가 있다
따라서 

모두 출력되는 값은 동일하지만 크기는 다르다
이해한 것을 토대로 아래 출력결과를 예상해보자.

#include <stdio.h>

int main() {
	int array[2][3] = { 10,20,30,40,50,60 };

	(1)printf("%d\n", sizeof(array));
	(2)printf("%d\n", sizeof(array[0]+0));
	(3)printf("%d\n", sizeof(*(array + 0)));
	(4)printf("%d\n", sizeof( & array[0][0])); 
    }
정답보기

(1)은 array가 배열의 전체를 가리키므로 24

(2)는 array[0]이 2차원배열의 0번째 행을 가리키고 0번째 행의 0번째 요소를 가리키므로 4출력

(3)는 array는 배열의 전체를 가리키지만 array+0을 함으로써 0번째 행을 가리키게 돼 12가 출력된다(2차원 배열에서는 배열이름의 주소에 대한 값이 특정행의 주소를 가리키게 된다)

(4)는 0행 0열의 주소를 가리키므로 4가 출력된다

#include <stdio.h>

int main() {
	int array[2][3] = { 10,20,30,40,50,60 };

	printf("%d\n", sizeof(array));  //24출력
	printf("%d\n", sizeof(array[0]+0)); //4출력(32비트 운영체제 기준)
    //64비트 운영체제의 경우 8로 출력된다
    
	printf("%d\n", sizeof(*(array + 0))); //12출력
	printf("%d\n", sizeof( &array[0][0])); //4출력(32비트 운영체제 기준)
    //64비트 운영체제의 경우 8로 출력된다
    }
사진 출처 : C언어본색-박정민

 
하나도 이해가 안된다면 그냥 외우면된다.
1차원 배열에서는 값을 가리켰던 녀석들이 2차원 배열에서는 주소를 가리키는 것이다
 
 
1차원 배열과 동일하게 2차원 배열도 포인터를 써서 똑같이 쓸 수 있지 않을까? 생각할 수있지만 아래처럼 코드를 짜면 에러가 난다. int *p는 1차원 포인터 변수이므로 2차원 배열을 저장해도 1차원처럼 작동되기 때문이다.
따라서 이 경우에 필요한 것이 바로 배열 포인터이다.

#include <stdio.h>

int main() {
	int array[2][3] = { 10,20,30,40,50,60 };
	int* p = array;
	printf("%d\n",p[0][0]);

	return 0;
}

 
2-1.배열 포인터
배열 포인터는 배열을 가리키는 포인터 변수이다.
배열 포인터는 다음과 같이 선언한다.

앞에 int는 자료형이고 (*p)는 배열 포인터 변수 이름이고 [3]은 열 길이이다
이렇게 배열포인터로 선언을 하면 2차원 배열처럼 포인터도 접근할 수 있게 된다

#include <stdio.h>

int main() {
	int array[2][3] = { 10,20,30,40,50,60 };
	int(* p)[3] = array; //포인터 변수에 배열의 시작 주소를 저장
	printf("%d\n", array[0][0]);
	printf("%d\n",p[0][0]);

	return 0;
}

 
2-2 포인터 배열
주소를 저장하는 배열
 
포인터 배열은 다음과 같이 선언한다

앞에 int*는 자료형이고 pointer는 포인터배열 변수 이름이고 [3]은 배열 길이이다

#include <stdio.h>

int main() {
	int a = 10, b = 20, c = 30;
	int* p[3] = {&a,&b,&c};
	printf("%d\n", *p[0]);
	printf("%d\n", *p[1]);
	printf("%d\n", *p[2]);
	printf("%d\n", **(p+0));
	printf("%d\n", **(p+1));
	printf("%d\n", **(p+2));

	return 0;
}

 

정답보기

p[0]에 주소가 들어있으니 *p[0]은 그 주소의 값 따라서 10

나머지도 마찬가지

 

p는 배열의 이름이니 배열의 시작주소인데 +0으로 p의 0번째 요소가 선택되고 0번째요소의 값의 값이니 10이 된다

#include <stdio.h>

int main() {
	int a = 10, b = 20, c = 30;
	int* p[3] = {&a,&b,&c};
	printf("%d\n", *p[0]); //10
	printf("%d\n", *p[1]);  //20
	printf("%d\n", *p[2]);  //30
	printf("%d\n", **(p+0)); //10
	printf("%d\n", **(p+1)); //20
	printf("%d\n", **(p+2));  //30

	return 0;
}

3. 포인터와 문자 배열
문자배열은 그냥 1차원 배열과 포인터 개념과 동일하다
 
아래 예문을 보자.

#include <stdio.h>

int main() {
	char array[3] = { 'A','B','C'};
	char* p = array; //포인터 변수에 배열의 시작 주소를 저장
	printf("%x %x %x \n", array, array + 0, &array[0]);  //배열 0번째 요소의 주소 출력
	printf("%c %c %c \n", *array, *(array + 0), *&array[0]);  //뱌욜 0번째 요소의 값 A 출력

	printf("%x %x %x \n", p, p + 0, &p[0]); //배열 0번째 요소의 주소 출력
	printf("%c %c %c \n", *p, *(p + 0), *&p[0]); //배열 0번째 요소의 값 A 출력
	
}

 
다 필요없고 위 같은 경우 포인터변수 p랑 array랑 똑같다고 생각해주면 된다


 
4. 포인터와 문자열 배열
C언어에서 문자열이란 큰 따옴표 내에 포함된 하나 이상의 문자를 의미하며 문자열의 맨 끝에는 문자열의 끝을 알리는 종료 문자 '\0'이 삽입되어 있으며 이런 종료 문자를 널(Null) 문자라 한다
 
문자열은 문자들이 메모리 공간에 연속적으로 저장되어 있어서 보통 주소로 관리되며 문자열을 출력할 때는 서식 문자 %s를 쓴다문자열은 문자들의 집합이므로 문자의 배열이라고 생각할 수 있다
 
아래 결과를 예상 해보자.

#include <stdio.h>

int main() {
	char array[4] = "ABC";
	printf("%c%c%c\n", array[0], array[1], array[2]);
	printf("%s", array);
	printf("%d %d %d\n", array[0], array[1], array[2]);
	return 0;
}
정답보기

A를 %d로 출력하면 A의 아스키코드인 65가 출력된다

#include <stdio.h>

int main() {
	char array[4] = "ABC";
	printf("%c%c%c\n", array[0], array[1], array[2]); //ABC 출력
	printf("%s\n", array); //ABC 출력
	printf("%d %d %d\n", array[0], array[1], array[2]); //65 66 67 출력
	return 0;
}

 
다음 결과도 예상해보자.

#include <stdio.h>

int main() {
	char array[4] = "ABC";
	printf("%s\n", array);
	printf("%s\n", array+1);
	printf("%s\n", array+2);
	return 0;
}
정답보기

array일때는 전체를 가리켜서 ABC가 출력

array+1일때는 arr[1]번째 시작주소를 가리켜서 BC

array+2일때는 array[2]번째 시작주소를 가리켜서 C

#include <stdio.h>

int main() {
	char array[4] = "ABC";
	printf("%s\n", array);  //ABC 출력
	printf("%s\n", array+1); //BC 출력
	printf("%s\n", array+2);  //C 출력
	return 0;
}

 
이런 문자열을 배열이 아니라 포인터로 바꾸면 아래와 같다

#include <stdio.h>

int main() {
	char array[4] = "ABC"; //배열 방식
	printf("%s\n", array);
	printf("%s\n", array+1);
	printf("%s\n", array+2);

	char* p = "ABC";  //포인터방식
	printf("%s\n", p);
	printf("%s\n", p + 1);
	printf("%s\n", p + 2);
	return 0;
}

5. 함수 심화(Call by Value, Call by Reference)
 
함수에서 이때 까지 사용했던 방식이 Call my Value인데 그것은 값을 복사하는 방식이다.

#include <stdio.h>
int func(int);
int main() {
	int a = 10;
	printf("%d\n",func(a));  //11출력
    printf("%d\n",a);  //10출력
	return 0;
}


int func(int a) {
	a = a + 1;
	return a;
}

 
main 함수에서 변수 a를 선언하면서 초기화 한 후 a변수를 인수로 전달하면서 매개변수 a로 값이 복사된다
값을 복사하는 방식으로 main()함수의 지역변수 a 값 변동에는 영향을 미치지 않는다

다음은 새로 배우게 될 Call by Reference(주소 참조) 방식이다
주소를 인수로 받아서 값을 바꾸기 때문에 main()함수의 지역변수 a에도 영향을 미친다

#include <stdio.h>
int func(int);
int main() {
	int a = 10;
	printf("%d\n",func(&a)); //11 출력
	printf("%d", a);  //11 출력
	return 0;
}


int func(int* a) {
	*a = *a + 1;
	return *a;
}

int func(int* a)인 이유는 주소를 전달받기 때문이다
쉽게 생각해서 포인터변수를 선언할 때 int *p=&a; 이런식으로 선언하는 것을 생각하면 된다
주소를 받기 때문에 함수의 매개변수 형태가 포인터변수가 되는 것이다
 


6. 구조체
하나 이상의 변수를 묶어 그룹화하는 사용자 정의 자료형이다
배열이 같은 자료형만 담을 수 있다면 구조체는 다른 자료형도 담을 수 있다
 
아래와 같은 형식으로 구조체를 정의한다
struct : 구조체 키워드
student : 구조체 이름
 
int score, char name은 구조체 멤버 변수

struct student
{
	int score;
	char name;
};

구조체 정의를 하고 구조체 변수 선언을 동시에 하는 방법은 아래와 같다.

struct student
{
	int score;
	char name;
}p1, p2, p3;

 
따로 선언하는 방식은 아래와 같다.

#include<stdio.h>
struct student
{
	int score;
	char name;
};

int main(){
	struct student p1, p2, p3;


return 0;
}

 
구조체 변수에 .(도트 연산자)를 이용하면 각 변수별 멤버변수에 접근할 수 있다
p1.score이라고 하면 p1의 score에 접근이 가능해지며
p1.score=30; 이라고 하면 p1.score 자리에 30이 저장된다

#include<stdio.h>
struct student
{
	int score;
	char name;
};

int main(){
	struct student p1, p2, p3;
	p1.score = 30;
	printf("%d", p1.score);  //30출력

return 0;
}

 
6-1 구조체와 배열
멤버변수로 배열도 사용 가능하다
 
학생 수가 많아지면 구조체 변수 p1,p2,p3가 무한히 많이 필요할 텐데 이런 문제를 해결하기 위해 구조체 변수를 배열로 사용하면 된다

#include<stdio.h>
struct student {
	char no[10];
	char name[20];
	double math;
	double english;
};

int main(){
	struct student stu[3] = {
		{"12345","Lee",90,90},
		{"12346","KIM",80,70},
		{"12347","Park",50,100}
	};

return 0;
}

 

 
6-2 구조체와 포인터

#include<stdio.h>
struct p {
	int* x;
	int* y;
};

int main(){
	int a = 5;
	int b = 4;
	struct p p1;
	p1.x = &a;
	p1.y = &b;

	printf("%d %d\n", a, b);
	printf("%d %d\n", *p1.x, *p1.y);

return 0;
}

출력결과를 예상해보자

 
 
구초제 변수로 포인터를 사용할 수 도 있다
구조체 변수로 포인터를 사용한다는 것은 구조체 변수에 간접 접근할 수 있다는 의미이다
 
(*p)를 한 이유는 .연산자가 *연산자보다 우선순위가 높아 *p를 먼저 연산하기 위함이다
p는 포인터변수인데 p가 구조체 변수 stu의 주소를 담고 있으므로 *p는 stu의 값을 가리킨다
->연산자는 포인터 변수만으로 구조체의 멤버 변수에 접근할 때 사용한다

#include<stdio.h>
struct student {
	char no[10];
	char name[20];
	double total;
};

int main(){
	struct student stu = {"12345","KIM",20};
	struct student* p = &stu;

	printf("%s %s %lf\n", (*p).no, (*p).name, (*p).total); //12345 KIM 20 출력
	printf("%s %s %lf\n", p->no, p->name, p->total);//12345 KIM 20 출력

return 0;
}

 
 
 
[정보처리기사/예상문제] - 2023 정보처리기사 실기 예상 문제 모음
C언어 기출문제 모음
비전공자용 C언어 요약 1탄
비전공자용 C언어 요약 2탄(조건문, 반복문, 배열)
비전공자용 C언어 요약 3탄(함수, 포인터)
비전공자용 C언어 요약 4탄(포인터 심화, 구조체)

[정보처리기사/예상문제] - 2023 정보처리기사 실기 예상 문제 모음

C언어 기출문제 모음

비전공자용 C언어 요약 1탄

비전공자용 C언어 요약 2탄(조건문, 반복문, 배열)

비전공자용 C언어 요약 3탄(함수, 포인터)

비전공자용 C언어 요약 4탄(포인터 심화, 구조체)

 


목차

1. 함수

2. 변수의 종류

3. 재귀 함수

4. 포인터

 


1. 함수

이처럼 함수는 특정 작업을 수행하는 코드의 집합이라고 생각하면 된다.

예를 들어 a+b라는 식이 있다고 생각하면 a+b가 한번만 나오면 상관없겠지만 엄청 많이 나오면 반복적인 작업임에도 코드가 길어지고 번거로울 수 있다. 그래서 더하기 작업을 수행하는 함수를 만들어 놓는 것이다. 

함수의 기본 구조는 다음과 같다.

사진 출처 : C언어 - 함수의 구조, 함수의 형태 4가지 정리글 (1차 수정) : 네이버 블로그 (naver.com)

 

입력 형태에서 보이는 int x, int y는 매개 변수(Parameter)라고 한다.

함수의 형태는 아래 그림과 같이 4가지가 있다.

출력 형태가 있으면 1, 없으면 0

입력 형태가 있으면 1, 없으면 0

 

return의 의미는 반환하고 함수를 종료라는 의미이므로 출력 형태가 1인 경우에는 반드시 return문이 있어야 한다

사진 출처 : C언어 - 함수의 구조, 함수의 형태 4가지 정리글 (1차 수정) : 네이버 블로그 (naver.com)

 

함수를 배웠으니 변수의 종류에 대해 살펴보자.


2. 변수의 종류

함수에서 사용되는 변수는 지역 변수, 전역 변수, 정적(static) 변수, 외부(extern) 변수, 레지스터(register) 변수가 있다.

이 중에 지역 변수, 전역 변수, 정적(static) 변수만 공부해보도록 하자.

 

1) 지역 변수(Local Variable)

main() 함수, 조건문, 반복문의 중괄호({}) 내부와 함수의 매개 변수(Parameter)로 사용되는 변수를 의미한다

 

아래의 예문 출력결과가 무엇인지 예측 해보자.

#include<stdio.h>
int hello(void);

int main() {
	int k = 20;
	
	printf("%d\n",hello()); 
    printf("%d", k);   
    
	return 0;
}


int hello(void) {
	int k = 100;
	return k;
}
더보기

100

20

#include<stdio.h>
int hello(void);

int main() {
	int k = 20;
	
	printf("%d\n",hello()); //100출력
    printf("%d", k);    //20 출력
    
	return 0;
}


int hello(void) {
	int k = 100;
	return k;
}

위 예문을 보면 int k=20으로 초기화를 시킨후 hello라는 함수를 호출해서 k는 100이라고 바뀐줄 알았는데 그게 아니고 20이 그대로 호출되었다.

즉, main()함수의 k와 hello()함수의 k는 다르다는 것이다.

 

심도있게 알기 위해서는 메모리에 대해 알아야한다. 

지역변수라서 함수가 종료되면 메모리 공간이 소멸된다.

 

2) 전역 변수(Global Variable)

중괄호 외부에 선언되는 변수로 어느 지역에서도 사용이 제한되지 않는 프로그램 어디에서든 접근이 가능한 변수이다

중괄호가 시작되면 메모리가 생성되고 해당 중괄호에서 빠져나오면 메모리가 소멸되는 지역변수와 달리 전역변수는 프로그램이 시작되면 메모리에 생성되고 프로그램이 종료되면 메모리가 소멸된다

 

그래서 전역 변수는 프로그램이 종료되지 않는 한 계속해서 메모리에 존재하고 영역에 제한받지 않으며 사용된다

그리고 전역 변수는 초깃값을 지정해주지 않아도 자동으로 값을 0으로 가진다.

 

전역 변수의 잘못된 사용과 남용은 공유 자원에 대한 잘못된 접근으로 부작용을 낳을 수 있는데 시스템의 변경과 유지 보수를 어렵게 하는 원인이 될 수있으므로 전역 변수의 사용은 최대한 피하는 것이 바람직하다

 

3) 정적 변수(Static)

전역 변수의 단점을 부분적으로 보완한 변수

정적변수는 변수의 자료형 앞에 static 키워드를 넣어서 만든다

static int a;

 

정적 변수는 전역 변수처럼 프로그램이 종료되지 않는 한 메모리가 소멸되지 않고 초깃값을 지정하지 않아도 자동으로 0으로 가지며 또한 정적 변수는 초기화가 단 1회만 수행된다.

 

다음 수행결과가 무엇일지 생각해보자.

#include<stdio.h>
void hello(void);

int main() {
	int k = 20;
	
	hello();
	hello();
	hello();

	return 0;
}


void hello(void) {
	static k = 1;
	int t = 0;
	k++;
	t++;
	printf("%d %d\n", k, t);
}
더보기

2 1

3 1

4 1

static 변수 k는 한번만 초기화되므로 k가 정상적으로 계속 증가하는 것을 볼 수 있지만

지역변수 t는 계속 초기화가 진행이 돼 모든 값이 1임을 확인할 수 있다.


3. 재귀함수(Recursive Function)

함수 내에서 자기 자신을 호출하는 함수

재귀 함수의 경우 계속적인 자기 자신의 함수 호출로 시간과 메모리 공간의 효율이 저하될 수 있다.

 

다음 코드의 결과값을 생각해보자.

#include<stdio.h>
int fact(int);

int main() {
	int a=4;
	printf("%d",fact(a));
	return 0;
}

int fact(int n) {
	if (n <= 1)
		return 1;
	else
		return n * fact(n-1);
}
더보기

24

 

fact(4)=return 4 * fact(3)

fact(3)=return 3* fact(2)

fact(2)=return 2*fact(1)

fact(1)=1

 

fact(4)=4*3*2*1=24

 

따라서 결과는 24

 

위 문제는 재귀함수를 이용한 팩토리얼 예제였다.


4. 포인터

포인터(pointer)란 메모리의 주소값을 저장하는 변수이며, 포인터 변수라고도 부른다.

말 그대로 뭔가를 가리키는 의미를 가지고 있다. (주소값을 가리킴)
char형 변수가 문자를 저장하고, int형 변수가 정수를 저장하는 것처럼 포인터는 주소값을 저장한다.

 

포인터 변수의 선언은 아래와 같이 선언한다. 

사진 출처  : C언어 포인터 사용시 주의점 : 네이버 블로그 (naver.com)

 

포인터 변수의 크기는 주소를 가리키기 때문에 항상 4byte의 동일한 크기를 가진다. (32bit 운영체제 한정)

 

1) 주소 연산자(&)
주소 연산자는 변수의 이름 앞에 사용하여, 해당 변수의 주소값을 반환한다.

 

2) 참조 연산자(*)
참조 연산자는 포인터의 이름이나 주소 앞에 사용하여, 포인터에 가리키는 주소에 저장된 값을 반환한다.

dereference라고 하여 역참조연산자라고도 한다.

 

다 필요없고 포인터는 하나만 알면 된다.

*p는 선언할 때를 제외하고 무조건 값을 가리킨다고 생각하면 되고 &는 주소를 가리킨다고 생각하면 된다.

 

아래는 대충 이해를 돕기위해 만든 표이다. (주소는 그냥 임시로 준 값이다)

#include <stdio.h>

int main() {
  int a=3;
  int b=&a;
  int *c=&a;
  printf("%d\n",a); //그냥 변수 a를 출력
  printf("%d\n",b); //b에 저장된 a의 주소의 값을 출력
  printf("%d\n",c); //포인터 변수 c에 저장된 a의 주소 출력
  printf("%d\n",*c); //포인터 변수 c에 저장된 a의 주소의 값 출력
  return 0;
}

printf("%d",a); //출력 결과 : 3

printf("%d",b); //출력 결과 : 1000K

printf("%d",c); //출력 결과 : 1000K

printf("%d",*c); //출력 결과 : 3

 

b는 a의 주소 값이 b라는 변수에 대입된 것이고

c는 a의 주소 값을 가리킬 수 있게 된 것이다

즉, a와 b는 연결고리가 없는 반면 a와 c는 연결고리가 있다.

 

3) 다차원 포인터

1차원 포인터에는 변수의 주소값이 저장되고 2차원포인터(이중포인터)에서는 1차원 포인터의 주소가 저장된다.

마찬가지로 3차원 포인터는 2차원 포인터의 주소가 저장된다

n차원 포인터는 n-1포인터의 주소를 넣는다.

#include <stdio.h>

int main() {
  int a=3;
  int* b=&a;   //1차원 포인터 : 변수의 주소값 저장
  int** c=&b;  //2차원 보인터 : 1차원 포인터의 주소값 저장
  
  printf("%d\n",b);  //7행과 8행 동일
  printf("%d\n",*c);  // 결과값 : a의 주소
  
  printf("%d\n",*b);  //9행과 10행 동일
  printf("%d\n",**c); //결과값  : 3
  return 0;
}

포인터 변수 b는 a의 주소를 가리키기 때문에 

b는 a의 주소이고

*b는 a의 값이다

 

포인터 변수 c는 b의 주소를 가리키기 때문에

c는 b의 주소이고

*c는 b의 값인데 b의 값이 a의 주소이므로 *c는 a의 주소이다.

*c가 a의 주소이므로 **c는 a의 주소의 값(즉, a의 값)이다 

 

포인터 아무것도 이해 안돼도 좋다. 

정보처리기사 실기는 어차피 결과값 묻는 문제가 대부분인데 포인터는 주소를 가리키는 녀석이다. 컴퓨터마다 메모리 주소가 다 다르므로 주소를 묻는 문제는 나올 수가 없다.

그래서 그냥 포인터변수가 나오면 상수 대입해서 대충 찍어봐라.

대부분 맞다.

아래는 기출문제이다. 한번 생각해보자.

#include <stdio.h>
 
int main(){
int *arr[3];
int a = 12, b = 24, c = 36;
arr[0] = &a;
arr[1] = &b;
arr[2] = &c;
 
printf("%d\n", *arr[1] + **arr + 1);
}
더보기

37

 

야매 풀이

*arr[1] 뭔지 모르겠는데 arr이 배열이니까 왠지 arr[1]=b의 값일 것 같다 =24

**arr 뭔지 모르겠지만 배열인데 아무것도 안적혀있는 거보니 arr[0]을 가리키는 것 같다 그러면 arr[0]+1하면 되지 않을까? =12+1

 

24+13=37

 

진짜 풀이

*arr[1]은 arr[1]의 값인데 arr[1]이 b의 주소를 가리키므로 b의 주소의 값(b의 값) = 24

arr은 배열의 이름이고 arr[0]의 주소와 동일한데 *arr은 a의 주소이고  **은 a의 주소의 값(a의 값) =12

24+12+1=37

(참고로 *가 우선순위 더 높아서 +1보다 먼저 수행된다)

 

4) 주소의 가감산

C언어에서 배우는 자료형은 모두 주소를 가지며 주소의 가감산이 가능하다.

char형 주소에 +1을 하면 1바이트 증가, int는 4바이트 증가, double은 8바이트 증가

 

다음은 기출문제이다. 출력결과를 예상해보자.

#include <stdio.h>

int main() {
	char* p = "KOREA";
	printf("%s\n", p);
	printf("%s\n", p + 3);
	printf("%c\n", *p);
	printf("%c\n", *(p + 3));
	printf("%c\n", *p + 2);
}
더보기

4번째 줄: KOREA라는 문자열의 주소를 포인터변수 p에 저장한다는 의미

5번째줄 :  char이 p라는 주소에서 시작해서 널값 전까지를 전부 출력해서 KOREA

6번째줄 :  char이 1byte이므로 +3하면 3칸이동한것이라고 생각하면 되는데 왼쪽으로 3칸이동해서 널값 전까지 전부 출력해서 EA

7번째줄 :  서식문자가 %c로 바뀌었고 p의 값을 물어보고 있으므로 K

8번째줄 :  서식문자가 %c로 바뀌었고 p+3(p에서 주소를 3칸 이동)의 값이므로 E

9번쨰줄 : 서식문자가 %c로 바뀌었고 p의 값에서 +2한 문자이므로 K,L,M 순이므로 M

 

그냥 p는 p의 제일 첫번째(p+0)를 가리키고 있다고 생각하면 됨

 

 

[정보처리기사/예상문제] - 2023 정보처리기사 실기 예상 문제 모음

C언어 기출문제 모음

비전공자용 C언어 요약 1탄

비전공자용 C언어 요약 2탄(조건문, 반복문, 배열)

비전공자용 C언어 요약 3탄(함수, 포인터)

비전공자용 C언어 요약 4탄(포인터 심화, 구조체)

1. 다음은 MySQL을 이용하여 회사 사원의 인적 사항을 관리하기 위한 DB 테이블(테이블명 : worker)의 구조이다.

필드명 데이터형 null 추가 사항 설명
num int not null primary key, auto_increment 일련번호
id char(20) not null   아이디
name char(10) not null   이름
gender char(1) not null   성별(남성: M, 여성 : F)
age int not null   나이
department char(20) not null   근무 부서
phone char(20) not null   휴대전화 번호
address char(100)     집 주소

 

① worker 테이블을 일괄 생성하기 위한 worker.sql 파일의 빈칸을 채우시오.

create (가) worker (
     num int not null (나),
     id char(20) not null,
     name char(10) not null,
     (다) char(1) not null,
     age int,
     department char(20) not null,
     phone char(20) not null,
     (라) char(100),
     primary key ( 마 )  );

② worker 테이블에 다음 데이터를 삽입하는 명령을 쓰시오.

chkim, 김창훈, M, 28, 총무부, 010-3838-8577, 경기도 용인시

③ worker 테이블의 구조를 보는 데 사용하는 명령을 쓰시오.

④ worker  테이블에 존재하는 모든 레코드의 모든 필드를 보여주는 명령을 쓰시오.

더보기

(가) : table

(나) : auto_increment

(다) : gender

(라) : address

(마) : num

 

② insert into worker (id, name, gender, age, department, phone, address) values ('chkim', '김창훈', 'M', 28, '총무부', '010-3838-8577', '경기도 용인시');

③ desc worker;

④ select*from worker;


2. 1번 문제에서 생성한 worker 테이블에 다음 표의 사원 데이터 필드를 일괄 실행하기 위한 파일을 작성하여 insert.sql로 저장하고 이를 phpMyAdmin에서 실행하시오.

id name gender age department phone address
jhjang 장종훈 M 38 총무부 010-3347-7474 서울시 강동구
jek98 안달훈 M 25 인사부 010-3744-4747 강원도 정선군
Ims2 이세영 F 48 경리부 010-3736-4784 전라북도 전주시
apple3 김소영 F 52 총무부 010-2845-8978 전라남도 순천시
sub093 김수정 M 53 홍보부 010-3834-1248 경상북도 상주시
chulsu 김철수 M 23 홍보부 010-1235-3788 경기도 오산시
yckim 김영철 M 32 인사부 010-2222-3334 경기도 용인시
yerin99 김예린 F 25 총무부 010-7777-9999 서울시 강서구
sjjung 정수정 F 35 경리부 010-3636-3563 충청남도 아산시
tuttle7 이예린 F 21 연구소 010-3756-9823 경기도 성남시
더보기

insert into worker (id, name, gender, age, department, phone, address) values ('jhjang', '장종훈', 'M', 38, '총무부', '010-3347-7474', '서울시 강동구');

insert into worker (id, name, gender, age, department, phone, address) values ('jek98', '안달훈', 'M', 25, '인사부', '010-3744-4747', '강원도 정선군');

insert into worker (id, name, gender, age, department, phone, address) values ('Ims2', '이세영', 'F', 48, '경리부', '010-3736-4784', '전라북도 전주시');

insert into worker (id, name, gender, age, department, phone, address) values ('apple3', '김소영', 'F', 52, '총무부', '010-2845-8978', '전라남도 순천시') ;

insert into worker (id, name, gender, age, department, phone, address) values ('sub093', '김수정', 'M', 53, '홍보부', '010-3834-1248', '경상북도 상주시');

insert into worker (id, name, gender, age, department, phone, address) values ('chulsu', '김철수', 'M', 23, '홍보부', '010-1235-3788', '경기도 오산시');

insert into worker (id, name, gender, age, department, phone, address) values ('yckim', '김영철', 'M', 32, '인사부', '010-2222-3334', '경기도 용인시');

insert into worker (id, name, gender, age, department, phone, address) values ('yerin99', '김예린', 'F', 25, '총무부', '010-7777-9999', '서울시 강서구');

insert into worker (id, name, gender, age, department, phone, address) values ('sjjung', '정수정', 'F', 35, '경리부', '010-3636-3563', '충청남도 아산시');

insert into worker (id, name, gender, age, department, phone, address) values ('tuttle7', '이예린', 'F', 21, '연구소', '010-3756-9823', '경기도 성남시');


3. 2번 문제에서 입력한 사원 데이터를 검색하는 SQL 명령에 관한 물음에 답하시오.

① 홍보부에서 근무하는 사원의 모든 필드를 검색하는 명령을 쓰시오.

② 20대(20~29세)인 사원의 아이디, 이름, 집 주소를 검색하는 명령을 쓰시오.

③ 경리부에 근무하는 이름에 '정'이 들어간 사원의 이름, 성별, 근무 부서, 휴대전화 번호를 검색하는 명령을 쓰시오.

④ 경기도에 사는 사원을 나이순으로 오름차순 정렬한 뒤 이름, 집 주소, 성별, 휴대전화 번호를 검색하는 명령을 쓰시오.

⑤ 여성 사원을 나이순으로 내림차순 정렬한 뒤 이름, 성별, 아이디, 휴대전화 번호, 집 주소를 검색하는 명령을 쓰시오.

⑥ 홍보부에서 근무하는 남성 사원의 이름과 휴대전화 번호를 검색하는 명령을 쓰시오.

⑦ 용인시에 사는 사원의 이름과 휴대전화 번호를 검색하는 명령을 쓰시오.

⑧ '김영철'의 휴대전화 번호를 '010-2222-3333'으로 수정하는 명령을 쓰시오.

⑨ 아이디가 'chulsu'인 사원의 나이를 '33'으로 수정하는 명령을 쓰시오.

⑩ 아이디가 'jhjang'인 레코드를 삭제하는 명령을 쓰시오.

⑪ 경리부에서 근무하는 '이세영'의 레코드를 삭제하는 명령을 쓰시오.

더보기

① SELECT num, id, name, gender, age, department, phone, address FROM worker WHERE department='홍보부';

② SELECT id, name, address FROM worker WHERE age>=20 and age<=29;

③ SELECT name, gender, department, phone FROM worker WHERE department='경리부' AND name LIKE '%정%';

④ SELECT name,address, gender, phone FROM worker WHERE address LIKE '경기도%' ORDER BY age;

⑤ SELECT name, gender, id, phone, address FROM worker WHERE gender='F' ORDER BY age DESC;

⑥ SELECT name, phone FROM worker WHERE department='홍보부' AND gender='M';

⑦ SELECT name, phone FROM worker WHERE address LIKE '%용인시';

⑧ UPDATE worker set phone='010-2222-3333' WHERE name='김영철';

⑨ UPDATE worker set age=33 WHERE id='chulsu';

⑩ DELETE from worker WHERE id='jhjang';

⑪ DELETE from worker WHERE name='이세영';


4. 명령 프롬포트에서 mydb 데이터베이스에 저장된 모든 데이터를 mydb.sql 파일에 백업하는 명령을 쓰시오. 계정은 'user', 비밀번호는 '12345'라고 가정한다.

더보기

mysqldump-uuser -p12345 mydb > mydb.sql


5. 4번 문제에서 백업한 mydb.sql 파일을 명령 프롬프트에서 복원하는 명령을 쓰시오.

더보기

mysql -uuser -p12345 mydb < mydb.sql

1. 다음은 MySQL에 관한 설명이다. 맞으면 O, 틀리면 X 표시를 하시오.

①MySQL 데이터베이스 관리 시스템의 하나이다. (   )

②주로 대형 서버 컴퓨터에 사용하고 중소 시스템이나 PC에는 잘 사용하지 않는다. (   )

③MySQL은 테이블을 기본 구조로 하는 관계형 데이터베이스 관리 시스템에 속한다. (   )

④PHP로 웹 프로그래밍을 할 때 데이터베이스로 MySQL을 많이 사용한다. (   )

⑤SQL은 데이터베이스를 조회하거나 데이터를 갱신할 수 있는 데이터베이스 표준 언어이다. (   )

⑥phpMyAdmin의 문자셋은 세계 표준인 euc-kr로 설정하는 것이 바람직하다. (   )

⑦MySQL은 처리 속도가 빠르고 성능이 우수한 편이지만 초보자가 익히기 어렵다. (   )

더보기

① O

② X(예전에는 주로 대형 메인 프레임에 DBMS를 적용했지만 요즘은 중소형 시스템과 PC에서도 많이 사용)

O

O

O

X(UTP-8)

X(초보자도 익히기 쉽다)


2. MySQL 데이터베이스에서 DB 테이블을 관리하는 SQL 명령에 관한 물음에 답하시오.

①데이터베이스를 생성하는 명령을 쓰시오.

②DB 테이블명이 member일 때 이 테이블의 구조를 보는 데 사용하는 명령을 쓰시오.

③member 테이블에 age 필드를 int 데이터형으로 추가하는 명령을 쓰시오.

④member 테이블에서 age 필드를 삭제하는 명령을 쓰시오.

⑤member 테이블의 이름을 members로 변경하는 명령을 쓰시오.

⑥member 테이블의 name 필드를 char(20)으로 변경하는 명령을 쓰시오.

더보기

① create table

② desc member

③ alter table member add age int

④ alter table member drop age

⑤ alter table member rename members

⑥ alter talbe member modify name char(20)


3. 다음은 회사 사원의 인적 사항을 관리하기 위한 DB 테이블(테이블명: employee)의 구조이다.

필드명 데이터형 설명
num int 일련번호
id char(20) 아이디
name char(10) 이름
gender char(1) 성별(남성: M, 여성: F)
age int 나이
department char(20) 근무 부서
hp char(20) 휴대전화 번호
address char(100) 집 주소

① employee 테이블을 생성하는 명령의 빈칸을 채우시오.

create table employee(
      num int,
      (가) char(20),
      name char(10),
      gender (나),
      age int
      department char(20),
      (다) char(20),
      address char(100),
      primary key(  (라)  )  );

② employee 테이블의 구조를 보는 데 사용하는 명령을 쓰시오.

③ employee 테이블에 email 필드를 char(30) 데이터형으로 추가하는 명령을 쓰시오.

④ employee 테이블의 이름을 member로 변경하는 명령을 쓰시오.

⑤ age 필드를 삭제하는 명령을 쓰시오.

⑥ employee 테이블을 삭제하는 명령을 쓰시오.

더보기

(가) : id

(나) : char(1)

(다) : hp

(라) : num(웹 사이트에서 회원 가입 양식을 작성할 때 주로 아이디나 일련번호를 사용하여 레코드를 식별한다)

 

② desc employee;

③ alter table employee add email char(30);

④ alter table employee rename member;

⑤ alter table employee drop age;

⑥ drop table employee;

[정보처리기사/예상문제] - 2023 정보처리기사 실기 예상 문제 모음

C언어 기출문제 모음

비전공자용 C언어 요약 1탄

비전공자용 C언어 요약 2탄(조건문, 반복문, 배열)

비전공자용 C언어 요약 3탄(함수, 포인터)

비전공자용 C언어 요약 4탄(포인터 심화, 구조체)

 


목차

1. 조건문

2. 반복문

3. 배열


1. 조건문

조건에 맞을 경우 해당 실행문장을 처리하고자 할 때 사용한다

조건문에는 if문과 swtich-case문이 있다.

 

먼저 if문

a>b가 참이면 a=a+b;가 실행이 되고 거짓이면 a=a-b;가 실행된다

아래 예문은a=a+b가 실행이 돼 13이 출력 된다

#include<stdio.h>
int main() {
	int a = 8, b = 5;
	if (a > b)
		a = a + b;
	else
		a = a - b;
	printf("%d\n", a);
	return 0;
}

비트연산자에서 봤듯이 조건문에서도 연산자가 있다. &&, ||

&&는 2개의 조건이 모두 참이어야 참(1)을 반환한다

||는 2개의 조건중 하나만 참이어도 참(1)을 반환한다

 

if외에도 또 다른 조건을 추가하고 싶을 때는 else if를 쓰기도 하고 그냥 if를 더 써주기도 하는데 

if-else if-else의 경우에는 조건문중 해당하는 1개만 실행되는 반면

if-if-if의 경우에는 해당 조건에 맞는 모든 조건문이 전부 실행된다.

#include<stdio.h>
int main() {
	int a = 90;
	int count = 0;
	if (a >= 90)
		count++;
	else if (a >= 80)
		count++;
	else if (a >= 70)
		count++;
	else
		count++;
	printf("%d", count);
	return 0;
}
더보기

출력 : 1

#include<stdio.h>
int main() {
	int a = 90;
	int count = 0;
	if (a >= 90)
		count++;
	if (a >= 80)
		count++;
	if (a >= 70)
		count++;
	printf("%d", count);
	return 0;
}
더보기

출력 : 3

 

switch-case문

switch 키워드 다음 괄호안에 들어가는 num이 무슨 값이냐에 따라 그 해당하는 경우를 실행한다.

break는 조건문을 종료하라는 키워드이고

default는 그냥 else라고 생각하면 된다

#include<stdio.h>
int main() {
	int num = 3;
	switch (num) {
	case 1:
		printf("안녕1");
		break;
	case 2:
		printf("안녕2");
		break;
	case 3:
		printf("안녕3");
		break;

	default : 
		printf("안녕안녕");
		break;
	}


		return 0;
}
더보기

출력 : 안녕3

 

break가 없으면 case3부터 아래 모든 조건문이 다 실행된다

#include<stdio.h>
int main() {
	int num = 3;
	switch (num) {
	case 1:
		printf("k");
	case 2:
		printf("o");
	case 3:
		printf("re");

	default : 
		printf("a");
	}


		return 0;
}
더보기

출력 : rea


2. 반복문

반복문에는 for, while, do-while이 있다

먼저 for문이다

 

i=0가 시작값

i<=5가 조건식

i++이 증가감값

 

i가 0부터 시작해서 i가 5가 될때 까지 i가 1씩 증가하면서 반복해라라는 의미이다

#include<stdio.h>
int main() {

	for (int i = 0; i <= 5; i++) {
		printf("%d", i);
	}
	return 0;
}

위 for문은 따라서 012345가 출력이 된다

 

 

다음은 while문이다

while 오른쪽 괄호에 조건식을 넣어준다

i가 5이하일때까지 반복해라

 

그리고 while문은 증감값을 출력문 아래에 적어준다.

출력결과는 012345이다

#include<stdio.h>
int main() {

	int i = 0;
	while (i <= 5){
		printf("%d", i);
	    i++;
	}
	
	return 0;
}

break를 이용해 해당 조건일 경우 while문이 종료되도록 해줄 수도 있다

아래는 0만 출력된다

#include<stdio.h>
int main() {

	int i = 0;
	while (i <= 5){
		if (i %2==1) break; 
		printf("%d", i);
	    i++;
		
	}
	
	return 0;
}

 

또한 continue라는 키워드를 통해 실행문을 실행하지 않고 건너 뛰라고 할 수도 있다

예를 들어 i==1에서 continue를 만났으면 i==1에서 실행될 예정이었던 모든 구문이 건너뛰기 되고 i==2구문이 시작된다

아래처럼 하면 024가 출력이 된다

#include<stdio.h>
int main() {

	int i = 0;
	while (i <= 5) {
		if (i % 2 == 1) {
			i++;
			continue;
		}
		printf("%d", i);
		i++;
	}
	return 0;
}

 

 

for 문은 반복횟수를 알고 있을 때, while 문은 반복 조건위주에 따라 반복할 때 사용된다.

 

 

마지막으로 do-while문이다

조건에 따라 반복을 계속할지를 결정할 때 사용하는 것은 while 문과 동일한데

조건 중괄호 {} 블럭을 한번 실행하고, 조건을 검사하여 반복을 결정한다

#include<stdio.h>
int main() {

	int i = 0;
	do {
		printf("%d", i);
		i++;
	} while (i <= 5);
	
	return 0;
}

출력결과는 012345이다

 

 

 

반복문이 중첩될 수 도 있다

그것을 중첩반복문이라고 한다.

 

간단한 예시를 보겠다. 

아래는 이중 for문이다.

#include<stdio.h>
int main() {
	int k = 1;
	for (int i = 1; i <= 3; i++) {
		
		for (int j = k; j <= k+2; j++) {
			printf("%d ", j);
		}
		printf("\n");
		k += 3;
	}
	return 0;
}

이중for문에 맨 처음for문은 행, 안쪽은 열이라고 생각하면 된다.

i가 1부터 3까지니까 3행이 생기는 것이다.

j도 k부터 k+2까지니까 3열이 생기는 것이다.

 

i=1일때(즉 1행이 만들어졌을 때) j를 k부터 k+2만큼 반복해라(즉 열을 k에서 k+3만큼 만들어라)

i=2일때(즉 1행이 만들어졌을 때) j를 k부터 k+2만큼 반복해라(즉 열을 k에서 k+3만큼 만들어라)

i=3일때(즉 1행이 만들어졌을 때) j를 k부터 k+2만큼 반복해라(즉 열을 k에서 k+3만큼 만들어라)

 

1행 만들어짐-> 3열 만듦

2행 만들어짐-> 3열 만듦

3행 만들어짐->3열 만듦

 

그래서 위에 이중 for문 출력결과를 생각해보자.


3. 배열

배열은 같은 데이터형 변수를 여러개로 묶어놓은 것이다

관련 있는 데이터를 하나의 변수에 할당하여 관리하기 위해 필요하다

아래와 같이 a[]형태가 배열 선언 형태이고 선언과 동시에 {1,2,3,4,5}로 초기화를 했다.

배열의 인덱스는 0부터 시작한다

a[0]=1

a[1]=2

a[2]=3

a[3]=4

a[4]=5

#include<stdio.h>
int main() {
	int a[5] = { 1,2,3,4,5 };
	for(int i=0;i<5;i++){
		printf("%d", a[i]);
	}
	return 0;
}

출력결과가 무엇일지 생각해보자

더보기

12345

 

포인터를 배울 때 얘기하겠지만 배열의 이름은 배열의 주소이다. 

int a=0; 이라는 변수가 있다고 하면 &a를 해야 a변수의 주소를 가리킬 수 있지만 

배열은 a라는 배열 이름만 있어도 변수의 주소를 가리킬 수 있다

 

 

◆배열에서 문자열

'a', 'A'같은 것을 문자라고 하고 문자가 2개이상 결합된 것을 문자열이라고 하는데 배열에 문자열을 저장할 때는 반드시 문자열의 개수+1의 크기로 선언해야한다

문자열의 마지막에는 \0(널 값)이 추가되기 때문이다.

즉, KOREA로 문자가 5개뿐이지만 널값을 포함해야해서 배열길이를 6개로 해줘야한다. 

널 값이 추가되는 이유는 메모리상에서 문자열은 이진 데이터로 저장되기 때문에 문자열의 시작과 끝이 표시되어 있지 않아 구분하기 위해서 존재한다.

#include<stdio.h>
int main() {
	char a[6] = "KOREA";
	printf("%s", a);
	return 0;
}

%s는 문자열을 받는 서식문자로 a나 &a로 사용 가능하다

a로 써도 되는 이유는 a라는 배열의 이름은 배열의 주소를 가리키기 때문이다

 

#include<string.h>를 추가해준 후 strlen이라는 함수로 a라는 배열안에 문자열길이를 구할 수 있다

5가 출력된다

#include<stdio.h>
#include<string.h>
int main() {
	char a[6] = "KOREA";
	
	printf("%d", strlen(a));  //5
	return 0;
}

위 문제를 토대로 문제를 하나 풀어보자.

#include<stdio.h>
#include<string.h>
int main() {
	char a[6] = "KOREA";
	int b = 0;
	b = strlen(a);
	for (int i = 0; i < b; i++) {
		printf("%c", a[i]);
	}
	return 0;
}
더보기

출력결과 : KOREA

 

 

 

◆2차원 배열

2차원 배열은 말그대로 2차원 배열이다.

배열이 이렇게 1차원이였다면 

2차원 배열은 이런식으로 생각하면된다.

아래 처럼 생각하면 된다 x[][] 이런 형태인데

x는 배열의 이름이고 첫번째 []는 행, 두번째 []는 열이라고 생각하면 된다

보기 편해서 이렇게 보는 것이고 실제로 메모리상에는 나란히 이어져있다.

출처 : c언어 2차원 배열 : 메모리 상에 어떻게 저장이 될까요? (tistory.com)

 

 

아래의 출력결과를 생각해보자.

#include<stdio.h>
int a[5][5];
int main() {
	int i, j, k = 1;
	for (i = 0; i < 5; i++) {
		for (j = 4; j >= 0; j--) {
			a[i][j] = k;
			k++;
		}
	}
	for (i = 0; i < 5; i++) {
		for (j = 0; j < 5; j++) {
			printf("%3d", a[i][j]);
		}
		printf("\n");
	}
	return 0;
}

 

 

[정보처리기사/예상문제] - 2023 정보처리기사 실기 예상 문제 모음

C언어 기출문제 모음

비전공자용 C언어 요약 1탄

비전공자용 C언어 요약 2탄(조건문, 반복문, 배열)

비전공자용 C언어 요약 3탄(함수, 포인터)

비전공자용 C언어 요약 4탄(포인터 심화, 구조체)

 

 

 

[정보처리기사/예상문제] - 2023 정보처리기사 실기 예상 문제 모음

C언어 기출문제 모음

비전공자용 C언어 요약 1탄

비전공자용 C언어 요약 2탄(조건문, 반복문, 배열)

비전공자용 C언어 요약 3탄(함수, 포인터)

비전공자용 C언어 요약 4탄(포인터 심화, 구조체)

 

 

C언어 공부하기는 싫고 대충이라도 알아야겠다싶으신분만 보세요

찍먹용이라 아주 간단하게만 적혀져 있습니다

대충 읽어보시고 c언어 기출문제 풀어보시면 될 것 같습니다


목차

1. C언어의 구조

2. 변수와 자료형

3. 서식문자

4. scanf()

5. 사칙연산과 증감연산자

6. 비트연산자


1. C언어의 구조

#include<stdio.h>     
	int main() {
		printf("안녕");
		return 0;
	}

1) #include<stdio.h>는 선행처리기라고 하는데 몰라도 된다.

이 문장이 있어야 printf()함수를 쓸 수가 있다.

 

2) int main()

main() 함수인데 자세한 건 몰라도 된다

그냥 main함수 안에 있는 {   }내용물을 제일 첫번째로 보라는 것이다

 

3) printf()

말그대로 printf() 괄호안에 있는 것을 출력하라는 의미이다

 

4) 맨 뒤 ;(세미콜론)

그냥 "밥을 먹었다."할 때 .(마침표)랑 똑같다고 생각하면 된다

 

5) return 0;

함수가 문제없이 종료됐다는 것을 의미한다

그냥 없다고 생각해도 된다


아래 처럼 코드를 작성하면 어떻게 출력될까?

#include<stdio.h>     
	int main() {
		printf("안녕");
		printf("안녕");
		printf("안녕");
		return 0;
	}

그림1

문장이 다르니까 그림1의 왼쪽 처럼 출력될 것 같지만 줄바꿈기능이 없어 오른쪽처럼 안녕안녕안녕으로 출력된다

 

그렇다면 왼쪽 처럼 출력하려면 어떻게 해야할까 

바로 줄바꿈 기호를 넣어주면 된다

#include<stdio.h>     
	int main() {
		printf("안녕\n");
		printf("안녕\n");
		printf("안녕\n");
		return 0;
	}

줄바꿈기호는 \n(역슬래시 n)인데 보통 엔터키 위에 위치한다.

시험에서 낚시문제로 잘 나오므로 \n있나 없나 잘 봐야 한다


2. 변수와 자료형

변수는 데이터를 저장하는 공간이다

a=1이라고하면 a만 나와도 a안에 1이 있다고 생각하면 된다

 

근데 이때 자료형(데이터 타입)마다 변수를 선언하는 키워드(?)가 다르다

정수형 데이터는 int

실수형 데이터는 double

문자형 데이터는 char

표 출처 :  자료형이란? - 한 눈에 끝내는 C언어 기초 (goorm.io)


3. 서식문자

아래와 같이 출력하면 뭐가 출력이 될까?

#include<stdio.h>     
	int main() {
		int a = 3;
		printf("%d", a);
		return 0;
	}

바로 변수 a의 값인 3이 출력된다.

%d가 출력이 되지 않고 3이라고 출력되는 이유는 %d는 서식문자이기 때문이다.

서식문자는 출력하고자 하는 변수의 출력 형태를 지정하기 위해 사용한다

사진 출처 : [C 언어] 서식문자 (변환문자) 의미 자료형 (tistory.com)

 

 

다음 출력 결과는 어떻게 될까

#include<stdio.h>     
	int main() {
		char a = 'A';
		printf("%d\n", a);
		printf("%c\n", a);

		printf("%d\n", a+1);
		printf("%c\n", a+1);
		return 0;
	}

그림2

출력결과는 그림2처럼 출력된다. char변수로 'A'를 받았는데 서식문자가 %d냐, %c냐에 따라 결과값이 완전 달라지는 것을 볼 수 있다.

A를 정수형으로 출력하면 A의 아스키코드 값인 65가 출력되고

A를 문자형으로 출력하면 A가 그대로 출력된다

 

A를 정수형에서 +1을 하면 아스키코드 값 65+1인 66이 출력되고

A를 문자형에서 +1을 하면 아스키코드 값 66인 문자 B가 출력 된다

 


4. scanf()

scanf()는 변수를 입력받을 수 있도록 해주는 함수이다.

scanf()도 printf()와 마찬가지로 #include<stdio.h>가 있어야 쓸 수 있다. 

 

3을 입력받았다고 할 때 아래의 출력결과는 뭘까?

#include<stdio.h>     
	int main() {
		int a;
		scanf("%d", &a);
		printf("%d", a);
		return 0;
	}

출력결과는 3이다

scanf("%d",&a); 에서

&는  앰퍼샌드(ampersand)라고 읽으며 주소연산자라고도 한다

&a이라고 하면 변수a의 주소를 가리킨다

 

왜 그냥 a가 아니라 &a를 쓸까?

임시공간에 저장됐다가 임시공간에 있는 값을 변수의 주소에 저장하고 그 임시공간을 삭제하므로 변수의 주소가 필요한 것이다.

(이해가 안되면 그냥 넘어가고 scanf가 숫자를 입력받는 함수라는 것만 알고가면 된다)

출처 : [C언어] scanf()함수에서 &(ampersand)를 쓰는 이유 (tistory.com)


5. 사칙연산과 증감연산자

사칙연산 순은 수학에서 배운 것과 동일하다

곱하기(*)와 나누기(/)가 먼저고 그다음이 더하기(+), 빼기(-)이다

여기서 추가로 나머지(%)연산자가 있는데 곱하기, 나누기 우선순위와 동일하다

 

9%2하면 나머지는 1이므로 1이 나온다

 

증감연산자라는 것이 있다. ++a, a++ 이런식으로 표현하는데 -도 동일한 방식이다

증감연산자는 곱하기, 나누기, 나머지보다 먼저 연산된다.

 

다음 출력결과를 맞춰보자

#include<stdio.h>     
	int main() {
		int a = 10;
		int b = 15;
		int c = a++ + --b;
		int d = ++a + b--;
        

		printf("%d\n",a); 
		printf("%d\n", b);  
		printf("%d\n", c); 
		printf("%d", d); 
		return 0;
	}
더보기

12

13

24

26

 

처음보면 살짝 머리터질 수 있는 부분인데

++a는 전치연산이고

a++은 후치연산이다

 

전치연산은 앞에 붙어있어서 바로 계산되고

후치연산은 뒤에 붙어있어서 그 다음 라인부터 계산된다

 

c=a(후치연산)+b(전치연산)=10+14=24

a가 후치연산이라 c에는 a가 10으로 계산되지만

변수c에 값이 저장되는 그 다음부터 바로 a가 11이 된다.

 

d=a(전치연산)+b(후치 연산)=12+14=26

a는 전치연산이라 바로 11에서 1이 더해진 값이 d에 들어가고

b는 후치연산이라 14에서 1만큼 빼진 값이 아니라 그냥 그대로 14가 된다

 

시험에 낚시문제로 많이 출제되니 무조건 이해하고 넘어가자


6. 비트연산자

비트연산자 관련문제는 한번도 출제된 적이 없긴 한데 그래도 쉬우니 보고 넘어가자.

우리가 평소에 사용하는 숫자는 10진수이고 컴퓨터에서는 2진수를 사용한다

2진수는 0과 1로 이루어져있다.

 

3을 2진수로 바꾸면 11이다

2진수를 10진수로 바꾸는 방법은 크게 2가지인데 2번째 방법을 많이 사용한다.

 

일단 빈 백지에 128 64 32 16 8 4 2 1 이렇게 숫자를 적는다.

예를 들어 20을 이진수로 바꿔야한다면 위에 있는 숫자로 20을 만들어야한다

20을 만들려면 16+4면 된다

128 64 32 16 8 4 2 1
0 0 0 1 0 1 0 0

16과 4를 제외한 나머지에는 다 0을 대입해준다.

20의 이진수는 10100이 된다

 

비트연산자에 &, ^ , | 부터 알아본다.

&는 둘다 1일 때만 1반환

^는 두개가 1개는 1, 나머지는 0이어야만 1반환

|는 둘중 1개만 1일 때도 1반환

 

3 & 2 라고 한다면 먼저 3과 2를 이진수로 바꾼다 11과 10이다

11

10

이렇게 봤을 때 둘다 1인 위치만 1이고 나머지는 0이 된다

3 & 2라고 한다면 10으로 2가 된다

3^2라고 한다면 01로 1이 된다

3|2라고 한다면 11로 3이 된다

 

다음은 >>, << 연산자이다

이건 자세한 건 생략하고 푸는 방법만 알려주도록 하겠다.

비트를 이동시키는 연산자인데 

8<<2 라고 하면 8에서 2의 제곱만큼 곱해주면된다

8*4이므로 32가된다

 

8>>2 라고 하면 8에서 2의 제곱만큼 나눠주면 된다

8/4이므로 2가 된다

 

>>1이면 2의 1제곱

>>3이면 2의 3제곱

으로 계산하면 된다

 

[정보처리기사/예상문제] - 2023 정보처리기사 실기 예상 문제 모음

C언어 기출문제 모음

비전공자용 C언어 요약 1탄

비전공자용 C언어 요약 2탄(조건문, 반복문, 배열)

비전공자용 C언어 요약 3탄(함수, 포인터)

비전공자용 C언어 요약 4탄(포인터 심화, 구조체)

[정보처리기사/예상문제] - 2023 정보처리기사 실기 예상 문제 모음

 

1. 인터넷에서 멀티캐스트를 위하여 사용되는 프로토콜

더보기

IGMP


2. 원격 컴퓨터에 안전하게 액세스하기 위한 유닉스 기반의 명령 인터페이스 및 프로토콜로, 기본적으로 22번 포트를 사용하고, 클라이언트/서버 연결의 양단은 전자 서명을 사용하여 인증되며, 패스워드는 암호화하여 보호되는 것

더보기

SSH

 

SSH는 Secure Shell의 약자


3. TLS를 통해 Application 계층 데이터를 암호화하여 보호해 주고 기본포트가 443으로 지정된 프로토콜

더보기

HTTPS


4. 물리적 하드웨어 주소를 IP Address로 매핑시키는 프로토콜

더보기

RARP


5. UDP 계열 프로토콜로, 원격 관리에 필요한 정보와 서버상태를 관리하는 것

더보기

SNMP


6. 동적 라우팅 프로토콜 중에 링크 상태(Link State) 라우팅 프로토콜

더보기

OSPF (Open Shortest Path First)


7. 연결 없는 IP 기반의 프로토콜로 최소한의 오버헤드를 갖고 재송신 처리를 실행하지 못하기 때문에 신뢰성이 떨어지며 한 번에 많은 양의 데이터를 송신할 때 사용하는 프로토콜


8.  RAID 시스템 중 한 드라이브에 기록되는 모든 데이터를 다른 드라이브에 복사해 놓는 방법으로 복구 능력을 제공하며, ′Mirroring′으로 불리는 것

더보기

RAID 1


9. 내부 통신에는 사설 IP 주소를 사용하고 외부와의 통신에는 공인 IP 주소를 사용할 수 있도록 하는 기술


10. 하이퍼바이저를 사용하거나 게스트 운영체제도 설치하지 않고 서버 운영에 필요한 프로그램과 라이브러리만 이미지로 만들어 프로세스처럼 동작시키는 경량화된 가상화 방식이며 실행되는 이미지는 컨테이너라고 부르며, 가상화 레이어가 존재하지 않고 운영체제도 존재하지 않기 때문에 파일 시스템,네트워크 속도가 상당히 빠르다

설명에 해당하는 용어

더보기

Docker 


11. 네트워킹에 필요한 모든 유형의 자원을 추상화하고, 소프트웨어 기반이며 자동으로 관리와 제어가 가능케 하는 가상화 기술

더보기

NFV (Network Functions Virtualization)


12. 한 대의 스위치에서 네트워크를 나누어 마치 여러 대의 스위치처럼 사용할 수 있게 하고, 하나의 포트에 여러 개의 네트워크 정보를 전송할 수 있게 해주는 기능

더보기

가상 랜(Virtual LAN)


13. 링크 상태 알고리즘을 적용한 대표적인 프로토콜로, 링크에서의 전송 시간을 링크 비용(거리)으로 사용하며 각 목적지 별 최단 경로를 다익스트라 알고리즘을 통해 알아내는 프로토콜

더보기

OSPF


14. 패킷 전송의 최적 경로를 위해 다른 라우터들로부터 정보를 수집하는데, 최대 홉이 15를 넘지 못하는 프로토콜


15. OSI 7 layer 참조 모델에서 사용되는 Protocols 중 TCP와 UDP port를 함께 사용하는 프로토콜


16. VPN의 터널링 프로토콜로 OSI 7계층 중 3계층 프로토콜로서 전송 모드와 터널 모드 2가지를 사용한다. 전송 모드는 IP 페이로드를 암호화하여 IP 헤더로 캡슐화하지만, 터널 모드는 IP 패킷을 모두 암호화하여 인터넷으로 전송한다

해당하는 용어

더보기

IPSec


17. Link State 알고리즘을 이용해 서로에게 자신의 현재 상태를 알려주며 네트워크 내 통신을 위해 사용하는 프로토콜

더보기

OSPF


18. TCP/IP 프로토콜 중에서 IP 계층의 한 부분으로 에러 메시지와 같은 상태 정보를 알려주는 프로토콜

더보기

ICMP(Internet Control Message Protocol)


19. 단말이 네트워크에 접근하기 전 보안정책 준수여부를 검사하고 IP 및 MAC address의 인가 여부를 검사하여 네트워크 자원의 이용을 허용하는 방식


20. 서로 연결된 네트워크에서 사용자간 파일을 전송할 수 있도록 해주는 기능을 제공하며, 파일 전송시 2개의 포트를 연결하여 데이터를 전송한다. 이 때 21포트는 데이터 전송을 담당하고 21포트는 제어를 담당하는 프로토콜

더보기

FTP(File Transfer Protocol)


21. 다음은 잘 알려진 포트번호이다. 해당하는 용어를 쓰시오.

포트  이름
21
22
23
25
53
80
110
143
194
443
더보기

ㄱ : FTP

ㄴ : SSH(원격 제어, 보안 기능)

ㄷ : telnet(원격 제어)

ㄹ : SMTP(이메일 전송)

ㅁ : DNS

ㅂ : HTTP(웹)

ㅅ : POP3(이메일 수신)

ㅇ : IMAP(이메인 수신, 보관 기능

ㅈ : IRC

ㅊ:  HTTPS

 

 

[정보처리기사/예상문제] - 2023 정보처리기사 실기 예상 문제 모음

2023 정보처리기사 실기 예상 문제 모음

확실히 출제진이 바뀌어서 그런가 난이도가 대폭 하락하였습니다.

합격률이 꽤나 높을 것으로 예상이 됩니다

 

기출과 똑같거나 비슷한 문제가 무려 9문제나 출제되었으며 2022년 출제되는 형태와 너무 달라 예상하기 더욱 힘들어졌습니다. 

코딩 6문제, SQL 2문제정도가 보통이였는데 이번에는 코딩만 8문제가 출제되었고 SQL도 2문제가 출제되었습니다.

 

SQL 같은 경우는 기존에는 SQLD 시험에 나오는 문제를 사용해서 빈칸을 뚫는 문제 형태였는데 올해는 기출과 완전 동일한 SQL문을 작성하는 형태로 출제가 되었습니다. 

 

코딩이나 SQL만으로 벌써 10문제가 되고 SQL, 코딩만 팠어도 50점은 먹었던 회차이며 기출문제만 봤어서 45점은 그냥 먹었던 회차입니다.

 

2회차를 보면 알수있겠지만 이제 '이 정도 난이도로 내겠다'는 의미인지 '불쌍한 n수생들 이제 좀 나가라'는 의미에서 1회차만 쉽게 낸 것인지는 모르겠으나 많은 사람들의 생각과 동일하게 저도 2회차는 역대급 헬난이도가 될 것으로 예상이 됩니다.

 

이번 1회차에서 기사보다 오히려 산업기사가 어려웠고 우스겟소리로 문제가 바뀌었다는 얘기도 있었는데.. 산업기사 문제는 정확하게 못봐서 모르겠지만 확실한 건 빡세게 공부해야될 것 같습니다

관련 파트 23년 1회
C언어 4
자바
3
파이썬 1
 SQL
계획
요구 분석
설계, 화면설계 1
디자인 패턴(proxy) 
테스트 1
화이트박스 제어흐름 
패키징  0
 DB 2
튜플/인스턴스/카디널리티 
외부/개념/내부 스키마
 네트워크
 4
가상회선/데이터그램
SSH
L2TP
ICMP
 웹
 1
AJAX
 보안
 1
바이러스/웜/트로이목마
운영체제
 자료구조와 정렬

 

이번 2023 정보처리기사 실기 1회를 통해 얻을 수 있는 깨달음

1) 기출은 역시나 무조건 봐야한다

2) SQL은 기존 기출과 동일한 문제가 나오긴 했지만 이제 다시 SQL문장 쓰는 연습도 병행해야한다

3) 보안쪽이 강세였던 2022년에 비해 네트워크 쪽이 왠지 강해진 것 같다(네트워크 공부해야할 듯)

4) c언어는 포인터, 구조체 / 자바는 클래스, 상속 공부 확실하게 해야할 것 같다

5) 코딩/SQL을 제외하고 매회차 무조건 출제되는 개념은 DB, 테스트 파트로 급하면 이 2개 파트는 무조건 봐야할 듯하고 최근 경향에 따라 네트워크, 보안도 보면 될 것 같다

 

기본적으로 공부방법은 아래 링크와 동일할 듯하다

 

정보처리기사 실기 공부법-네이버블로그

2023 정보처리기사 실기 예상 문제 모음

 

+ Recent posts