1

関数の実行を時間制限するプログラムを開発しようとしています。以下のコードにIncは、多くの反復を行う (無限ループによってシミュレートされた) という名前の関数があります。各反復の最初の部分は非常に長く、その後に非常に高速な 2 番目の部分が続きます。

コードの最初の部分で実行を横取りしてもかまいませんが、2 番目の部分で書き込み操作を行っているときにアラームが鳴るのを避けたいと思います。

私の最初のアイデアは、「安全な地域」に入る前にアラームをオフにして、残り時間を節約することでした。退出後、保存した時間でアラームを設定します。これを実装する方法がわかりません。誰かが私を助けることができますか?別の方法も歓迎します。

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

pthread_t thread;
FILE *fout;

void *Inc(void *param){

    int i;
    long long int x = 0;

    fout = fopen("file.txt", "w");

    /* Large number of iterations */
    while(1){

        int k = 0;
        for(i=0; i<5000000; i++)
            k += (rand())%3;
        x += k;

        printf("%lld\n", x);
        /* Enter Safe Region */
        fprintf(fout, "%lld\n", x);
        /* Exit Safe Region */
    }   
}

void Finish(int param){
    pthread_cancel(thread);
    fclose(fout);
}

main (){

    pthread_attr_t attr;
    void *status;

    signal(SIGALRM, Finish);
    alarm(10);

    pthread_attr_init(&attr);
    pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);

    pthread_create(&thread, &attr, Inc, NULL);
    pthread_attr_destroy(&attr);
    pthread_join(thread, &status);

    printf("Program Finished\n");
}
4

1 に答える 1

3

明らかなことは、 を呼び出す前にロックを取得しpthread_cancel、「安全な領域」全体で同じロックを保持することです。

残念ながら、シグナル ハンドラでミューテックスやセマフォを待機することはできません。しかし、10 秒後に何かを行う方法はアラーム シグナルだけではありません。代わりに、メイン スレッドを 10 秒間スリープ状態にしてから、ウェイクアップし、ロックを取得し、ワーカー スレッドをキャンセルしてから参加することもできます。

もちろん、これはワーカー スレッドが 5 秒後に終了しても、メイン スレッドが 10 秒間スリープすることを意味します。そのため、スリープする代わりに、セマフォで 10 秒間待機するようにメイン スレッドを設定し、終了時にワーカー スレッドがポストします。

スリープと同様に、時間指定待機はシグナルによって早期に完了する可能性があるため、必ず EINTR で時間指定待機を再試行してください。重大なケースは、EINTR (もう一度待機)、成功 (ワーカー スレッドに参加 - セマフォを投稿したためキャンセルする必要はありません)、ETIMEDOUT (ロックを取得、キャンセル、参加)、および必要に応じてその他のエラーです。ただし、影響を受けるエラーは他にリストされsem_timedwaitていません。

別のアイデアは、「安全な領域」全体で SIGALRM をブロックすることです。これは、(a) シグナルを無効にして I/O を安全に行う方法を思い出せないこと、および (b) ワーカー スレッドがおそらく「同時に実行されている可能性がある」ことを除いて、より簡単です。 " シグナル ハンドラーを使用して (真に同時であるか、プリエンプションのために明らかにそうであるかのいずれか)、つまりシグナルを受け取ることができた場合、ワーカーはシグナルを無効にしてクリティカル領域に入り、シグナル ハンドラーはそれをキャンセルします。詳細を実際に覚えている人が誰も答えない場合は、シグナルのブロックに関する情報を次に示します。

于 2010-09-01T21:14:22.863 に答える