2017-09-23 4 views
0

私は2つのスレッドを作成するプログラムを作成しています。各スレッドは、各行に1つの文字を含む1つのテキストファイルを読み込みます。第二のようにフォーマットされPthreadの同期化:2つのテキストファイルの逆順読み込み

h 
0 
h 
0 
... 

0 
i 
0 
i 
0 
i 

時にはお互いの後にお互いの後に複数の文字、または複数のゼロが存在することができる

まずようにフォーマットされます。しかし、1つのファイルの1行に文字がある場合、2番目のファイルの対応する行は0になり、その逆も同様です。

スレッドは、ゼロに達するまで、ファイル入力をグローバルchar配列に読み込み続けることになっています。この時点で、他のスレッドが引き継ぐことができます。そして、両方のファイルが完全に読み込まれるまで、彼らは前進し続けます。

この時点で、私は(1)多くのhの後に多くのiまたは(2)(の正解)のヒヒの連続ストリーム、続いて多くのhが続きます。だから、私は自分の同期方法がオフであることを知っています。ここで

は私のスレッドの1の例である:

int main(void) 
{ 
pthread_t id1; 
pthread_t id2; 

pthread_create((&id1), NULL, getMessage1, NULL); 
pthread_create((&id2), NULL, getMessage2, NULL); 

pthread_join(id1, NULL); 
pthread_join(id2, NULL); 

int j; 

for (j = 0; j < 1001; j++) 
{ 
    printf ("%c ",message[j]); 
} 

return 0; 
} 
:ここ (。両方のスレッドが開かれているファイルを除いて、まったく同じであることに注意してください)

void *getMessage1() 
{ 
FILE *studentOne = fopen("Student1", "r"); 

size_t howManyChars; 
char *placeHolderChars; 
int count = 1; 
while (count < 501) 
{ 
    placeHolderChars = NULL; 
    getline(&placeHolderChars, &howManyChars, studentOne); 

    if(strcmp(placeHolderChars, "0\n") == 0) //if we've reached a zero 
    { 

     pthread_mutex_unlock(&lock); 
    } 
    else 
    { while(1) 
     { 
      if(pthread_mutex_trylock(&lock) == 0) 
      { 

       break; 
      } 
     } 

     if(strlen(placeHolderChars)>0) 
     { 
      placeHolderChars[1] = '\0'; 
     } 

     strcat(message,placeHolderChars); 
    } 

    free(placeHolderChars); 

    if(feof(studentOne)) 
    { 

     pthread_mutex_unlock(&lock); //unlock 
     fclose(studentOne); 
     break; 
    } 
    count++; 

} 

return 0; 
} 

は私の主な方法であり、

ロック、アンロック、待機、および/または信号を使用して、一貫性のある結果が得られる同期同期テクニックを作成する方法についてのガイダンスをお寄せいただきありがとうございます。

+0

セマフォ。あなたはミューテックスでこれを行うことはできません。マルチデュープ。 –

+0

私の教授は実際にmutexを使うことを提案しました。だから、私はミューテックスをスクラップし、バイナリセマフォまたは何かを使用する必要がありますか? – Selena

+0

また、両方のスレッドが使用する1つの機能しか持たないと考えるべきですか? (そして、ファイルをパラメータとして渡しますか?)それで何か違いはありますか? – Selena

答えて

1

ここでは、あなたが望むことをするプログラムでの試みがあります。しかし、不十分にテストされた;)

#include <iostream> 
#include <fstream> 
#include <string> 

#include <cassert> 
#include <pthread.h> 



using std::cout; 
using std::ifstream; 
using std::string; 

const string FILE1("file1.txt"); 
const string FILE2("file2.txt"); 

enum State 
{ 
    UNINITIALIZED, 
    THREAD_ONE_READS, 
    THREAD_TWO_READS 
}; 

struct ThreadInfo 
{ 
    State state; 
    string filename; 
}; 

State state = UNINITIALIZED; 
pthread_mutex_t mut = PTHREAD_MUTEX_INITIALIZER; 
pthread_cond_t cond = PTHREAD_COND_INITIALIZER; 

void* thread_func(void* arg) 
{ 
    // Open file 'h'.                                                                                                     

    ThreadInfo ti = *reinterpret_cast<ThreadInfo*>(arg); 
    ifstream infile; 
    infile.open (ti.filename.c_str(), std::ifstream::in); 

    // while (not EOF)                                                                                                     
    // Read 'h' or 'i': until 0 reached. Wake up other thread.                                                                                          

    string line; 
    getline(infile, line); 
    while (infile.good()) 
    { 
     cout << "Thread " << pthread_self() << " read " << line << '\n'; 
     pthread_mutex_lock(&mut); 
     while (state == ti.state) 
     { 
      pthread_cond_wait(&cond, &mut); 
     } 
     pthread_mutex_unlock(&mut); 

     assert(line.length() == 1); 
     if (line[0] == '0') 
     { 
      pthread_mutex_lock(&mut); 
      state = ti.state; 
      cout << "Got 0, transferring, setting state to " << state << '\n'; 
      pthread_cond_signal(&cond); 
      pthread_mutex_unlock(&mut); 
     } 
     else 
     { 
      cout << "Read char: " << line << '\n'; 
     } 
     getline(infile, line); 
    } 

    pthread_mutex_lock(&mut); 
    state = ti.state; 
    cout << "Finishing thread, transferring, setting state to " << state << '\n'; 
    pthread_cond_signal(&cond); 
    pthread_mutex_unlock(&mut); 
} 

int main() 
{ 
    // Create thread 1                                                                                                     
    // Create thread 2                                                                                                     

    pthread_t thread_one_handle; 
    pthread_t thread_two_handle; 
    state = THREAD_ONE_READS; 
    int result; 

    ThreadInfo info1 = { THREAD_TWO_READS, FILE1 }; 
    result = pthread_create(&thread_one_handle, NULL, thread_func, &info1); 
    assert(result == 0); 

    ThreadInfo info2 = { THREAD_ONE_READS, FILE2 }; 
    result = pthread_create(&thread_two_handle, NULL, thread_func, &info2); 
    assert(result == 0); 

    result = pthread_join(thread_one_handle, NULL); 
    assert(result == 0); 
    result = pthread_join(thread_two_handle, NULL); 
    assert(result == 0); 
    cout << "main(): joined both worker threads, ending program.\n"; 

    return 0; 
} 
+0

@Selenaこれが役に立ちましたか?私はあなたが同じ質問を2回掲載したことを見て、他の同等の質問に対する回答を受け入れました;( –

+0

この質問は常にCのみでタグ付けされていましたが、C++ではタグ付けされていませんでした。私はまた、同じ話題(上のコメント)で同じOPの別の質問に注意しました。 –

+0

@JonathanLeffler:C++固有の機能の使用は非常に私の答えでは限られているので、old-skool Cは自明でなければならない。 –

関連する問題