2014년 2월 3일 ~ 2월 28일

약 한 달동안의 단기 집중 교육이 무사히 끝났습니다.


한 달이라는 시간이 누구에게는 짧은 시간이였을 수도, 길었던 시간이였을 수도 있습니다.

저에게 이번 단기 교육은 하루하루를 보면 길었지만, 한 달 전체를 보면 짧았던 시간이 아니였나 싶습니다.


학교다닐 때 아무리 과제가 많아도 몇 일 연속으로 하루에 잠을 한 두시간 자면서 보냈던 적이 없으니까요.


하루를 24시간이 아닌 30시간인 것처럼 보내고, 뭔가 해야할 것도 많고, 

다른 사람과 비교해서 내가 얼마나 부족한 사람인지 느끼게되는 시간이다보니 스트레스도 많이 받았었습니다.


<개인기술세미나>(이하 개기세)라는 것이 있는데, 본인이 갖고 있는 기술과 지원작품을 설명하는 시간이 있습니다.

말이 설명하는 것이지 실제로는 본인을 어필하는 시간이라고 보시면 됩니다.

발표 30분 + 질문 20분(발표 20분 + 질문 30분) = 총 50분 동안 발표를 해야됩니다.


저는 개기세라는 것이 있고 한 번씩 꼭 해야된다는 사실을 알고 정말 걱정이 많았습니다.

처음 멤버십에 들어올 때도 제 실력이 어떤지 알기에 들어오고나서도 불안했고,

동기끼리 기술적인 것들을 말할때도 못알아 듣는 경우도 있었고요.


저의 개기세 결론부터 말하자면...

정말 개망이였습니다ㅋㅋㅋㅋㅋㅋ..

제 밑바닥을 많은 사람들에게, 그것도 어느정도 실력이 있다하는 사람들에게 보였다는 사실이 너무나 창피했습니다.

발표 준비도 시간 채우려고, 보고 또 보고 동영상도 찍어서 편집하고 열심히 준비한다고 했는데

정말 20분도 안되서 끝났습니다...


정말 매일 하루도 빠지지 않고

'내가 왜 뽑혔지?' 라는 생각을 수도 없이 했었습니다.

운영자님들은 그에 대한 대답은 '뽑을 만 하니까 뽑았지! 자신감을 가져!' 라고 말씀해주셨습니다.


솔직히 단기가 끝난 지금도 저 생각은 계속 되고 있습니다.

하지만 처음 저 생각을 할 때와 다른 점이 있다면, '그래! 뽑힐만 하니까 뽑았지! 실력은 공부하고 노력해서 채우자!' 라는 생각을 갖게 되었습니다.


이미 시작된 멤버십 생활!

1년 반동안 위축되있을 수만은 없는 거 아니겠습니까!

어차피 저같은 경우에는 4학년 2학기까지 모두 마쳐서 휴학한 상태이기때문에 멤버십에서 수료할 때까지 지낼 예정인데요.

수업도 없겠다. 공부하고 과제만 하면은 어느정도 따라잡을 수 있지 않을까요?ㅎㅎ


그리고 단기 기간동안 <단기과제>를 진행하게 되는데요.

저희 기수의 경우 한 달 반동안 진행하게 되서 현재 단기는 끝났지만 3월 말까지 단기 과제를 진행한답니다.

단기과제 자체도 아이디어를 짜내는데 정말 힘들었습니다. 

단기과제 끝나고 들어가야 할 <창의과제> 아이디어도 어떤 걸 내야할지 벌써부터 걱정되네요 ㅋㅋㅋ

아! 참고로 단기과제에서 괜찮은 주제의 경우 창의과제로 연계가 가능합니다. 즉, 힘들게 아이디어를 또 낼 필요도 없고 남들보다 프로젝트가 빨리 끝나서 여유로울 수 있다는 장점이 있습니다.


여러분들도 평소 아이디어같은게 있으면 그냥 흘려보내지말고 어디에 적어두는 습관을 들여두시는게 좋습니다.

좋든 나쁘든 나중에 조합하다보면 좋은 아이디어가 나올 수도 있으니까요.


아이디어하니까 생각난건데, 처음 단기가 시작되고 바로 <OT>준비를 하는데요.

OT때 팀파워라고 전국 신입 멤버십이 모인 OT에서 각 멤버십 별로 장기자랑을 하는데, 

여기서도 참신한 아이디어가 필요하답니다ㅋㅋㅋ

그래도 무엇을 하든 과정은 힘들지만 결과는 뿌듯하고 개운한거아니겠습니까~

