POSIX线程库

与线程有关的函数构成了一个完整的系列,绝大多数函数的名字都是以“pthread_”开头,要使用这些函数库,要通过引入头文<pthread.h>,而且链接这些线程函数库时要使用编译器命令的“-lpthread”选项[Ubuntu系列系统需要添加的是”-pthread”选项而不是”-lpthread”,如Ubuntu 14.04版本,深度Ubuntu等]

1.pthread_create

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

创建一个新的线程

参数

thread:线程ID

attr:设置线程的属性,一般设置为NULL表示使用默认属性

start_routine:是个函数地址,线程启动后要执行的函数

arg:传给线程启动函数的参数

返回值:成功返回0;失败返回错误码;

附-Posix错误检查

UNIX传统的函数:成功返回0,失败返回-1,并且对设置全局变量errno以指定错误类型。然而pthreads函数出错时不会设置全局变量errno(而其他的大部分POSIX函数会设置errno)。而是将错误代码通过返回值返回;

pthreads同样也提供了线程内的errno变量,对于每一个线程, 都有一个errno的值, 以支持其它使用errno的代码。对于pthreads函数的错误,建议通过返回值进行判定,因为读取返回值要比读取线程内的errno变量的开销更小!

/** 实践: 新的错误检查与错误退出函数 **/
inline void err_check(const std::string &msg, int retno)
{if (retno != 0)err_exit(msg, retno);
}
inline void err_exit(const std::string &msg, int retno)
{std::cerr << msg << ": " << strerror(retno) << endl;exit(EXIT_FAILURE);
}

2.pthread_exit

void pthread_exit(void *value_ptr);

线程终止

value_ptr:指向该线程的返回值;注意:value_ptr不能指向一个局部变量。

3.pthread_join

int pthread_join(pthread_t thread, void **value_ptr);

等待线程结束

value_ptr:它指向一个指针,后者指向线程的返回值(用户获取线程的返回值)

/** 示例: 等待线程退出 **/
void *thread_rotine(void *args)
{for (int i = 0; i < 10; ++i){printf("B");fflush(stdout);usleep(20);}pthread_exit(NULL);
}int main()
{pthread_t thread;int ret = pthread_create(&thread, NULL, thread_rotine, NULL);err_check("pthread_create", ret);for (int i = 0; i < 10; ++i){printf("A");fflush(stdout);usleep(20);}ret = pthread_join(thread, NULL);err_check("pthread_join", ret);putchar('\n');return 0;
}

4.pthread_self

pthread_t pthread_self(void);

返回线程ID

/** 示例:主控线程与子线程传递数据 **/
typedef struct _Student
{char name[20];unsigned int age;
} Student;void *threadFunction(void *args)
{cout << "In Thread: " << pthread_self() << endl;Student tmp = *(Student *)(args);cout << "Name: " << tmp.name << endl;cout << "Age: " << tmp.age << endl;pthread_exit(NULL);
}int main()
{Student student = {"xiaofang",22};pthread_t thread;//启动创建并启动线程pthread_create(&thread,NULL,threadFunction,&student);//等待线程结束pthread_join(thread,NULL);return 0;
}

5.pthread_cancel

int pthread_cancel(pthread_t thread);

取消一个执行中的线程

6.pthread_detach

int pthread_detach(pthread_t thread);

将一个线程分离-如果在新创建的线程结束时主线程没有结束同时也没有调用pthread_join,则会产生僵线程,次问题可以通过设置线程为分离的(detach)来解决;

总结:进程 VS. 线程

进程(pid_t)

线程(pthread_t)

Fork

Pthread_create

Waitpit

Pthread_join/Pthread_detach

Kill

Pthread_cancel

Pid

Pthead_self

Exit/return

Pthread_exit/return

僵尸进程(没有调用wait/waitpid等函数)

僵尸线程(没有调用pthread_join/pthread_detach)

/** 将并发echo server改造成多线程形式
注意线程竞速问题的解决
**/
void echo_server(int clientSocket);
void *thread_routine(void *arg);
int main()
{int sockfd = socket(AF_INET,SOCK_STREAM,0);if (sockfd == -1)err_exit("socket error");int optval = 1;if (setsockopt(sockfd,SOL_SOCKET,SO_REUSEADDR,&optval,sizeof(optval)) == -1)err_exit("setsockopt error");struct sockaddr_in serverAddr;serverAddr.sin_family = AF_INET;serverAddr.sin_port = htons(8002);serverAddr.sin_addr.s_addr = INADDR_ANY;    //绑定本机的任意一个IP地址if (bind(sockfd,(struct sockaddr *)&serverAddr,sizeof(serverAddr)) == -1)err_exit("bind error");if (listen(sockfd,SOMAXCONN) == -1)err_exit("listen error");while (true){int peerSockfd = accept(sockfd, NULL, NULL);if (peerSockfd == -1)err_exit("accept error");pthread_t tid;/**注意: 下面这种用法可能会产生"竞速问题"当另一个连接快读快速到达, peerSockfd的内容更改,新创建的线程尚未将该值取走时,线程读取的就不是我们想让线程读取的值了int ret = pthread_create(&tid, NULL, thread_routine, (void *)&peerSockfd);**///解决方案: 为每一个链接创建一块内存int *p = new int(peerSockfd);int ret = pthread_create(&tid, NULL, thread_routine, p);if (ret != 0)err_thread("pthread_create error", ret);}close(sockfd);
}
void *thread_routine(void *args)
{//将线程设置分离状态, 避免出现僵尸线程pthread_detach(pthread_self());int peerSockfd = *(int *)args;//将值取到之后就将这块内存释放掉delete (int *)args;echo_server(peerSockfd);cout << "thread " << pthread_self() << " exiting ..." << endl;pthread_exit(NULL);
}
void echo_server(int clientSocket)
{char buf[BUFSIZ] = {0};int readBytes;while ((readBytes = read(clientSocket, buf, sizeof(buf))) >= 0){if (readBytes == 0){cerr << "client connect closed" << endl;break;}if (write(clientSocket, buf, readBytes) == -1){cerr << "server thread write error" << endl;break;}cout << buf;bzero(buf, sizeof(buf));}
}

