Linux应用开发基础知识——多线程编程(十)

简介: Linux应用开发基础知识——多线程编程(十)

线程的使用

多进程的原理逻辑,需要俩个进行进行通信,效率低

多线程的原理逻辑, 俩个线程之间特别方便的进行传输通信,效率高

注意:调度是以线程为单位的,资源分配是以进程为单位的。

1.线程概念

       所谓线程,就是操作系统所能调度的最小单位。普通的进程,只有一个线程在执行对应的逻辑。我们可以通过多线程编程,使一个进程可以去执行多个不同的任务。相比多进程编程而言,线程享有共享资源,即在进程中出现的全局变量, 每个线程都可以去访问它,与进程共享“4G”内存空间,使得系统资源消耗减少。我们主要学习研究 Linux 下 POSIX 线程

2.线程的标识 pthread_t

       对于进程而言,每一个进程都有一个唯一对应的 PID 号来表示该进程,而对于线程而言,也有一个“类似于进程的 PID 号”,名为 tid,其本质是一个pthread_t 类型的变量。线程号与进程号是表示线程和进程的唯一标识,但是对于线程号而言,其仅仅在其所属的进程上下文中才有意义。

3.线程的创建

int pthread_create(pthread_t *thread, const pthread_attr_t *attr,
                          void *(*start_routine) (void *), void *arg);

第一个参数pthread_t *thread pthread_t 指针,用来保存新建线程的线程号

第二个参数const pthread_attr_t *attr表示了线程的属性,一般传入NULL 表示默认属性; 第三个参数void *(*start_routine) (void *)是一个函数指针,就是线程执行的函数。这个函数返回值为 void*, 形参为 void*。

第四个参数void *arg则表示为向线程处理函数传入的参数,若不传入,可用 NULL 填充

  1 #include <pthread.h>
  2 #include <stdio.h>
  3 #include <unistd.h>
  4
  5
  6 static void *my_thread_func (void *data)
  7 {
  8     while (1)
  9     {
 10         sleep(1);
 11     }
 12 }
 13
 14
 15 int main(int argc, char **argv)
 16 {
 17     pthread_t tid;
 18     int ret;
 19
 20     /* 1. 创建"接收线程" */
 21     ret = pthread_create(&tid, NULL, my_thread_func, NULL);
 22     if (ret)
 23     {
 24         printf("pthread_create err!\n");
 25         return -1;
 26     }
 27
 28
 29     /* 2. 主线程读取标准输入, 发给"接收线程" */
 30     while (1)
 31     {
 32         sleep(1);
 33     }
 34     return 0;
 35 }

第6~12行: 定义一个结构体函数

第17行: 定义线程ID

第18行: 接收第21行pthread_create返回值

book@100ask:~/source/13_thread/02_视频配套源码$ gcc -o pthread1 pthread1.c -lpthread

4.线程的接收

char *fgets(char *s, int size, FILE *stream);    //用于读取字符串

int sem_init(sem_t *sem, int pshared, unsigned int value);
// 初始化
#include <semaphore.h>
int sem_init(sem_t *sem,int pshared,unsigned int value);
等待/释放:
#include <pthread.h>
int sem_wait(sem_t *sem);
int sem_post(sem_t *sem);
  1 #include <pthread.h>
  2 #include <stdio.h>
  3 #include <unistd.h>
  4 #include <semaphore.h>
  5
  6 static char g_buf[1000];
  7 static sem_t g_sem;
  8 static void *my_thread_func (void *data)
  9 {
 10     while (1)
 11     {
 12         //sleep(1);
 13         /* 等待通知 */
 14         //while (g_hasData == 0);
 15         sem_wait(&g_sem);
 16
 17         /* 打印 */
 18         printf("recv: %s\n", g_buf);
 19     }
 20
 21     return NULL;
 22 }
 23
 24
 25 int main(int argc, char **argv)
 26 {
 27     pthread_t tid;
 28     int ret;
 29
 30     sem_init(&g_sem, 0, 0);
 31
 32     /* 1. 创建"接收线程" */
 33     ret = pthread_create(&tid, NULL, my_thread_func, NULL);
 34     if (ret)
 35     {
 36         printf("pthread_create err!\n");
 37         return -1;
 38     }
 39
 40
 41     /* 2. 主线程读取标准输入, 发给"接收线程" */
 42     while (1)
 43     {
 44         fgets(g_buf, 1000, stdin);
 45
 46         /* 通知接收线程 */
 47         sem_post(&g_sem);
 48     }
 49     return 0;
 50 }