저희도 비록 상은 받지 못했지만 끝나고 나서 그 때 찍은 동영상 보면서 추억 회상하듯 좋았습니다.


예비 멤버십 회원분들도 이 때 좋은 추억을 남기셨으면 좋겠습니다~ㅋㅋㅋ


이 밖에도 <세미나>, <체육활동>, <게임대회>, <AI대회>, <천문대가기>가 있는데요.

세미나는 전문가분이 오셔서 5일 동안하는 전문가 교육이 있고, 

한 달동안 기존 회원분들이 돌아가면서 다양한 분야의 세미나를 해주십니다.

전문가 교육 내용은 제 블로그에도 포스팅 해놓긴 했지만..저도 너무 어려워서 제대로 포스팅은 못했답니다.

그래도 참고는 할 정도?ㅎㅎ

체육활동, 게임대회, AI대회, 천문대가기 등 되게 재밌고 좋은 활동도 많았습니다.


Posted by 밍쫑
,

이 날은 아이디어 생각하고 수강신청 도와주려다가 밤을 새는 바람에....많이 졸아서 필기가 영 부족하다 ㅠㅠ..

그래서 대신 강사님의 소스 파일을 공유하고자 합니다.



1일차에 이어서 동일하게 프로젝트를 생성합니다.


프로젝트명 : 20140211


1일차 복습하고 시작합니다.

파일명 : 1.c



파일명 : 2.c

// void 포인터
#include <stdio.h>

void swap(int *a, int *b) {
	int t = *a;
	*a = *b;
	*b = t;
}

void swap(int *a, int *b) {
	short t = *a;
	*a = *b;
	*b = t;
}

void swap(int *a, int *b) {
	char t = *a;
	*a = *b;
	*b = t;
}

void main() {
	int a = 30, b = 1000;
	swap(&a, &b);
	printf("a = %d, b = %d\n", a, b);
}

c언어에서는 동일한 이름의 함수이름들을 전부 swap이 _swap으로 바뀐다.


그래서 swap_xx함수 이름을 바꿔준다. -> swap_int

a

// void 포인터
#include <stdio.h>

void swap_int(int *a, int *b) {
	int t = *a;
	*a = *b;
	*b = t;
}

void main() {
	int a = 30, b = 1000;
	swap(&a, &b);
	printf("a = %d, b = %d\n", a, b);
}

전처리 기능은 문자열을 단순히 치환만 하기 때문에 .....

// void 포인터
#include <stdio.h>

#define SWAP(x,y,T) do { 
T t = a; 
a - b; 
b= t; 
} while(0)

void main() {
	int a = 30, b = 1000;
	//swap(&a, &b);
	SWAP(a, b, int);
	printf("a = %d, b = %d\n", a, b);
}

하나의 함수를 여러 타입을 핸들링하고 싶은 것의 문제이다..!

함수가 어떤 타입이 되든지 간에 모두 SWAP을 하려면 중간에 있는 임시 변수를 char로 한 바이트를 주고

아래 소스 코드는 타임에 의존적이지 않는 generic swap을 만든것이다.


#include <stdio.h>

