原文地址:http://blog.csdn.net/liang890319/article/details/8393120

 

进程简单的说就是把一段代码复制成多份,并让他们同时执行。进程间通信是为了让他们有序的运行

线程简单的说就是让多个函数同时执行,线程间通信是为了让他们有序的运行

编译线程程序时会警告说线程函数找不到

pthread 库不是 Linux 系统默认的库,连接时需要使用静态库 libpthread.a,所以在使用pthread_create()创建线程,以及调用 pthread_atfork()函数建立fork处理程序时,需要链接该库。

问题解决:
    在编译中要加 -lpthread参数
    gcc thread.c -o thread -lpthread
    thread.c为你些的源文件,不要忘了加上头文件#include<pthread.h>

http://blog.csdn.net/llqkk/article/details/2854558

实例1:创建两个线程,同时执行同一个函数

/* ex7-1.c */
#include <stddef.h>
#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <string.h>

void print_msg(char *ptr);

int main()
{
pthread_t thread1, thread2;
int i,j;
char *msg1="do sth1\n";
char *msg2="do sth2\n";
pthread_create(&thread1,NULL, (void *)(&print_msg), (void *)msg1);
pthread_create(&thread2,NULL, (void *)(&print_msg), (void *)msg2);
sleep(1);
return 0;
}

void  print_msg(char *ptr)
{
int retval;
int id=pthread_self();
printf("Thread ID: %x\n",id);
  printf("%s",ptr);
pthread_exit(&retval);
}

执行gcc ex7-1.c -lpthread

./a.out

实例2  创建多个线程执行不同函数

代码来自 http://www.cnblogs.com/BiffoLee/archive/2011/11/18/2254540.html

#include <pthread.h>
#include <stdio.h>
#include <sys/time.h>
#include <string.h>
#define MAX 10
 
pthread_t thread[2];
pthread_mutex_t mut;
int number=0, i;
 
void *thread1()
{
    printf ("thread1 : I'm thread 1\n");
    for (i = 0; i < MAX; i++)
        {
            printf("thread1 : number = %d\n",number);
            pthread_mutex_lock(&mut);
            number++;
            pthread_mutex_unlock(&mut);
            sleep(2);
        }
    printf("thread1 :主函数在等我完成任务吗?\n");
 
    pthread_exit(NULL);
 
}
 
void *thread2()
{
    printf("thread2 : I'm thread 2\n");
    for (i = 0; i < MAX; i++)
        {
            printf("thread2 : number = %d\n",number);
            pthread_mutex_lock(&mut);
            number++;
            pthread_mutex_unlock(&mut);
            sleep(3);
        }
    printf("thread2 :主函数在等我完成任务吗?\n");
    pthread_exit(NULL);
}
 
 
void thread_create(void)
{
    int temp;
    memset(&thread, 0, sizeof(thread)); //comment1
    //创建线程
    if((temp = pthread_create(&thread[0], NULL, thread1, NULL)) != 0) //comment2
        printf("线程1创建失败!\n");
    else
        printf("线程1被创建\n");
    if((temp = pthread_create(&thread[1], NULL, thread2, NULL)) != 0) //comment3
        printf("线程2创建失败");
    else
        printf("线程2被创建\n");
}
 
void thread_wait(void)
{
    //等待线程结束
    if(thread[0] !=0) { //comment4
        pthread_join(thread[0],NULL);
        printf("线程1已经结束\n");
    }
    if(thread[1] !=0) { //comment5
        pthread_join(thread[1],NULL);
        printf("线程2已经结束\n");
    }
}
 
int main()
{
    //用默认属性初始化互斥锁
    pthread_mutex_init(&mut,NULL);
    printf("我是主函数哦,我正在创建线程,呵呵\n");
    thread_create();
    printf("我是主函数哦,我正在等待线程完成任务阿,呵呵\n");
    thread_wait();
    return 0;
}

编译 :

gcc -lpthread -o thread_example lp.c

实例3:信号量控制线程运行顺序

/* thread_sem.c */

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

#define THREAD_NUMBER 3
#define REPEAT_NUMBER 3
#define DELAY_TIME_LEVELS 10.0

sem_t sem[THREAD_NUMBER];

void * thrd_func(void *arg)
{
int thrd_num = (int)arg;
int delay_time = 0;
int count = 0;

sem_wait(&sem[thrd_num]);

printf("Thread %d is starting\n", thrd_num);

for (count = 0; count < REPEAT_NUMBER; count++)
{

delay_time = (int)(rand() * DELAY_TIME_LEVELS/(RAND_MAX)) + 1;
sleep(delay_time);
printf("\tThread %d: job %d delay = %d\n", thrd_num, count, delay_time);
}

printf("Thread %d finished\n", thrd_num);

pthread_exit(NULL);
}

