philosophers - KimTaebin-ai/study_posts GitHub Wiki

ํ•„์š” ๊ฐœ๋… ๋ฐ ํ•™์Šต ๋‚ด์šฉ

๋ฉ€ํ‹ฐ ํƒœ์Šคํ‚น, ๋ฉ€ํ‹ฐ ์“ฐ๋ ˆ๋”ฉ

๋™๊ธฐํ™”

  • ๋ฎคํ…์Šค(๋ฎคํ…์Šค ๋ฝ)

  • ์„ธ๋งˆํฌ์–ด

  • ์‹์‚ฌํ•˜๋Š” ์ฒ ํ•™์ž

  • ๋ฐ์ดํ„ฐ ๋ ˆ์ด์Šค

๊ฐœ์š”

int sum;

void *run1(void *param)
{
    int i;
    for (i = 0; i < 10000; i++)
        sum++;
    pthread_exit(0);
}

void *run2(void *param)
{
    int i;
    for (i = 0; i < 10000; i++)
        sum++;
    pthread_exit(0);
}

int main()
{
    pthread_t tid1, tid2;
    pthread_create(&tid1, NULL, run1, NULL);
    pthread_create(&tid2, NULL, run2, NULL);
    pthread_join(tid1, NULL);
    pthread_join(tid2, NULL);
    printf("%d\n", sum);
}

// register1 = sum;
// register1 = register1 + 1;
// sum = register1;
#include <pthread.h>
#include <stdio.h>

int sum;
pthread_mutex_t mutex;

void *run1(void *param)
{
    int i;
    for (i = 0; i < 10000; i++) {
        pthread_mutex_lock(&mutex);
        sum++;
        pthread_mutex_unlock(&mutex);
    }
    pthread_exit(0);
}

void *run2(void *param)
{
    int i;
    for (i = 0; i < 10000; i++) {
        pthread_mutex_lock(&mutex);
        sum++;
        pthread_mutex_unlock(&mutex);
    }
    pthread_exit(0);
}

int main()
{
    pthread_t tid1, tid2;
    pthread_mutex_init(&mutex, NULL);

    pthread_create(&tid1, NULL, run1, NULL);
    pthread_create(&tid2, NULL, run2, NULL);

    pthread_join(tid1, NULL);
    pthread_join(tid2, NULL);

    printf("%d\n", sum);

    pthread_mutex_destroy(&mutex);
    return 0;
}

์ด๋ก  ํ•ด๊ฒฐ

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

#define true 1
#define NUM_PHILS 5 // ์ฒ ํ•™์ž๋“ค์˜ ์ธ์›์ˆ˜

// THINKING : ์ƒ๊ฐ์ค‘์ธ  ์ƒํƒœ
// HUNGRY   : ๋ฐฐ๊ณ ํ”ˆ    ์ƒํƒœ
// EATING   : ๋ฐฅ์„ ๋จน๋Š” ์ƒํƒœ
enum
{
    THINKING,
    HUNGRY,
    EATING
} state[NUM_PHILS];

pthread_mutex_t mutex_lock;
pthread_cond_t cond_vars[NUM_PHILS];

void init();
int leftOf(int i);
int rightOf(int i);
void *philosopher(void *param);
void think(int id);
void eat(int id);
void pickup(int id);
void putdown(int id);
void test(int i);

int main()
{

    pthread_t tid;
    init();
    for (int i = 0; i < NUM_PHILS; i++)
    {
        pthread_create(&tid, NULL, philosopher, (void *)&i);
    }
    for (int i = 0; i < NUM_PHILS; i++)
    {
        pthread_join(tid, NULL);
    }
    return 0;
}

void init()
{
    for (int i = 0; i < NUM_PHILS; i++)
    {
        state[i] = THINKING;
        pthread_cond_init(&cond_vars[i], NULL);
    }
    pthread_mutex_init(&mutex_lock, NULL);
    srand(time(0));
}

int leftOf(int i)
{
    return (i + NUM_PHILS - 1) % NUM_PHILS;
}

int rightOf(int i)
{
    return (i + 1) % NUM_PHILS;
}

void *philosopher(void *param)
{
    int id = *((int *)param);
    while (true)
    {
        think(id);
        pickup(id);
        eat(id);
        putdown(id);
    }
}

void think(int id)
{
    printf("%d: Now, I'm thinking...\n", id);
    usleep((1 + rand() % 50) * 10000);
}

void eat(int id)
{
    printf("%d: Now, I'm eating...\n", id);
    usleep((1 + rand() % 50) * 10000);
}

void pickup(int id)
{
    pthread_mutex_lock(&mutex_lock);

    state[id] = HUNGRY;
    test(id);
    while (state[id] != EATING)
    {
        pthread_cond_wait(&cond_vars[id], &mutex_lock);
    }

    pthread_mutex_unlock(&mutex_lock);
}

void putdown(int id)
{
    pthread_mutex_lock(&mutex_lock);

    state[id] = THINKING;
    test(leftOf(id));
    test(rightOf(id));

    pthread_mutex_unlock(&mutex_lock);
}

void test(int i)
{
    // ๋งŒ์•ฝ ์ฒ ํ•™์ž i๊ฐ€ ๋ฐฐ๊ณ ํ”ˆ ์ƒํƒœ์ด๊ณ  ์ด์›ƒํ•˜๋Š” ์ “๊ฐ€๋ฝ์ด eating ์ƒํƒœ๊ฐ€ ์•„๋‹ˆ๋ผ๋ฉด
    // ๋ฐฅ์„ ๋จน์Šต๋‹ˆ๋‹ค.
    if (state[i] == HUNGRY &&
        state[leftOf(i)] != EATING &&
        state[rightOf(i)] != EATING)
    {
        state[i] = EATING;
        pthread_cond_signal(&cond_vars[i]);
    }
}