其完整源代码:download.csdn.net/detail/hanqing280441589/8440763

Linux多线程实践(2) --线程基本API相关推荐

  1. Linux多线程实践(1) --线程理论

    线程概念 在一个程序里的一个执行路线就叫做线程(thread).更准确的定义是:线程是"一个进程内部的控制序列/指令序列"; 一切进程至少有一个执行线程; 进程  VS. 线程  ...

  2. Linux多线程实践(4) --线程特定数据

    线程特定数据 int pthread_key_create(pthread_key_t *key, void (*destr_function) (void *)); int pthread_key_ ...

  3. Linux多线程实践(3) --线程属性

    初始化/销毁线程属性 int pthread_attr_init(pthread_attr_t *attr); int pthread_attr_destroy(pthread_attr_t *att ...

  4. Linux多线程实践(二)线程基本API(POSIX)

    我们知道,进程在各自独立的地址空间中运行,进程之间共享数据需要用进程间通信机制,有些情况需要在一个进程中同时执行多个控制流程,这时候线程就派上了用场,比如实现一个图形界面的下载软件,一方面需要和用户交 ...

  5. Linux多线程实践(一)线程基本概念和理论

    线程概念 在一个程序里的一个运行路线就叫做线程(thread).更准确的定义是:线程是"一个进程内部的控制序列/指令序列"; 对于每一个进程至少有一个运行线程; 进程  VS. 线 ...

  6. Linux多线程实践(9) --简单线程池的设计与实现

    线程池的技术背景 在面向对象编程中,创建和销毁对象是很费时间的,因为创建一个对象要获取内存资源或者其它更多资源.在Java中更是如此,虚拟机将试图跟踪每一个对象,以便能够在对象销毁后进行垃圾回收.所以 ...

  7. Linux多线程实践(四 )线程的特定数据

    在单线程程序中.我们常常要用到"全局变量"以实现多个函数间共享数据, 然而在多线程环境下.因为数据空间是共享的.因此全局变量也为全部线程所共同拥有.但有时应用程序设计中有必要提供线 ...

  8. Linux多线程实践(10) --使用 C++11 编写 Linux 多线程程序

    在这个多核时代,如何充分利用每个 CPU 内核是一个绕不开的话题,从需要为成千上万的用户同时提供服务的服务端应用程序,到需要同时打开十几个页面,每个页面都有几十上百个链接的 web 浏览器应用程序,从 ...

  9. Linux多线程实践(8) --Posix条件变量解决生产者消费者问题

    Posix条件变量 int pthread_cond_init(pthread_cond_t *cond, pthread_condattr_t *cond_attr); int pthread_co ...

最新文章

  1. Response.Redirect 打开新窗体的两种方法
  2. App 组件化/模块化之路——如何封装网络请求框架
  3. React入门---react脚手架
  4. python函数调用位置_python函数定义,调用,传参,位置参数及关键字参数,返回值
  5. 自定义scoll样式
  6. 一个利用sql 语句来实现分页的存储过程
  7. web app 自适应方案总结 弹性布局之rem
  8. 2021靠谱的IT培训机构排名重磅来袭!
  9. 谷歌浏览器加载外部 DLL 文件 关于chrome上的网银安全控件开发技术(chrome 调用本地dll)
  10. python用户名和密码登录_Python爬虫:账号密码登入扇贝
  11. vue——动态吸顶组件
  12. android渠道占有率,硬核联盟分发量在安卓渠道占比高达64%,2019年将发力四大方向...
  13. Linux下载神器XDM,代替IDM
  14. 英文间隔符占位html,HTML空格占位
  15. Shopee打包贴单商品代发选星卓越货代服务系统
  16. GBase 8c 迁移工具 DMT 简述
  17. 哪里有云南ip服务器,云南那些服务商可以提供云南本地ip服务器
  18. 用计算机怎么求反三角函数图像及性质,反三角函数图像及性质
  19. RLC振荡原理与RC Snubber吸收电路
  20. 【大数据处理技术】实验11

热门文章

  1. Android开发:5-2、ListView、GridView、Spinner
  2. (王道408考研数据结构)第七章查找-第二节3:分块查找
  3. 深入理解计算机的字长
  4. LeetCode 122 买卖股票的最佳时机 II
  5. 网络安全设备常用默认弱口令
  6. UWP Acrylic Material
  7. C# Linq to Entity Lamda方式分组并求和求平均值
  8. 最佳调度问题(搜索回溯)
  9. 【网络基础】路由表,分组转发算法
  10. Oracle体系结构四(学习笔记)