每一个线程都包括线程ID、一组寄存器值、栈、调度优先级、策略、信号屏蔽字、errno变量、线程私有数据。使用_POSIX_THREADS
来测试是否支持这个功能,使用_SC_THREADS
运行时肯定,都须要添加#include <pthread.h>
,对于pthread
库的函数成功返回0,错误返回错误编号。shell
使用thread_t
来标识一个线程,在不一样系统中实现不同,须要使用函数来比较。bash
int pthread_equal(pthread_t t1, pthread_t t2);
// 相等返回非0值
复制代码
获取自身线程pthread_t
数据结构
pthread_t pthread_self(void);
复制代码
经过线程pthread_t
来分配任务如图所示:函数
int pthread_create(pthread_t *restrict tidp,
cosnt pthread_attr_t *restrict attr,
void *(start_rtn)(void *), void *restrict arg);
// 失败时返回错误码
复制代码
attr用于定制线程的属性,为NULL
时是默认属性。新建立的线程从start_rtn
开始运行,将参数放到结构体中,经过void *restrict arg
传递。测试
#include "../include/apue.h"
#include <pthread.h>
pthread_t ntid;
void printids(const char *s) {
pid_t pid;
pthread_t tid;
pid = getpid();
tid = pthread_self();
printf("%s pid:%lu, tid:%lu\n", s, pid, tid);
}
void *thread_func(void * arg) {
printids("new~~~");
sleep(1);
return (void*)0;
}
int main(void) {
int err;
err = pthread_create(&ntid, NULL, thread_func, NULL); // 第二个是pthread线程参数, 第四个是函数参数
if (err != 0) {
err_exit(err, "can't create thread");
}
printids("main~~~");
// sleep(1);
exit(0);
}
复制代码
若是新线程睡眠1s,而后主线程退出就不会输出新线程了。若是主线程睡眠1s,则两个线程的进程号相同。在Linux里输出以下:ui
main~~~ pid:8081, tid:139790962181888
new~~~ pid:8081, tid:139790953903872
复制代码
在任意线程中调用exit、_Exit、_exit
都会将整个进程终止。线程终止的方法有三种:spa
pthread_exit
void pthread_exit(void * rval_ptr);
// rval_ptr指向返回值
复制代码
也就是把须要返回的状态传进去,用于和等待的线程通讯。可使用pthread_join
来等待指定线程完成线程
#include <pthread.h>
int pthread_join(pthread_t thread, void **rval_ptr);
复制代码
若是被等待的线程返回,rval_ptr
包含返回码,若是线程被取消,rval_ptr
指定内存单元设置为PTHREAD_CANCELED
。rest
#include "../include/apue.h"
#include <pthread.h>
void printids(const char *s) {
pid_t pid;
pthread_t tid;
pid = getpid();
tid = pthread_self();
printf("%s pid:%lu, tid:%lu\n", s, (long unsigned)pid, (long unsigned)tid);
}
void *return_thread(void * arg) {
printids("thread returning~~~");
return (void*)0;
}
void *exit_thread(void * arg) {
printids("thread exiting~~~");
pthread_exit((void*) 2); // 参数能够返回结构体,可是这个结构体必须返回后还能使用(不是在栈上分配)
}
int main(void) {
int err;
pthread_t tid1, tid2;
void * rVal;
err = pthread_create(&tid1, NULL, return_thread, NULL); // 第二个是pthread线程参数, 第四个是函数参数
if (err != 0) {
err_exit(err, "can't create thread");
}
err = pthread_create(&tid2, NULL, exit_thread, NULL); // 第二个是pthread线程参数, 第四个是函数参数
if (err != 0) {
err_exit(err, "can't create thread");
}
pthread_join(tid1, &rVal);
printf("return_thread return:%ld\n", (long)rVal);
pthread_join(tid2, &rVal);
printf("exit_thread return:%ld\n", (long)rVal);
printids("main~~~");
// sleep(1);
exit(0);
}
复制代码
程序输出结果以下:code
thread returning~~~ pid:5832, tid:139867556669184
thread exiting~~~ pid:5832, tid:139867548276480
return_thread return:0
exit_thread return:2
main~~~ pid:5832, tid:139867564947200
复制代码
int pthread_cancel(pthread_t tid);
复制代码
pthrea_cancel不等待线程终止,而是提出请求。
线程安排本身的退出函数,多个清理函数会注册到栈中,按找栈里顺序执行。
void pthread_cleanup_push(void (*rtn)(void *), void *arg);
void pthread_cleanup_pop(int execute);
复制代码
触发时机:
pthread_exit
时pthread_cleanup_pop
,使用pthread_cleanup_pop(0)
不会调用清理函数,只是删除清理函数。输出:
thread 1 start up
thread 2 start up
thread1 return:1
clean up in thread2 second handler
clean up in thread2 first handler
thread2 return:2
main~~~ pid:10122, tid:140514088802048
复制代码
只有第二个线程的清理函数被调用,这是由于系统正常终止是不会调用清理函数,即return
结束
进程原语 | 线程原语 | 描述 |
---|---|---|
fork | pthread_create | 建立新的控制流 |
exit | pthread_exit | 从现有控制流退出 |
waitpid | pthread_join | 从控制流中获得退出状态 |
atexit | pthread_cleanup_push | 注册在退出时调用的函数 |
getpid | pthread_self | 获取控制流的ID |
abort | pthread_cancel | 请求控制流的非正常退出 |
int pthread_detach(pthread_t tid);
复制代码
当线程B在线程A的读写间隔中读取数据就会出现不一致的值:
在存储操做须要多个总线周期时:
互斥变量本质是一把锁。对互斥量加锁后,任何试图再次对互斥量加锁的线程都会被阻塞。释放互斥量后,其余阻塞的线程变为可运行状态,第一个变为可运行状态的线程对互斥量加锁,其余变量依然变为阻塞。
互斥变量使用pthrea_mutex_t
数据表示,使用前必须初始化,能够设置为pthread_mutex_t t = PTHREAD_MUTEX_INITIALIZER;
用于静态初始化互斥量。
int pthread_mutex_init(pthread_mutex_t *restrict mutex, const pthread_mutexattr_t *restrict attr);// 使用函数初始化
int pthread_mutex_destroy(pthread_mutex_t *mutex); // 使用malloc动态生成的,须要desotry函数销毁
复制代码
加锁与解锁操做:
int pthread_mutex_lock(pthread_muex_t *mutex);
int pthread_mutex_trylock(pthread_muex_t *mutex); // 线程不但愿被阻塞,就使用trylock,成功返回0, 失败返回EBUSY
int pthread_mutex_unlock(pthread_muex_t *mutex);
复制代码
#include "../include/apue.h"
#include <pthread.h>
struct foo {
int f_count;
pthread_mutex_t f_lock;
int f_id;
};
struct foo * foo_alloc(int id) {
struct foo *fp;
if ((fp=malloc(sizeof(struct foo))) != NULL) {
fp->f_count = 1;
fp->f_id = id;
if (pthread_mutex_init(&fp->f_lock, NULL) != 0) {
free(fp);
return (NULL);
}
}
return fp;
}
void foo_hold(struct foo *fp) {
pthread_mutex_lock(&fp->f_lock);
fp->f_count++;
pthread_mutex_unlock(&fp->f_lock);
}
void foo_release(struct foo *fp) {
pthread_mutex_lock(&fp->f_lock);
if (--fp->f_count == 0) {
pthread_mutex_unlock(&fp->f_lock);
pthread_mutex_destroy(&fp->f_lock);
free(fp);
} else {
printf("id:%d count:%d\n", fp->f_id, fp->f_count);
pthread_mutex_unlock(&fp->f_lock);
}
}
void * thread(void* arg) {
struct foo* f = (struct foo*)arg;
foo_hold(f);
}
int main(int argc, char const *argv[]) {
pthread_t tid;
struct foo* f = foo_alloc(12);
pthread_create(&tid, NULL, thread, (void*)f);
pthread_create(&tid, NULL, thread, (void*)f);
pthread_join(tid, NULL);
foo_release(f);
foo_release(f);
return 0;
}
复制代码
能够看到结果中第一次释放foo时count为2,每次运行都是。
id:12 count:2
id:12 count:1
复制代码
当线程对同一互斥量加锁两次时就会死锁。经过仔细控制互斥量加锁顺序来避免死锁发生。另外一种方法: 若是已经占有某些锁,则使用pthread_mutex_trylock
,若是成功则继续,若是失败则释放锁,作好清理工做,等待一段时间后再试试。
当程序师徒获取一个已加锁的互斥量时,pthread_mutex_timedlock
互斥量原语绑定线程阻塞的时间。到达超时时间后pthread_mutex_timedlock
不会对互斥量加锁而是返回错误码ETIMEDOUT
#include <pthread.h>
#include <time.h>
int pthread_mutex_timedlock(pthread_mutex_t *restrict mutex, const struct timespec *restrict tsptr);
复制代码
#include <time.h>
中的timespec
使用秒和纳秒描述时间。
struct timespec {
__time_t tv_sec; /* Seconds. */
__syscall_slong_t tv_nsec; /* Nanoseconds. */
};
复制代码
#include <time.h>
中的tm
表示年月日星期等。
struct tm {
int tm_sec; /* Seconds. [0-60] (1 leap second) */
int tm_min; /* Minutes. [0-59] */
int tm_hour; /* Hours. [0-23] */
int tm_mday; /* Day. [1-31] */
int tm_mon; /* Month. [0-11] */
int tm_year; /* Year - 1900. */
int tm_wday; /* Day of week. [0-6] */
int tm_yday; /* Days in year.[0-365] */
int tm_isdst; /* DST. [-1/0/1]*/
}
复制代码
#include "../include/apue.h"
#include <pthread.h>
void printTime() {
char buf[64];
struct timespec tout;
struct tm* tmp;
clock_gettime(CLOCK_REALTIME, &tout);
tmp = localtime(&tout.tv_sec);
strftime(buf, sizeof(buf), "%r", tmp);
printf("current time is %s\n", buf);
}
int main(int argc, char const *argv[])
{
int err;
struct timespec tout;
struct tm *tmp;
pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;
printTime();
pthread_mutex_lock(&lock);
printf("mutex is lock\n");
clock_gettime(CLOCK_REALTIME, &tout);
tout.tv_sec += 10;
pthread_mutex_timedlock(&lock, &tout);
printTime();
return 0;
}
复制代码
由于已经得到了lock的锁,再次用pthread_mutex_lock
锁住会致使死锁,使用pthread_mutex_timedlock
只会阻塞指定时间。 输出结果以下,只阻塞了10秒
current time is 06:55:12 PM
mutex is lock
current time is 06:55:22 PM
复制代码
读写锁有三种状态:读模式下加锁状态、写模式下加锁状态、不加锁状态。一次只有一个线程占有写模式的读写锁,可是多个线程能够同时占有读模式的读写锁
读写锁必须在使用前初始化、使用后释放内存前销毁
int pthread_rwlock_init(pthread_rwlock_t *restrict rwlock, const pthread_rwlockattr_t *restrict attr);
// 初始化函数,若是读写锁默认属性则传入null给attr
int pthread_rwlock_destroy(pthread_rwlock_t * rwlock);
// 在free前调用
pthread_rwlock_t lock = PTHREAD_RWLOCK_INITIALIZER; // 静态初始化(Signle UNIX Specification)
int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock); //读加锁
int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock); // 写加锁
int pthread_rwlock_unlock(pthread_rwlock_t *rwlock); // 解锁
int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock); // 条件版本(Signle UNIX Specification)
int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock); // 条件版本(Signle UNIX Specification)
// 得到锁是返回0,不然返回错误EBUSY
复制代码
做业队列,使用单个读写锁保护队列,插入、删除会尝试给队列加写锁,查找队列时会给队列加读锁。
#include "../include/apue.h"
#include <pthread.h>
struct job {
struct job *j_next;
struct job *j_prev;
pthread_t j_id; // 哪个线程处理这个任务
};
struct queue {
struct job *q_head;
struct job *q_tail;
pthread_rwlock_t q_lock;
};
int queue_init(struct queue *qp) {
int err;
qp->q_head = NULL;
qp->q_tail = NULL;
err = pthread_rwlock_init(&qp->q_lock);
if (err != 0) {
return err;
}
return 0;
}
// 从队列后面插入job
void job_insert_tail(struct queue *qp, struct job *jp) {
pthread_rwlock_wrlock(&qp->q_lock);
jp->j_next = NULL;
jp->j_prev = qp->q_tail;
if (qp->q_tail != NULL) {
qp->q_tail->j_next = jp;
} else {
qp->q_head = jp; // 链表为空
}
qp->q_tail = jp;
pthread_rwlock_unlock(&qp->q_lock);
}
// 从队列前面插入job
void job_insert_front(struct queue *qp, struct job *jp) {
pthread_rwlock_wrlock(&qp->q_lock);
jp->j_next = qp->q_head;
jp->j_prev = NULL;
if (qp->q_head != NULL) {
qp->q_head->j_prev = jp;
} else {
qp->q_tail = jp; // 链表为空
}
qp->q_head = jp;
pthread_rwlock_unlock(&qp->q_lock);
}
// 从队列中删除job
void job_remove(struct queue *qp, struct job *jp) {
pthread_rwlock_wrlock(&qp->q_lock);
if (jp == qp->q_head) {
qp->q_head = jp->j_next;
if (jp == qp->q_tail) {
qp->q_tail = NULL;
} else {
jp->j_next->j_prev = jp->j_prev;
}
} else if (jp == qp->q_tail) {
jp->j_prev->j_next = jp->j_next;
qp->q_tail = jp->j_prev;
} else {
jp->j_prev->j_next = jp->j_next;
jp->j_next->j_prev = jp->j_prev;
}
pthread_rwlock_unlock(&qp->q_lock);
}
// 经过线程id查找某个任务
struct job* job_find(struct queue *qp, pthread_t id) {
struct job* jp;
if (pthread_rwlock_rdlock(&qp->q_lock) != 0) {
return NULL;
}
for (jp = qp->q_head; jp != NULL; jp = jp->j_next) {
if (pthread_equal(jp->j_id, id)) {
printf("Find you!\n");
break;
}
}
pthread_rwlock_unlock(&qp->q_lock);
return jp;
}
复制代码
每次只能有一个写锁,因此对于job结构体不须要对它加锁。
int pthread_rwlock_timedrdlock(pthread_rwlock_t *restrict rwlock, const struct timespec *restrict tsptr);
int pthread_rwlock_timedwrlock(pthread_rwlock_t *restrict rwlock, const struct timespec *restrict tsptr);
复制代码
超时会返回ETIMEDOUT
条件变量有互斥量保护,线程在改变条件状态以前首先锁住互斥量,锁住后计算条件。
// 1.动态初始化
int pthread_cond_init(pthread_cond_t *restrict cond, const pthread_condattr_t *restrict attr);`
// 2.静态初始化
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;`
// 销毁方式:
int pthread_cond_destroy(pthread_cond_t *cond);
// 等待条件变量变为真
int pthread_cond_wait(pthread_cond_t *restrict cond, pthread_mutex_t *restrict mutex);
int pthread_cond_timedwait(pthread_cond_t *restrict cond, pthread_mutex_t *restrict mutex, const struct timespec *restrict tsptr);
复制代码
使用互斥量对条件进行保护,调用者把锁住的互斥量传给函数,函数而后自动把调用线程放到等待条件的线程列表中,而后对互斥量解锁。
等待时间使用的是绝对时间,不是以前的时间差而是将将来时间传入,使用clock_gettime
得到timespec
表示的当前时间,也能够经过gettimeofday
得到timeval
结构表示的当前时间,再转换为timespec
,函数以下所示:
#include <sys/time.h>
#include <stdlib.h>
// 以分钟做为时间间隔
void maketimeout(struct timespec *tsp, long minutes) {
struct timeval now;
gettimeofday(&now, NULL);
tsp->tv_sec = now.tv_sec;
tsp->tv_nsec = now.tv_usec * 1000;
tsp->tv_sec += minutes * 60;
}
复制代码
从pthread_cond_wait
或pthread_cond_timedwait
调用成功返回时,线程须要从新计算条件。pthread_cond_signal
能唤醒至少一个睡眠线程,pthread_cond_broadcast
能唤醒因此等待而睡眠的线程。
int pthread_cond_signal(pthread_cond_t *cnd);
int pthread_cond_broadcast(pthread_cond_t *cond);
复制代码
必须在改变条件状态以后再给线程发送信号
#include "../include/apue.h"
#include <pthread.h>
#define WORKS_NUMS 10
struct msg {
struct msg *m_next;
char message[64];
};
struct msg *workq;
pthread_cond_t qready = PTHREAD_COND_INITIALIZER;
pthread_mutex_t qlock = PTHREAD_MUTEX_INITIALIZER;
void process_msg(char* name) {
struct msg *mp;
pthread_mutex_lock(&qlock);
while (workq == NULL)
pthread_cond_wait(&qready, &qlock);
// 将锁住的互斥量传入,pthread_cond_wait会将线程放入等待队列中,而后解锁qlock,
// 此时阻塞在pthread_cond_wait,当被pthread_cond_signal或pthread_cond_broadcast
// 唤醒后此时qlock会锁住,若是workq==NULL,就会再次等待
mp = workq;
workq = workq->m_next;
pthread_mutex_unlock(&qlock);
printf("[process-%s]:%s\n", name, mp->message);
}
void enqueue_msg(struct msg *mp) {
pthread_mutex_lock(&qlock); // 条件workq是由互斥量mutex保护
mp->m_next = workq; // 修改条件这个操做须要保持一致
workq = mp;
pthread_mutex_unlock(&qlock);
pthread_cond_signal(&qready); // 修改条件后才发送信号
}
void *worker(void *arg) {
printf("worker %s create!\n", (char *)arg);
process_msg((char*)arg);
}
void *sender(void *arg) {
printf("sender create!\n");
for (int i = 0; i < WORKS_NUMS; i++)
{
struct msg *m = (struct msg *)malloc(sizeof(struct msg));
sprintf(m->message, "msg-%d", i);
enqueue_msg(m);
sleep(1); // 发送后等待1s,让worker处于等待状态
}
return 0;
}
int main(int argc, char const *argv[]) {
pthread_t send;
pthread_t works[WORKS_NUMS];
pthread_create(&send, NULL, sender, NULL);
// sleep(1);
for(int i= 0; i < WORKS_NUMS; i++) {
char *name = (char*)malloc(10);
sprintf(name, "%d", i);
pthread_create(&works[i], NULL, worker, (void *)name);
}
pthread_join(send, NULL);
for(int i= 0; i < WORKS_NUMS; i++) {
pthread_join(works[i], NULL);
}
return 0;
}
复制代码
输出:
sender create!
worker 0 create!
[worker-0]:msg-0
worker 3 create!
worker 2 create!
worker 4 create!
worker 1 create!
worker 5 create!
worker 6 create!
worker 7 create!
worker 8 create!
worker 9 create!
[worker-3]:msg-1
[worker-2]:msg-2
[worker-4]:msg-3
[worker-1]:msg-4
[worker-5]:msg-5
[worker-6]:msg-6
[worker-7]:msg-7
[worker-8]:msg-8
[worker-9]:msg-9
复制代码
由于pthread_cond_wait
是在while循环中,若是不知足条件会继续进入循环。