第6行: 设置全局变量

第7行: 设置标记位

             使用信号量进行同步操作

book@100ask:~/source/13_thread/02_视频配套源码$ gcc -o pthread3 pthread3.c -lpthread

5.互斥访问

       打印信息的过程中可能出现前半端打印新数据,后半段打印旧数据的情况,所以为了防止这种现象的发生,我们选择加入互斥量进行优化。

  1 #include <pthread.h>
  2 #include <stdio.h>
  3 #include <unistd.h>
  4 #include <semaphore.h>
  5 #include <string.h>
  6
  7 static char g_buf[1000];
  8 static sem_t g_sem;
  9 static pthread_mutex_t g_tMutex  = PTHREAD_MUTEX_INITIALIZER;
 10
 11 static void *my_thread_func (void *data)
 12 {
 13     while (1)
 14     {
 15         //sleep(1);
 16         /* 等待通知 */
 17         //while (g_hasData == 0);
 18         sem_wait(&g_sem);
 19
 20         /* 打印 */
 21         pthread_mutex_lock(&g_tMutex);
 22         printf("recv: %s\n", g_buf);
 23         pthread_mutex_unlock(&g_tMutex);
 24     }
 25
 26     return NULL;
 27 }
 28
 29
 30 int main(int argc, char **argv)
 31 {
 32     pthread_t tid;
 33     int ret;
 34     char buf[1000];
 35
 36     sem_init(&g_sem, 0, 0);
 37
 38     /* 1. 创建"接收线程" */
 39     ret = pthread_create(&tid, NULL, my_thread_func, NULL);
 40     if (ret)
 41     {
 42         printf("pthread_create err!\n");
 43         return -1;
 44     }
 45
 46
 47     /* 2. 主线程读取标准输入, 发给"接收线程" */
 48     while (1)
 49     {
 50         fgets(buf, 1000, stdin);
 51         pthread_mutex_lock(&g_tMutex);
 52         memcpy(g_buf, buf, 1000);
 53         pthread_mutex_unlock(&g_tMutex);
 54
 55         /* 通知接收线程 */
 56         sem_post(&g_sem);
 57     }
 58     return 0;
 59 }

第9行: 定义一个互斥量

第21~23和51~53行分别设置子线程和父线程的互斥锁

book@100ask:~/source/13_thread/02_视频配套源码$ gcc -o pthread4 pthread4.c -lpthread

6.同步操作

 
  1 #include <pthread.h>
  2 #include <stdio.h>
  3 #include <unistd.h>
  4 #include <semaphore.h>
  5 #include <string.h>
  6
  7 static char g_buf[1000];
  8 static pthread_mutex_t g_tMutex  = PTHREAD_MUTEX_INITIALIZER;
  9 static pthread_cond_t  g_tConVar = PTHREAD_COND_INITIALIZER;
 10
 11 static void *my_thread_func (void *data)
 12 {
 13     while (1)
 14     {
 15         //sleep(1);
 16         /* 等待通知 */
 17         //while (g_hasData == 0);
 18         pthread_mutex_lock(&g_tMutex);
 19         pthread_cond_wait(&g_tConVar, &g_tMutex);
 20
 21         /* 打印 */
 22         printf("recv: %s\n", g_buf);
 23         pthread_mutex_unlock(&g_tMutex);
 24     }
 25
 26     return NULL;
 27 }
 28
 29
 30 int main(int argc, char **argv)
 31 {
 32     pthread_t tid;
 33     int ret;
 34     char buf[1000];
 35
 36     /* 1. 创建"接收线程" */
 37     ret = pthread_create(&tid, NULL, my_thread_func, NULL);
 38     if (ret)
 39     {
 40         printf("pthread_create err!\n");
 41         return -1;
 42     }
 43
 44
 45     /* 2. 主线程读取标准输入, 发给"接收线程" */
 46     while (1)
 47     {
 48         fgets(buf, 1000, stdin);
 49         pthread_mutex_lock(&g_tMutex);
 50         memcpy(g_buf, buf, 1000);
 51         pthread_cond_signal(&g_tConVar); /* 通知接收线程 */
 52         pthread_mutex_unlock(&g_tMutex);
 53     }
 54     return 0;
 55 }

第18和19行:线程A:等待条件成立

第52行:操作临界资源

第51行:线程B:唤醒等待g_tConVar的线程


