C에서 변수는 값이 저장되는 메모리 영역이나 변수의 유효범위에 따라 구분되는데 auto, static, extern 이 세가지 키워드(keyword)를 이용하여 구분을 할 수 있다.

변수의 유효범위는 크게 지역변수(local variable)와 전역변수(global variable)로 나뉘며, 실제 프로그램 실행시에 변수가 저장되는 메모리상의 위치에따라 높은 주소값부터 거꾸로 사용하여 내려오는 스택(stack)과, 낮은 주소값부터 올라가면서 사용하는 정적데이터영역(.data, .bss)과 힙(heap)으로 구분된다1. 최적화 관점에서 살펴보면, 스택의 경우 해당 범위(scope)에서 자주 액세스 되며 범위가 끝나면 없어지는 임시변수들이 저장되는데, 스택포인터를 순차적으로 이동해가면서 할당되기 때문에 할당속도가 빠르며, 해당 변수들이 자주사용되어서 CPU 캐시의 힛트율이 높은 경우 접근 속도도 더 빨라질 가능성이 있다. 하지만 범위가 끝나면 없어지기때문에 계속적으로 값을 유지하는 것이 불가능하다. 이러한 문제를 해결하기위해서는 정적인 변수들은 정적 데이터 영역에 저장하여 계속 유지되며, 동적인 변수들은 힙에 변수를 할당하고 저장해야한다. 힙은 비순차적으로 메모리 할당/해제가 계속 일어나는 동적인 특성을 갖고있다. 때문에 스택처럼 메모리 할당을 순차적으로 하기가 힘들어서 정적인 방식에 비해 할당 속도가 느린 특성을 갖는다.

이제 각각의 키워드에 따라 해당 변수의 유효범위와 저장되는 영역이 어떻게 결정되는지를 케이스별로 살펴보도록 하자.

    1. auto (자동범위변수)
      • 일반적인 지역 변수 형태로 블럭 안에서만 유효하며 블럭의 실행이 끝나면 소멸
      • 스택에 메모리 할당
      • 일반적으로 C에서 auto 키워드는 생략되어있음. 즉 아무 표시하지 않은 변수는 auto와 같은 의미.
      • C++에서 auto 키워드를 사용할 경우 “자동 타입 추론”이라는 완전히 다른 의미를 가지게되므로 주의할것.
    2. static (정적변수)
      • 블럭 안에서만 유효한 값을 가지지만 자동변수와 같이 없어지지 않고 블럭으로 다시
        돌아왔을 때 이전 값을 다시 이용 가능(스택이 아닌 정적데이터 영역에 저장됨)
      • 초기화를 생략하면 0으로 자동 초기화
      • 정적 데이터영역에 할당
      • static은 사용되는 위치에 따라 의미가 달라지니 주의해서 사용할것.
        • 내부정적변수 : 함수 내부의 변수에 static이 사용된경우, 해당 변수는 함수 내부에서만 사용이 가능. 하지만, 프로그램이 실행되는 동안 계속 값이 유지됨
        • 외부정적변수 : 함수 외부의 전역범위의 변수에 static이 사용된경우, 해당 파일 내부에서는 전역변수처럼 사용되지만, 아닌 다른 소스파일에서는 참조할 수 없음
          (예: 보통 하나의 example.c 파일 내부에서만 사용하기위한 전역변수로 사용됨. 다른 파일이 example.h를 include하더라도 해당 static으로 선언된 전역변수는 보이지 않아 접근 불가)

       

    3. extern (외부변수)
  • 함수 밖의 전역 범위에 선언되며, 프로그램 전체에서 유효하고 다른 파일에서도 참조 가능
  • 초기화를 생략하면 0으로 자동 초기화
  • 정적 데이터영역에 할당
  • extern 변수는 편리하지만 남발하면 프로그램을 복잡하게 하고 나중에 유지보수가
    힘들기 때문에 사용을 최소화 하는 것이 바람직함
  • 외부변수 정의하는 방법은 아래 extern_example 예제를 참조하고 실제 이를 다른파일에서 사용하는 예시는 myfunction.c 를 보면된다.

 

블로그 이미지

Or71nH

,

https://blog.naver.com/tipsware/221275585536

 

C 언어 표준 함수를 위한 헤더 파일과 소스 파일은 어디에 있을까?

: C 언어 관련 전체 목차 http://blog.naver.com/tipsware/221010831969 이 글은 Visual Studio 2017 버...

blog.naver.com

 

C언어 헤더파일 위치 /usr/include

 

1 개요[편집]

Header file location in CC언어 헤더 파일 위치

CentOS /usr/include
Ubuntu /usr/include

2 참고[편집]