int main(void)
{
pthread_t thread[THREAD_NUMBER];
int no = 0, res;
void * thrd_ret;

srand(time(NULL));

for (no = 0; no < THREAD_NUMBER; no++)
{
sem_init(&sem[no], 0, 0);
res = pthread_create(&thread[no], NULL, thrd_func, (void*)no);
if (res != 0)
{
printf("Create thread %d failed\n", no);
exit(res);
}
}

printf("Create treads success\n Waiting for threads to finish...\n");
sem_post(&sem[THREAD_NUMBER - 1]);
for (no = THREAD_NUMBER - 1; no >= 0; no--)
{
res = pthread_join(thread[no], &thrd_ret);
if (!res)
{
printf("Thread %d joined\n", no);
}
else
{
printf("Thread %d join failed\n", no);
}
sem_post(&sem[(no + THREAD_NUMBER - 1) % THREAD_NUMBER]);

}

for (no = 0; no < THREAD_NUMBER; no++)
{
sem_destroy(&sem[no]);
}

return 0;        
}

实例4:互斥锁的使用

在这个程序中,一个线程要往缓冲区写数据,另一个线程要读数据,每次只能让一个线程操作缓冲区

/*ex7-3.c*/

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

#define FALSE 0
#define TRUE 1

void readfun();
void writefun();

char buffer[256];
int buffer_has_item=0;
int retflag=FALSE,i=0;
pthread_mutex_t mutex;

int main()
{
void *retval;
pthread_t reader;
pthread_mutex_init(&mutex,NULL);
pthread_create(&reader,NULL,(void *)&readfun,NULL);
writefun();
pthread_join(reader,&retval);

}

void readfun()
{
while(1)
{
if(retflag)
return;
pthread_mutex_lock(&mutex);
if(buffer_has_item==1)
{
printf("%s",buffer);
buffer_has_item=0;
}
pthread_mutex_unlock(&mutex);
}
}
void writefun()
{
int i=0;
while(1)
{
if(i==10)
{
retflag=TRUE;
return;
}
pthread_mutex_lock(&mutex);
if(buffer_has_item==0)
{
sprintf(buffer,"This is %d\n",i++);
buffer_has_item=1;
}
pthread_mutex_unlock(&mutex);
}
}

实例5 条件变量

text

  1. /* ex7-4.c */
  2. #include <stdio.h>
  3. #include <pthread.h>
  4. #define BUFFER_SIZE 4
  5. #define OVER (-1)
  6. struct producers
  7. {
  8. int buffer[BUFFER_SIZE];
  9. pthread_mutex_t lock;
  10. int     readpos, writepos;
  11. pthread_cond_t  notempty;
  12. pthread_cond_t  notfull;
  13. };
  14. void init(struct producers *b)
  15. {
  16. pthread_mutex_init(&b->lock,NULL);
  17. pthread_cond_init(&b->notempty,NULL);
  18. pthread_cond_init(&b->notfull,NULL);
  19. b->readpos=0;
  20. b->writepos=0;
  21. }
  22. void put(struct producers *b, int data)
  23. {
  24. pthread_mutex_lock(&b->lock);
  25. while((b->writepos+1)%BUFFER_SIZE==b->readpos)
  26. {
  27. pthread_cond_wait(&b->notfull,&b->lock);
  28. }
  29. b->buffer[b->writepos]=data;
  30. b->writepos++;
  31. if(b->writepos>=BUFFER_SIZE) b->writepos=0;
  32. pthread_cond_signal(&b->notempty);
  33. pthread_mutex_unlock(&b->lock);
  34. }
  35. int get(struct producers *b)
  36. {
  37. int data;
  38. pthread_mutex_lock(&b->lock);
  39. while(b->writepos==b->readpos)
  40. {
  41. pthread_cond_wait(&b->notempty,&b->lock);
  42. }
  43. data=b->buffer[b->readpos];
  44. b->readpos++;
  45. if(b->readpos>=BUFFER_SIZE) b->readpos=0;
  46. pthread_cond_signal(&b->notfull);
  47. pthread_mutex_unlock(&b->lock);
  48. return data;
  49. }
  50. struct producers  buffer;
  51. void *producer(void *data)
  52. {
  53. int n;
  54. for(n=0;n<10;n++)
  55. {
  56. printf("Producer : %d-->\n",n);
  57. put(&buffer,n);
  58. }
  59. put(&buffer,OVER);
  60. return NULL;
  61. }
  62. void *consumer(void *data)
  63. {
  64. int d;
  65. while(1)
  66. {
  67. d=get(&buffer);
  68. if(d==OVER) break;
  69. printf("Consumer: --> %d\n",d);
  70. }
  71. return NULL;
  72. }
  73. int main()
  74. {
  75. pthread_t tha,thb;
  76. void *retval;
  77. init(&buffer);
  78. pthread_create(&tha,NULL,producer,0);
  79. pthread_create(&thb,NULL,consumer,0);
  80. pthread_join(tha,&retval);
  81. pthread_join(thb,&retval);
  82. return 0;
  83. }