๋ฌธ์ œ ๊ฐœ์š”

arguments

์ฒ ํ•™์ž ์ˆ˜, ์ˆ˜๋ช…, ๋ฐฅ์„ ๋จน๋Š”๋ฐ ๊ฑธ๋ฆฌ๋Š” ์‹œ๊ฐ„, ์ž ์ž๋Š” ์‹œ๊ฐ„, ๊ฐ ์ฒ ํ•™์ž๊ฐ€ ์ตœ์†Œํ•œ ๋ฐฅ์„ ๋จน์–ด์•ผ ํ•˜๋Š” ํšŸ์ˆ˜

int		number_of_philosophers;
/*
	์ฒ ํ•™์ž์˜ ์ˆ˜ :
	์ฒ ํ•™์ž์˜ ์ˆ˜์™€ ํฌํฌ์˜ ์ˆ˜์ž…๋‹ˆ๋‹ค.
*/

int		time_to_die;
/*
	์ฒ ํ•™์ž์˜ ์ˆ˜๋ช…(๋ฐ€๋ฆฌ์ดˆ ๋‹จ์œ„) :
	์ฒ ํ•™์ž๊ฐ€ ๋งˆ์ง€๋ง‰์œผ๋กœ ๋ฐฅ์„ ๋จน๊ธฐ ์‹œ์ž‘ํ•œ ์‹œ์ ์œผ๋กœ๋ถ€ํ„ฐ
	time_to_die ์‹œ๊ฐ„๋งŒํผ์ด ์ง€๋‚˜๊ฑฐ๋‚˜,
	ํ”„๋กœ๊ทธ๋žจ ์‹œ์ž‘ ํ›„ time_to_die ์‹œ๊ฐ„๋งŒํผ์ด ์ง€๋‚˜๋„๋ก
	์‹์‚ฌ๋ฅผ ์‹œ์ž‘ํ•˜์ง€ ์•Š์œผ๋ฉด ํ•ด๋‹น ์ฒ ํ•™์ž๋Š” ์‚ฌ๋งํ•ฉ๋‹ˆ๋‹ค.
*/

int		time_to_eat;
/*
	๋ฐฅ์„ ๋จน๋Š”๋ฐ ๊ฑธ๋ฆฌ๋Š” ์‹œ๊ฐ„(๋ฐ€๋ฆฌ์ดˆ ๋‹จ์œ„) :
	์ฒ ํ•™์ž๊ฐ€ ๋ฐฅ์„ ๋จน๋Š”๋ฐ ๊ฑธ๋ฆฌ๋Š” ์‹œ๊ฐ„์ž…๋‹ˆ๋‹ค.
	ํ•ด๋‹น ์‹œ๊ฐ„๋™์•ˆ ์ฒ ํ•™์ž๋Š” ๋‘ ๊ฐœ์˜ ํฌํฌ๋ฅผ ์žก๊ณ  ์žˆ์–ด์•ผ ํ•ฉ๋‹ˆ๋‹ค.
*/

int		time_to_sleep;
/*
	์ž ์ž๋Š” ์‹œ๊ฐ„(๋ฐ€๋ฆฌ์ดˆ ๋‹จ์œ„) :
	์ž ์„ ์ž๋Š” ๋ฐ ์†Œ๋ชจ๋˜๋Š” ์‹œ๊ฐ„ ์ž…๋‹ˆ๋‹ค.
*/
int		number_of_times_each_philosopher_must_eat;
/*
	๊ฐ ์ฒ ํ•™์ž๊ฐ€ ์ตœ์†Œํ•œ ๋ฐฅ์„ ๋จน์–ด์•ผ ํ•˜๋Š” ํšŸ์ˆ˜(์„ ํƒ์  ์ธ์ž):
	๋ชจ๋“  ์ฒ ํ•™์ž๊ฐ€ number_of_times_each_philosopher_must_eat ๋ฒˆ ์ด์ƒ
	๋ฐฅ์„ ๋จน์œผ๋ฉด ์‹œ๋ฎฌ๋ ˆ์ด์…˜์ด ์ข…๋ฃŒ๋ฉ๋‹ˆ๋‹ค.
	์ง€์ •๋˜์ง€ ์•Š์€ ๊ฒฝ์šฐ, ์ฒ ํ•™์ž๊ฐ€ ์ฃฝ์„ ๋•Œ ์‹œ๋ฎฌ๋ ˆ์ด์…˜์ด ์ข…๋ฃŒ๋ฉ๋‹ˆ๋‹ค.
*/

์ถœ๋ ฅ๊ฐ’

timestamp_in_ms X has taken a fork
timestamp_in_ms X is eating
timestamp_in_ms X is sleeping
timestamp_in_ms X is thinking
timestamp_in_ms X died

๊ฐœ๋ฐœ ์ˆœ์„œ

  • 1 argv
  • 2 invalid check
  • 3 table init
  • 4 philosopher init
  • 5 eat func
  • 6 sleep func
  • 7 think func
  • 8-1 argc๊ฐ€ 5๊ฐœ์ผ ๋•Œ ์ฃฝ๋Š” ์ฒ ํ•™์ž ์ˆ˜ die
  • 8-2 argc๊ฐ€ 6๊ฐœ์ผ ๋•Œ must ์ฒ˜๋ฆฌ
  • 9 pthread join ๊ธฐ๋‹ค๋ฆฐ ํ›„ free
โš ๏ธ **GitHub.com Fallback** โš ๏ธ