Windows (Visual Studio) Program Files (x86)\Microsoft Visual Studio 12.0\VC\include
블로그 이미지

Or71nH

,

이녀석들은 키다

 

저 sem init 가 열쇠고

 

sem_t 가 세트인듯하다

 

#include <stdio.h>
#include <pthread.h>
#include <semaphore.h>

void * read(void * arg);
void * accu(void * arg);
static sem_t sem_one;
static sem_t sem_two;
static int num;

int main(int argc, char *argv[])
{
	pthread_t id_t1, id_t2;
	sem_init(&sem_one, 0, 0);
	sem_init(&sem_two, 0, 1);
	
	pthread_create(&id_t1, NULL, read, NULL);
	pthread_create(&id_t2, NULL, accu, NULL);
	
	pthread_join(id_t1, NULL);
	pthread_join(id_t2, NULL);	

	sem_destroy(&sem_one);
	sem_destroy(&sem_two);
	return 0;
}

void * read(void * arg)
{
	int i;
	for(i=0; i<5; i++)
	{
		fputs("Iput num: ", stdout);

		sem_wait(&sem_two);
		scanf("%d", &num);
		sem_post(&sem_one);
	}
	return NULL;
}

void * accu(void * arg)
{
	int sum=0, i;
	for(i=0; i<5; i++)
	{
		sem_wait(&sem_one);
		sum+=num;
		sem_post(&sem_two);
	}

	printf("Result: %d \n", sum);
	return NULL;
}

저거 음 저 

 

0

1

키 홈모양인듯 다른 자물쇠와 키인듯

 

'[ 충남인력개발원 ] (2019) > ┗TCP&IP' 카테고리의 다른 글

Mysql 리눅스 연동하기  (0) 2019.11.28
리눅스 체팅 연동  (0) 2019.11.28
쓰레드 주고 받기 덮어쓰기 없에기  (0) 2019.11.27
쓰레드 주고 받기 우아아  (0) 2019.11.27
프로세스 쓰레드  (0) 2019.11.27
블로그 이미지

Or71nH

,

쓰레드를 주고 받을때 같은 값을 가져가서 덮어 쓴느 경우가 생긴다

 

그러면 계산된 값이 둘중에 한 애꺼만 받게 되는데 

그래서 한녀석 계산 끝날때까지 기다렷다가

하면 된다

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <pthread.h>
#define NUM_THREAD 100

void * thread_inc(void * arg);
void * thread_des(void * arg);

long long num=0;
pthread_mutex_t mutex;

int main(int argc, char *argv[])
{
	pthread_t thread_id[NUM_THREAD];
	int i;

	pthread_mutex_init(&mutex, NULL);

	for(i=0; i<NUM_THREAD; i++)
	{
		if(i%2)
			pthread_create(&(thread_id[i]), NULL, thread_inc, NULL);
		else
			pthread_create(&(thread_id[i]), NULL, thread_des, NULL);
	}

	for(i=0; i<NUM_THREAD; i++)
		pthread_join(thread_id[i], NULL);

	printf("result: %lld \n", num);
	pthread_mutex_destroy(&mutex);
	return 0;
}

void * thread_inc(void * arg)
{
	int i;
	pthread_mutex_lock(&mutex);
	for(i=0; i<50000000; i++)
		num+=1;
	pthread_mutex_unlock(&mutex);
	return NULL;
}

void * thread_des(void * arg)
{
	int i;
	pthread_mutex_lock(&mutex);
	for(i=0; i<50000000; i++)
		num-=1;
	pthread_mutex_unlock(&mutex);
	return NULL;
}

이똥색의 뜻을 알았다

쓰레드 중복되고 계속 돌아갈거같은거 똥색임

상관없음 리눅스에선 

 

언제나 에러와 함께하는 나~

 

변함없는 모습 음음 좋아

 

gcc mutex.c -D_REENTRANT -o mutex -lpthread 

이게 중요함 선언할때 스레드 그 장굼도 넣어준다는 뜻

'[ 충남인력개발원 ] (2019) > ┗TCP&IP' 카테고리의 다른 글

리눅스 체팅 연동  (0) 2019.11.28
쓰레드 키주고 받으면서 하기  (0) 2019.11.27
쓰레드 주고 받기 우아아  (0) 2019.11.27
프로세스 쓰레드  (0) 2019.11.27
git hub 추가 하기  (0) 2019.11.27
블로그 이미지

Or71nH

,

int a=0

int *p =0

 

p=&a;

 

void * if(void * arg)

{

     int*p = malloc()

 

return p;

}

 

pthread_join( t_id, &p);

 

남의 void 가서 *변수 하면 값을 가져옴

 

void 를 넘길 때 는 주소를 넘겨 (&변수) 받을 그릇을 넘겨야 값을 받아올수있다

 

 