/*#define SWAP(x,y,T) do { 
T t = a; 
a - b; 
b= t; 
} while(0) */
void swap(void *a, char *b, int size) {
	char t;
	char *aa = (char*)a;
	char (bb = (char*)b;
	int i;

	for(i=0;i


파일명 : 3.c






파일명 : 4.c









알아둡시다!!

구조체와 포인터


구조체 포인터 연산자: (*)., / ->

// 자기 참조 구조체 : 자기 자신의 타입에 대한 포인터


파일명 : 5.c



메모리를 보통 노드라고 한다.

리스트 자료구조 : 데이터를 저장하기 위한 메모리 -> 노드(node)

본인의 마지막 노드를 터미널이라고 한다. 종료 노드와 시작 노드를 구분하기 위해서 마지막 노드에 null을 넣어 놓는다.

끝을 tail이라고 한다.

리스트 자료구조 구현 방법은 여러개 있다.


파일명 : 6.c






한 방향으로 연결된 노드의 경우!

만약 3000개가 연결되었다.. 근데 찾고자하는게 마지막에 있다면??? 최악이지 않은가...

그래서 배울게 바로!! reverse() - PPT 13p : 기존의 display()는 head부터 시작한다. 따라서 tail부터 시작하는 reverse용 display를 따로 만들어야 된다.

void reverse(NODE *head, NODE *tail){
	NODE *prev = head;
	NODE *curr = head->next;
	NODE *next;

	while(cur != tail) {
		next = curr->next;
		curr->next = prev;
		prev = curr;
		curr = next;
	}

	curr->next = prev;
}

a

문제점이 있다면, 어떤게 reverse로 보고 어떤것을 reverse_display로 봐야 할지 모른다. 근데 별 쓸모없는 기능이다.

그리고 이 함수는 역방향(reverse)에 대한 순방향 복원이 안된다. 오버헤드가 크다!!

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

typedef struct __NODE
{
	int data;
	struct __NODE *next;
	struct __NODE *prev;	//double linked list!
} NODE;

void init_list(NODE *head)
{
	head->next = head;
	head->prev = head;
}

//PPT 17p
void insert_front(NODE *head, NODE *node)
{
	node->next = head->next;
	head->next = node;

	node->prev = node->next->prev;
	node->next->prev = node;
}

void insert_end(NODE *head, NODE *node)
{
	NODE *cur = head;

	while(cur->next != head)
		cur = cur->next;

	node->next = cur->next;
	cur->next = node;
}

void reverse(NODE *head)
{
	NODE *prev = head;
	NODE *curr = head->next;
	NODE *next;

	while(curr != head)
	{
		next = curr->next;
		curr->next = prev;
		prev = curr;
		curr = next;
	}

	curr->next = prev;
}

void display(NODE *head)
{
	NODE *node;

	system("cls");
	printf("\n[head]->");
	for(node = head->next; node != head; node = node->next)
		printf("[%d]->", node->data);
	printf("[head]");
	getchar();
}



void main()
{
	int i;

	NODE head;

	NODE arr[5];

	init_list(&head);

	display(&head);
	for(i = 0; i < 5; i++)
	{
		arr[i].data = i+1;
		//insert_front(&head, &arr[i]);
		insert_end(&head, &arr[i]);
		display(&head);
	}

	reverse(&head);
	display(&head);

	reverse(&head);
	display(&head);
}

// 역방향에 대한 순방향 복원이 안된다!

양방향 노드를 이용하면 라이브러리의 최고 동력을 살릴 수 있지만, 메모리는 2배 희생이 필요하다.

이 양방향 노드를 더블 링크라고 한다.




1.c


2.c


3.c


4.c


5.c


6.c







Posted by 밍쫑
,

집중 단기 시작 세미나 ('자료구조와 알고리즘')

2014년 2월 10일~ 2월 14일 진행


강형민 외부 강사님(checkdisk@ioacademy.co.kr)

강의 시간 : 10시 ~ 17시


사용할 언어 : C언어

개발환경 : MS Visual Studio 2010


1일차     : 고급 C 언어

2~3일차 : 자료구조(리스트, 트리, 스택, 큐, ...)

4~5일차 : 알고리즘(압축, 암호화, 패턴 매칭, ...)

마지막 날에 테스트 볼 예정. 시험 난이도는 수업시간에 배운 내용 정도




1. 

프로젝트명 : 20140210

추가 옵션 : 빈 프로젝트 체크



2. 솔루션 탐색기의 소스 파일 폴더에서 새 항목을 추가한다.


3. c++파일을 선택하고 파일명은 1.c로 준다.

확장명이 c언어가 없다. 하지만 c언어로 만들 수 있다. 확장자를 통해서 c++/c를구분한다.

따라서 c언어로 개발을 할 경우 *.cpp를 선택 한 후 파일 이름을 쓴 후확장명을 *.c로 해주어야한다. )


4. 소스코드 작성 후 디버깅하지 않고 시작(Ctrl + F5)을 한다.


5.

1.c 파일의 속성에서 '빌드에서 제외' - '예' 로 바꿔준다.


6. 빌드에서 제외 됬음이 나타난 것을 확인한다.

decay 사용하는 이유는 성능상의 오버헤드를 줄이기 위해서이다.

int arr[3];의 경우, arr이 전체 타입인 int를 따른다. 하지만 


배열을 선언할 때에는 배열의 심봉를 항상 타입과 길이 사이에 와야 한다.(중요중요!)



#include <stdio.h>

// arr와 &arr의 의미

void main()
{
	int arr[3];

	arr;	// 배열의 시작주소, 타입은 첫 번째 원소
	&arr;	// 배열의 시작주소, 타입은 배열 전체

	printf(" arr = 0x%p\n", arr);
	printf("&arr = 0x%p\n", &arr);
}

#include <stdio.h>

// arr와 &arr의 의미

void main()
{
	int arr[3];

	int *p1 = arr; // 1. 일중 포인터 : 다른 지역에 있는 메모리에 접근하기 위해서 사용
	int [3] *p2 = &arr /*하지만 (중요중요!)때문에 int *p2[3] = &arr;을 해야하는데 이 경우 오류이다.
	첫번째 심볼을 찾고 뒤에 있는 기호를 본다. 여기서 심볼은 p2. 뒤에 있는 [이다. 따라서 배열! 
		이게 포인터변수인 것을 알리려면 int (*p2)[3]2.배열 포인터으로 해 줘야 한다.
		이때 p1은 일중포인터 p2는 배열 포인터라고 한다.*/

	printf(" arr = 0x%p\n", arr);
	printf("&arr = 0x%p\n", &arr);
}


파일명 : 4.c

#include <stdio.h>
void main() {
	int arr[2][3];	// 원소의 갯수는 몇 개?

int arr[2][3]; 이것은 사실 2차원 배열의 원소가 아니라 1차원 배열이다.

2차원 배열은 2차원이 아니라 1차원 배열의 확장이다.

따라서 위의 것을 다시 쓰면 int[3] int[2];라 쓸수 있고, 이것은 int 타입의 길이가 3개인 1차원 배열을 원소로 하는 1차원 배열이다.



#include <stdio.h>

void foo(int arr[][3]) {

}

void main() {
	int arr[2][3]; // int arr[][3] = arr; // int arr[][3] = arr;	//error, 함수의 매개변수에 특화된 문법
}

11111111111111
 

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

void foo(int (*arr)[][3]) {
}

void main() {
	int arr[2][3]; // int arr[][3] = arr; // int arr[][3] = arr;	//error, 함수의 매개변수에 특화된 문법
int (*pArr)[3] = arr }

테스트1! 힙에 행은 2개이고 열은 3개인 int타입의 동적 2차원 배열을 선언해보세요!

#include <stdio.h>
#include <stdlib.h>
void main() {
  int(*p)[3] =  malloc(sizeof(int)*2*3); // 왜 malloc에서 빨간 줄이 나타난걸까?

지금 우리의 컴파일러는 cpp컴파일러이기 때문에


for(int i=0;i<2;i++) {
  for(int j=0;j<3;j++) 
   p[i][j] = 3;
 }

  for(int i=0; i<2; i++) {
   for(int j=0; j<3; j++) 
    printf("%d\n", p[i][j]); 
  }
}

여기서 컴파일 에러가 없고, 분명 실행도 잘 되는데 malloc 부분에서 빨간 줄이 나 있을 것이다. 이것은 현재 우리의 컴파일러가 cpp컴파일이기 때문에 그런 것이지, 소스 코드 상의 문제는 없는 것이다.

malloc의 빨간 줄이 신경 쓰이는 경우(암식적인 표현)에는 명시적 표현으로 바꿔준다.
int(*p)[3] = (int(*)[3])malloc(sizeof(int)*2*3); //이 경우 cpp컴파일러에서 빨간 줄이 나타나지 않을것이다.
하지만 이와 같은 경우 c언어 스타일이기 때문에 cpp스타일로 가려면 
static_cast<int(*)[3]>(malloc....);을 넣어주면된다.


파일명 : 5.c

2차원 배열이 왜 2차원이 아닌가?

여기서 배열의 주소는 왜 0부터 시작할까? .........

포인터와 포인터의 사칙연산은 되지 않지만, 제한적인 연산은 가능하다.

#include <stdio.h>
// 포인터의 연산은 사칙 연산은 불가능하나
// 제한적인 연산은 가능하다. void main() {
	int *p, *q;

	p + q;	/*과연 실제로 안될까?된다. 
			이것을 값으로 볼지 주소로 볼지를 선택하는건 사용자의 선택인데, 
			이 경우 컴파일러가 막은 것이다.
			보안상의 위험성을 방지하고자 OS가 죽여버린다.
			주소와 주소가 더해지면은 의미없는 주소가 나올 확률이 높다.
			이것을 역참조해서 찾아가면 허가없는 메모리 참조가 되지 않을까요?
			그러므로 컴파일 에러를 내는 것이 좋다. 그래서 컴파일러가 막는 것이다.*/
	p - q;	//포인터 - 포인터 = 정수
	p * q;	//곱셈은 덧셈의 연속이기 때문에 되지 않는다.
	p / q;
}

p + q의 경우, 실제로는 되는 연산이다. 

하지만 이것을 값으로 볼지 주소로 볼지를 선택하는건 사용자의 선택인데, 이 경우 컴파일러가 막은 것이다.

보안상의 위험성을 방지하고자 OS가 죽여버린다.

주소와 주소가 더해지면은 의미없는 주소가 나올 확률이 높다.

이것을 역참조해서 찾아가면 허가없는 메모리 참조가 되지 않을까?

그러므로 컴파일 에러를 내는 것이 좋다. 그래서 컴파일러가 막는 것이다.


외웁시다!

//포인터 + 정수 = 포인터

// 포인터 - 정수 = 포인터

// 포인터 - 포인터 = 정수


테스트2! 다음 배열에서 첨자 연산을 사용하지 않고 배열의 중간값을 구해보세요.


void main() {
	int arr[5] = {1,2,3,4,5}; 
	int *center =??????????; //&arr[2];//&arr[2];

	printf("%d\n", *center);	//3
}




배열 첨자 연산




void main(){
	int arr[2][3] = {{1,2,3},{4,5,6}};

	printf("%d\n", arr[1][2]);

	printf("%d\n", *(arr+1)[2]);
	//printf(" arr = 0x%p\n", arr);
	//getchar();
	//printf("arr+1 = 0x%p\n", arr+1);/
}
/*이와같이 비정상적인 연산 값이 나온다.
별표보다 첨자가 우선이다. 그러므로 행 단위부터 접근하기 위해서 별표를 먼저 해석시키기 위해
(*(arr+1))을 해야한다.*/

// 결론 : 2차원 배열은 이중 포인터가 아니다.

void main() {
	int arr[2][3] = {1,2,3,4,5,6};

	printf("%d\n", **arr);	// 1이 나온다. 
				// 이것은 arr[0][0] = *(*(arr + 0) + 0); 와 같다. 
				// 결국 2차원 배열은 2중 포인터처럼 보인다. 0이 전부 상쇄되고 있다.
				// 이것은 어느순간 죽게 되어 있다. 이것은 그저 1차원 포인터의 확장일 뿐..
}


파일명 : 8.c


//이중 포인터의 개념

#include <stdio.h>
int div(int a, int b) {
	 return a/b;
}
void main() {
	int a = 4, b = 2;

	printf("%d\n", div(a,b));
} 



return -1;의 리턴 결과 값이 숫자 -1일지 에러 코드 값 -1일지 헷갈리므로 좋은 코드는 아니다.그래서 C++나 Java에서는 예외처리로 해준다.


void main() {
	char name[1024];

	printf("input name: ");
	scanf("%s", name);
	
	printf("your name: %s\n", name);
}

좋은 코드가 아니다. 왜냐하면 사용자로부터 이름을 5번 입력받는다면 이 코드가 5번이 있어야된다.. 이렇게 반복되는 코드는 함수로 빼는게 좋다. (밑에 get_name() 부분)





정리!!!

일중 포인터를 쓰는 이유는 다른 지역의 일반 변수의 심볼에 접근하기 위하여 사용!

이중 포인터를 쓰는 이유는 다른 지역의 포인터 변수의 심볼에 접근하기 위하여 사용!





파일명 : 9.c

이러한 포인터 배열은 쓰는 이유는 모든 데이터를 힙에다가 저장하기 위해서



//3개의 이름을 저장하는 프로그램을 구현해보자!
void mian() {
	char names[3][1024];

	int i;
	for(i=0;i<3;i++) {
		printf("input name: ");
		scanf("%s", names[i]);
	}

	for(i=0;i<3;i++) {
		printf("name[%d]: %s", i, names);
	}
}
/*이 프로그램은 메모리 낭비가 심하다*/

asdf1





삼중 포인터


	//삼중포인터
#include <stdio.h>

void get_name(char** *p) {

}

void main() {
	char **names;

	get_name(&names);
}

숙제!! 이중 포인터를 삼중포인터로 바꾸기


파일명 : a.c

함수의 선언 : 함수의 심볼은 반드시 리턴 타입과 매개 변수 사이에 위치해야한다.

리턴타입 함수명([파라미터, ...]);








void(* hoo())(int) : 심볼은 hoo인데 앞에는 포인터 뒤에는 함수호출연산자. 뒤가 함수호출이니까 hoo는 함수호출연산자

void(* zoo( void(*fp)(int)))(int) : 심볼은 zoo인데 리턴하는 것은 포인터인데 포인터의 타입은 int이다. 함수포인터를 리턴하고 있다..


근데 누가 이렇게 길고 복잡하게 쓸까? 아니..안쓴다. 그래서 alias(별명)을 붙여서 쓴다.






1.c


2.c


3.c


4.c


5.c


6.c


7.c


8.c


9.c


Posted by 밍쫑
,