linux c编程 多线程(初级)《转载》---赠人玫瑰,手有余香!的更多相关文章

  1. linux串口编程设置(转载)

    (转载)在嵌入式Linux中,串口是一个字设备,访问具体的串行端口的编程与读/写文件 的操作类似,只需打开相应的设备文件即可操作.串口编程特殊在于串 口通信时相关参数与属性的设置.嵌入式Linux的串 ...

  2. Linux网络编程--多线程实现echo服务器与客户端“一对多”功能,是网络编程的“Hello World!”

    在linux平台下,用多线程实现echo服务器与客户端“一对多”(即是一台服务器可以响应多个客户端的请求).本人写了个demo,和大家一起分享,有不足的地方,请多多指教,我是壮壮熊. 编译时,在后面加 ...

  3. Linux系统编程@多线程编程(二)

    线程的操作 线程标识 线程的ID表示数据类型:pthread_t (内核中的实现是unsigned long/unsigned int/指向pthread结构的指针(不可移植)几种类型) 1.对两个线 ...

  4. Linux系统编程@多线程编程(一)

    多线程编程 涉及操作系统原理概念 时间片 进程状态 上下文: 对进程来说,就是进程的执行环境,具体就是各个变量和数据,包括所有的寄存器变量.打开的文件.内存信息等. 进程的写时复制:由于一般 fork ...

  5. Linux系统编程@多线程与多进程GDB调试

    博客内容参考自 http://www.cnblogs.com/xuxm2007/archive/2011/04/01/2002162.html http://blog.csdn.net/pbymw8i ...

  6. Linux系统编程——多线程实现多任务

    概述 每一个进程都拥有自己的数据段.代码段和堆栈段.这就造成进程在进行创建.切换.撤销操作时,须要较大的系统开销. 为了降低系统开销,从进程中演化出了线程.为了让进程完毕一定的工作.进程必须至少包括一 ...

  7. 授人玫瑰 手留余香 --纪念python3.2.3官方文档翻译结束

    当你点击看到这篇文章的时候.你已经得到了祝福. 一个来自夜深人静的码农,在2014年5月19号的01:18分.默默为你献上祝福. 希望你.我和他,每个在IT行业中奋斗的人.能找到属于自己一片天空. 在 ...

  8. Linux系统编程——线程私有数据

    在多线程程序中.常常要用全局变量来实现多个函数间的数据共享.因为数据空间是共享的,因此全局变量也为全部线程共同拥有. 測试代码例如以下: #include <stdio.h> #inclu ...

  9. Linux网络编程入门 (转载)

    (一)Linux网络编程--网络知识介绍 Linux网络编程--网络知识介绍客户端和服务端         网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户 ...

随机推荐

  1. 将EC2里的实例导出到RAW文件并进行修改

    你可能有自己的instance在amazon云环境里面,或者是你想深度修改一下marketplace里面提供的那些系统又估计运行中的instance改动不方便 亚马逊作为云计算领域的大哥大,我不得不说 ...

  2. linux下Python网络编程框架-Twisted安装

    Twisted是python下的用来进行网络服务和应用程序编程的框架,安装Twisted前需要系统预先安装有python. 一.安装Twisted http://twistedmatrix.com/R ...

  3. 演练5-2:Contoso大学校园管理2

    一.添加列标题排序功能 我们将增加Student/Index页面的功能,为列标题添加超链接,用户可以点击列标题对那一列进行排序. 1.修改Index方法 public ActionResult Ind ...

  4. Android 判断当前线程是否为主线程

    public static boolean isInMainThread() { return Looper.myLooper() == Looper.getMainLooper(); }

  5. 京香julia_百度百科

    京香julia_百度百科 京香julia

  6. 解决yum升级的问题“There was a problem importing one of the Python modules”

    yum命令升级的时候,报出这个错误. There was a problem importing one of the Python modules required to run yum. The ...

  7. <Win32_18>平滑的人物走动 —— 解决闪屏

    今天咋一看,发现很久没写博客了 的确,开学之后,写博客的时间越来越少了…… 今天来做一个比较实用的小应用——平滑的人物走动,同时解决常见的闪屏问题.实现透明位图 这些技术在游戏开发中是很常见的 --- ...

  8. spring+mybatis利用interceptor(plugin)兑现数据库读写分离

    使用spring的动态路由实现数据库负载均衡 系统中存在的多台服务器是"地位相当"的,不过,同一时间他们都处于活动(Active)状态,处于负载均衡等因素考虑,数据访问请求需要在这 ...

  9. 基于visual Studio2013解决C语言竞赛题之1047百马问题

      题目 解决代码及点评 /* 47.马百瓦问题.有100匹马,100块瓦,大马驮3块, 小马驮2块,两个马驹驮1块.问大马.小马.马驹各多少? 要求:① 不许用for循环; ② 循环次数 ...

  10. Embedded Linux Primer----嵌入式Linux基础教程--章节介绍

    章节介绍 第一章,“导引”,简要介绍了Linux被迅速应用在嵌入式环境的驱动因素,介绍了与嵌入式Linux相关的几个重要的标准和组织. 第二章,“第一个嵌入式经历”,介绍了与后几章所构建的嵌入式Lin ...