gcc thread.c -o thread -lpthread

이거 엘피 스레드 꼭 써줘야함 주위!!!!!

 

 

 

tread1.c

 

#include <stdio.h>
#include <pthread.h>
#include <unistd.h>

void* thread_main(void *arg);

int main(int argc, char *argv[])
{
	pthread_t t_id;
	int thread_param=5;

	if(pthread_create(&t_id, NULL, thread_main, (void*)&thread_param)!=0)
	{
		puts("pthread_create() error");
		return -1;
	};

	sleep(10); puts("end of main");
	return 0;
}

void* thread_main(void *arg)
{
	int i;
	int cnt=*((int*)arg);
	for(i=0;i<cnt;i++)
	{
		sleep(1); puts("running thread");
	}
	return NULL;
}

 

 

 

예는 걍 주고 받는 애

부모쓰레드가 10초 기다려주고

5초 기다리는 자식폼

 

thread2.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <pthread.h>
#include <unistd.h>

void* thread_main(void *arg);

int main(int argc, char *argv[])
{
	pthread_t t_id;
	int thread_param=5;
	void * thr_ret;

	if(pthread_create(&t_id, NULL, thread_main, (void*)&thread_param)!=0)
	{
		puts("pthread_create() error");
		return -1;
	};

	if(pthread_join(t_id, thr_ret)!=0)
	{
		puts("pthread_join() error");
		return -1;
	};

	printf("Thread return message: %s \n", (char*)thr_ret);
	free(thr_ret);
	return 0;
}

void* thread_main(void *arg)
{
	int i;
	int cnt=*((int*)arg);
	char * msg=(char *)malloc(sizeof(char)*50);
	strcpy(msg, "Hello, I'am thread~ \n");
	
	for(i=0; i<cnt; i++)
	{
		sleep(1); puts("running thread");
	}
	return (void*)msg;
}

애는 닫아 주기도 하고

 

 

 

thread3.c

#include <stdio.h>
#include <pthread.h>
#include <unistd.h>

void * thread_summation(void * arg);
int sum=0;

int main(int argc, char *argv[])
{
	pthread_t id_t1, id_t2;
	int range1[]={1, 5};
	int range2[]={6, 10};

	pthread_create(&id_t1, NULL, thread_summation, (void *) range1);
	pthread_create(&id_t2, NULL, thread_summation, (void *) range2);
	
	pthread_join(id_t1, NULL);
	pthread_join(id_t2, NULL);
	printf("result: %d \n", sum);
	return 0;
}

void * thread_summation(void * arg)
{
	int start=((int*)arg)[0];
	int end=((int*)arg)[1];
	
	while(start<=end)
	{
		sum+=start;
		start++;
	}
	return NULL;
}

위에 썸이 충돌위험이 있음 왜냐면

전역변수에서 같이 써서 이거 sum 에 같이 드리부움

 

 

thread4.c

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <pthread.h>
#define NUM_THREAD 100

void * thread_inc(void * arg);
void * thread_des(void * arg);
long long num=0;   // long long형은 64 비트 정수 자료형

int main(int argc, char *argv[])
{
	pthread_t thread_id[NUM_THREAD];
	int i;

	printf("sizeof long long: %ld \n", sizeof(long long));
	for(i=0; i<NUM_THREAD; i++)
	{
		if(i%2)
			pthread_create(&(thread_id[i]), NULL, thread_inc, NULL);
		else 
			pthread_create(&(thread_id[i]), NULL, thread_des, NULL);
	}

	for(i=0; i<NUM_THREAD; i++)
		pthread_join(thread_id[i], NULL);

	printf("result: %lld \n", num);
	return 0;
}

void * thread_inc(void * arg)
{
	int i;
	for(i=0; i<50000000; i++)
		num+=1;
	return NULL;
}
void * thread_des(void * arg)
{
	int i;
	for(i=0; i<50000000; i++)
		num-=1;
	return NULL;
}

와 드뎌 최종임 음 왜 갈색이지

 

 

 

스레드를 이용해서 넘겨주고 받고 음 

잘 할당해야지 싸우지않음

 

 

 

애는 오류 날꺼임 이거 해결하는방법임

 

 

'[ 충남인력개발원 ] (2019) > ┗TCP&IP' 카테고리의 다른 글

쓰레드 키주고 받으면서 하기  (0) 2019.11.27
쓰레드 주고 받기 덮어쓰기 없에기  (0) 2019.11.27
프로세스 쓰레드  (0) 2019.11.27
git hub 추가 하기  (0) 2019.11.27
브로드 캐스트  (0) 2019.11.26
블로그 이미지

Or71nH

,