目录
相关文章
|
2月前
|
Java 开发者
Java多线程编程中的常见误区与最佳实践####
本文深入剖析了Java多线程编程中开发者常遇到的几个典型误区,如对`start()`与`run()`方法的混淆使用、忽视线程安全问题、错误处理未同步的共享变量等,并针对这些问题提出了具体的解决方案和最佳实践。通过实例代码对比,直观展示了正确与错误的实现方式,旨在帮助读者构建更加健壮、高效的多线程应用程序。 ####
|
2月前
|
安全 Java UED
深入浅出Java多线程编程
【10月更文挑战第40天】在Java的世界中,多线程是提升应用性能和响应能力的关键。本文将通过浅显易懂的方式介绍Java中的多线程编程,从基础概念到高级特性,再到实际应用案例,带你一步步深入了解如何在Java中高效地使用多线程。文章不仅涵盖了理论知识,还提供了实用的代码示例,帮助你在实际开发中更好地应用多线程技术。
57 5
|
18天前
|
存储 监控 Linux
嵌入式Linux系统编程 — 5.3 times、clock函数获取进程时间
在嵌入式Linux系统编程中,`times`和 `clock`函数是获取进程时间的两个重要工具。`times`函数提供了更详细的进程和子进程时间信息,而 `clock`函数则提供了更简单的处理器时间获取方法。根据具体需求选择合适的函数,可以更有效地进行性能分析和资源管理。通过本文的介绍,希望能帮助您更好地理解和使用这两个函数,提高嵌入式系统编程的效率和效果。
83 13
|
13天前
|
存储 安全 Java
Java多线程编程秘籍:各种方案一网打尽,不要错过!
Java 中实现多线程的方式主要有四种:继承 Thread 类、实现 Runnable 接口、实现 Callable 接口和使用线程池。每种方式各有优缺点,适用于不同的场景。继承 Thread 类最简单,实现 Runnable 接口更灵活,Callable 接口支持返回结果,线程池则便于管理和复用线程。实际应用中可根据需求选择合适的方式。此外,还介绍了多线程相关的常见面试问题及答案,涵盖线程概念、线程安全、线程池等知识点。
93 2
|
2月前
|
设计模式 Java 开发者
Java多线程编程的陷阱与解决方案####
本文深入探讨了Java多线程编程中常见的问题及其解决策略。通过分析竞态条件、死锁、活锁等典型场景,并结合代码示例和实用技巧,帮助开发者有效避免这些陷阱,提升并发程序的稳定性和性能。 ####
|
1月前
|
缓存 Java 调度
多线程编程核心:上下文切换深度解析
在现代计算机系统中,多线程编程已成为提高程序性能和响应速度的关键技术。然而,多线程编程中一个不可避免的概念就是上下文切换(Context Switching)。本文将深入探讨上下文切换的概念、原因、影响以及优化策略,帮助你在工作和学习中深入理解这一技术干货。
47 10
|
2月前
|
缓存 Java 开发者
Java多线程编程的陷阱与最佳实践####
本文深入探讨了Java多线程编程中常见的陷阱,如竞态条件、死锁和内存一致性错误,并提供了实用的避免策略。通过分析典型错误案例,本文旨在帮助开发者更好地理解和掌握多线程环境下的编程技巧,从而提升并发程序的稳定性和性能。 ####
|
29天前
|
安全 算法 Java
Java多线程编程中的陷阱与最佳实践####
本文探讨了Java多线程编程中常见的陷阱,并介绍了如何通过最佳实践来避免这些问题。我们将从基础概念入手,逐步深入到具体的代码示例,帮助开发者更好地理解和应用多线程技术。无论是初学者还是有经验的开发者,都能从中获得有价值的见解和建议。 ####
|
29天前
|
Java 调度
Java中的多线程编程与并发控制
本文深入探讨了Java编程语言中多线程编程的基础知识和并发控制机制。文章首先介绍了多线程的基本概念,包括线程的定义、生命周期以及在Java中创建和管理线程的方法。接着,详细讲解了Java提供的同步机制,如synchronized关键字、wait()和notify()方法等,以及如何通过这些机制实现线程间的协调与通信。最后,本文还讨论了一些常见的并发问题,例如死锁、竞态条件等,并提供了相应的解决策略。
50 3
|
1月前
|
算法 调度 开发者
多线程编程核心:上下文切换深度解析
在多线程编程中,上下文切换是一个至关重要的概念,它直接影响到程序的性能和响应速度。本文将深入探讨上下文切换的含义、原因、影响以及如何优化,帮助你在工作和学习中更好地理解和应用多线程技术。
42 4