【线程】多线程编程

news2024/11/13 9:35:38

目录

一、概念

二、线程函数

1.pthread_create

2.pthread_exit

3.pthread_join 

4.pthread_cancel

三、线程的使用

1.线程的基本操作

2.理解并发运行


一、概念

  • 线程是程序中完成一个独立任务的完整执行序列,即一个可调度的实体。
  • 根据运行环境和调度者的身份,县城可分为内核线程和用户线程。
  • 线程运行在内核空间,由内核来调度。
  • 当进程的一个内核线程获得CPU的使用权时,他就加载并执行一个用户线程
  • 线程库负责管理所有执行线程,比如线程的优先级、时间片等。线程库利用longjmp来切换线程的执行,使他们看起来像”并发“执行,但实际内核仍然是把整个进程作为最小单位来调度。
  • 一个进程的所有执行线程共享该进程的时间片,他们对外表现出相同优先级。
  • 线程的实现方式分为三种模式:完全在用户空间实现、完全由内核调度和双层调度。

常问:进程与线程的区别是?

  • 进程:一个正在运行的程序 ,程序执行完,系统进行回收
  • 线程:进程内部的一条执行路径(序列)
  • 不同语言不同平台上线程的实现机制有送不同

二、线程函数

头文件:

#include<pthread.h>

1.pthread_create

int pthread_create(pthread_t* thread,const pthread_attr_t* attr,void* (*start_routine)(void*),void* arg);
  • 作用:创建一个线程
  • 参数:
    • 第一个参数thread是新线程的标识符,后续pthread_*函数通过它来引用新进程。其类型的pthread_t定义为:
      • #include<bits/pthreadtypes.h>
        typedef unsigned long pthread_t;
      • 其中可见,pthread_t是一个整型类型。
    • 第二个参数attr用于设置新线程的属性。传递NULL表示使用默认线程属性
    • 第三个参数是返回值、参数变量都为void*的函数指针
    • 第四个参数arg表示新线程的参数
  • 返回值:成功时返回0,失败时返回错误码

2.pthread_exit

        线程一旦被创建好,内核就可以调度内核线程来执行start_rountine函数指针所指向的函数。线程函数在结束时最好调用如下函数:

void pthread_exit(void* retval);

该函数通过retval参数向进程的回收者传递其退出信息。

3.pthread_join 

int pthread_join(pthread_t thread,void**retval);
  • 作用:来回收其他线程(前提是目标线程可回收),即等待其他线程结束,类似于wait的使用
    • 一个进程中的所有线程都可以调用pthread_join函数
    • 该函数会一直被阻塞,知道被回收的线程结束为止。
  • 参数:
    • thread是目标现成的标识符
    • retval是目标线程返回的退出信息
  • 返回值:成功0,失败返回错误码

4.pthread_cancel

int pthread_cannel(pthread_t thread);
  • 作用:终止一个线程,即取消线程
  • 参数:
    • thread是目标线程标识符
  • 返回值:成功0,失败返回错误码

三、线程的使用

1.线程的基本操作

通过下面代码理解,首先主函数内执行一个主线程,fun函数执行一个线程,两线程一起执行:

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

void* fun(void* arg)
{
	for(int i=0;i<5;i++)
	{
		printf("fun run\n");
		sleep(1);
	}
}

int main()
{
	pthread_t id;
	pthread_create(&id,NULL,fun,NULL);
	for(int i=0;i<2;i++)
	{
		printf("main run\n");
		sleep(1);
	}
	exit(0);
}

编译执行时,如果不在执行文件后面加上指定库名''-lpthread'',就会出现如下错误:

 执行结果:

执行结果可以得到一个结论:
主函数先结束,线程也跟着结束(fun只输出了4次)线程先结束,主函数不会结束。所以,为了使主函数”活到“最后,需要添加pthread_join函数阻塞。再添加pthread_exit对main函数返回一个值:

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

void* fun(void* arg)
{
	for(int i=0;i<5;i++)
	{
		printf("fun run\n");
		sleep(1);
	}
	pthread_exit("fun over\n");
}


int main()
{
	pthread_t id;
	pthread_create(&id,NULL,fun,NULL);
	for(int i=0;i<2;i++)
	{
		printf("main run\n");
		sleep(1);
	}
	char* s=NULL;
    pthread_join(id,(void**)&s);
    printf("s=%s",s);
	exit(0);
}

执行结果如下,主函数先结束,然后阻塞住,知道线程也执行结束(输出5次),然后线程结束后向主函数发送一个”fun over“传递其退出信息。

2.理解并发运行

并发与并行:

并行:任意时刻,两条程序都在执行(多个处理器)

并发:一个处理器,抢占式方式执行

并行执行

通过创建5个线程,每个线程打印出自己是第几个创建的来观察并行执行的执行结果

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

void* fun(void* arg)
{
	int index=*(int*)arg;
	for(int i=0;i<3;i++)
	{
		printf("index=%d\n",index);
		sleep(1);
	}

}

int main()
{
	pthread_t id[5];
	int i;
	for(i=0;i<5;i++)
	{
		pthread_create(&id[i],NULL,fun,(void*)&i);
	}	
	for(i=0;i<5;i++)
	{
        	pthread_join(id[i],NULL);
	}
	exit(0);
}

每一次的执行结果都会有差异,甚至不是连续01234的数字输出,是因为并发执行是对资源共同操作,当一个线程还在执行index++=2时,还是执行出来index=2的结果,另一个线程对index=1的值进行执行。

 

并发执行:

通过创建5个线程,每个线程打印出i++,来观察并行执行的执行结果

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

int m_val=1;
void* fun(void* arg)
{
        for(int i=0;i<2;i++)
        {
                printf("m_val=%d\n",m_val++);
        }
}

int main()
{
        pthread_t id[5];
        int i;
        for(i=0;i<5;i++)
        {
                pthread_create(&id[i],NULL,fun,(void*)&i);
        }
        for(i=0;i<5;i++)
        {
                pthread_join(id[i],NULL);
        }
        exit(0);
}

执行结果如下,并发执行的结果每次都一样,最终输出值为5个进程,每个打印2次val,最终m_val=10

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/13421.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Maven基础概念【仓库和坐标】这篇看完懂了

Maven下载 官网: Welcome to Apache Maven 下载地址: Maven – Download Apache Maven 01 仓库&#xff1a; 仓库:用于存储资源&#xff0c;包含各种jar包 ​ ​ 仓库&#xff1a;用于存储资源&#xff0c;包含各种jar包 仓库分类&#xff1a; ♦ 本地仓库&#xff1a;自己电…

4_Git

一、Git学习网站 廖雪峰大神 Git 教程Git-flow 包教不包会阮一峰-常用 Git 命令清单 二、Git简介 1.Git是什么 分布式版本控制系统 2.版本控制 3.集中式 vs 分布式 4.Git简史 5.工作区、暂存区、版本库 流程 工作区&#xff1a;写代码 --> git add --> 暂存区 --…

uniapp项目搭建 请求配置

uniapp项目搭建 请求配置请求配置utils/request/index.js代码分析几个常用的方法配置文件的引入编写接口,并测试调用每个人项目用的请求接口不一样,这里就看下实现思路就好了 请求配置 在 uniapp 当中有封装好的 request 插件, request插件地址 在项目的 utils/request/index.…

【华为设备命令最全大合集,快快收藏】

01 华为交换机基础配置命令 01 常用命令视图 02 创建VLAN //用户视图&#xff0c;一般display命令查看信息比较多。 system-view //准备进入系统视图。 [Huawei]vlan 100 //创建vlan 100。 [Huawei-vlan100]quit //退回系统视图。 03 将端口加入到vlan中 [Huawei] interf…

数字源表如何助力miniled光电性能测试

概述 LED在光电子领域中是一种能将电能转化为光能的半导体二极管&#xff0c;包括砷化镓LED(红光)、磷化镓LED(绿光)、氮化镓LED(蓝光)等。Mini LED&#xff0c;则是指尺寸为50-200微米的LED芯片(参照《Mini LED商用显示屏通用技术规范》的定义)&#xff0c;介于小间距LED和Mic…

软件配置 | mac M1 上 imagemagick 的安装

本文简单记录在 mac M1 上安装 Imagemagick 的过程及其简单使用。 2022 / 11 / 16 软件配置 | mac M1 上 imagemagick 的安装安装brew1.2.make参考链接ImageMagick 是免费软件&#xff0c;以随时可用的二进制分发版或源代码形式提供&#xff0c;您可以在开放和专有应用程序中使…

02-Linux

1 Linux文件管理 1.1 touch命令 在Windows系统中&#xff0c;我们如果想创建一个文本文档或者word文件的时候&#xff0c;通常的做法是 鼠标右键---新建---文本文档&#xff0c;这样的话&#xff0c;我们就成功的创建了一个文件&#xff0c;而在Linux中&#xff0c;我们可以通…

好代码 ,坏代码:你的代码和其他工程师的代码

如果你以团队一员的身份编写代码&#xff0c;你所编写的代码很可能建立在其他工程师编写的代码层次的基础上&#xff0c;其他人也可能以你的代码为基础构建新的代码层次。如果你在工作期间解决了各种各样的子问题&#xff0c;并将其分解为清晰的抽象层次&#xff0c;其他工程师…

WebDAV之葫芦儿·派盘+百灵创作

百灵创作 支持webdav方式连接葫芦儿派盘。 心血来潮想写故事,于是进入了创作、码字状态,不写不知道一码吓一跳,这也太累了吧。 基于创作不易,码字辛苦。对文字,我始终怀有尊重与敬畏之心。不知有什么创作码字软件可以解决这些问题,提高写作效率呢?并且防止写好的文章…

spring框架源码十六、BeanDefinition加载注册子流程

BeanDefinition加载注册子流程时序图时序图1step0、new ClassPathXmlApplicationContextstep1、ClassPathXmlApplicationContext#ClassPathXmlApplicationContext(java.lang.String)step2、ClassPathXmlApplicationContext#ClassPathXmlApplicationContext(java.lang.String[],…

初识类和对象

即使是初学者应该对类和对象也不算陌生吧&#xff0c;是不是因为老有些人动不动就&#xff1a;你知道伐&#xff0c;Java是一款面向对象的语言……阿巴阿巴……我的老师告诉我Java难的一部分就是如何把一个对象给抽象出来&#xff0c;那阿涛不才&#xff0c;今天就先来会一会这…

超市售货统计程序

代码price{"牛奶":5.5,"可乐":6.7,"饼干":10,"糖果":10} day1{"day":"11.23","牛奶":10,"可乐":10,"饼干":10,"糖果":10} day2{"day":"11.24",&quo…

机器学习笔记之条件随机场(六)学习任务介绍(Learning)

机器学习笔记之条件随机场——学习任务介绍引言回顾&#xff1a;条件随机场求解边缘概率分布场景设计前向后向算法关于条件随机场的学习任务关于模型参数λ\lambdaλ求解梯度梯度求解梯度的简化过程总结引言 上一节介绍了使用前向后向算法求解基于链式条件随机场中某隐状态的边…

MySQL—优化数据库

优化MySQL数据库是数据库管理员的必备技能&#xff0c;通过不同的优化方式达到提高MySQL数据库性能的目的。本节将介绍优化的基本知识。 MySQL数据库的用户和数据非常少的时候&#xff0c;很难判断一个MySQL数据库性能的好坏。只有当长时间运行&#xff0c;并且有大量用户进行…

GoogLenet网络详解

GoogLenet VGG在2014年由牛津大学著名研究组vGG (Visual Geometry Group)提出&#xff0c;斩获该年lmageNet竞赛中Localization Task (定位任务)第一名和 Classification Task (分类任务)第二名。Classification Task (分类任务)的第一名则是GoogleNet 。GoogleNet是Google研发…

vue-router 使用与原理分析,测试结果来啦

简介 Vue Router 是Vue.js的官方路由。与Vue.js核心深度集成&#xff0c;让用Vue.js构建单页应用&#xff08;SPA&#xff09;变得更加简单。 对于开发和维护管理后台类的前端项目&#xff0c;页面结构和组合可能非常复杂&#xff0c;所以正确的理解和使用Vue Router就显得尤为…

[附源码]SSM计算机毕业设计ssm新冠疫苗预约接种信息管理JAVA

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

Spring事务管理

认识事务 可以把一系列&#xff08;多条sql语句&#xff09;要执行的操作称为事务&#xff0c;而事务管理就是管理这些操作要么完全执行&#xff0c;要么完全不执行&#xff08;很经典的一个例子是&#xff1a;A要给B转钱&#xff0c;首先A的钱减少了&#xff0c;但是突然的数…

EMQX数据流转MySQL踩坑日记:EMQX VER 4.2.3

总结&#xff1a; &#xff08;0&#xff09;数据库报错问题&#xff0c;详细参考这篇文档&#xff0c;链接&#xff0c;ln -s 源 目标 https://blog.csdn.net/weixin_42110159/article/details/118945136 &#xff08;1&#xff09;数据库建立数据&#xff0c;要注意大小写&am…

数字化开采|AIRIOT智慧矿山自动化生产解决方案

由于矿山地形复杂&#xff0c;生产自动化水平低&#xff0c;安全监管技术落后&#xff0c;事故频发等很多因素对煤矿开采技术提出了数据化、可视化、智能化的要求。通过目前的煤矿开采现状可以发现煤矿开采过程中&#xff0c;在生产、监管、巡检、安全、效率等方面还存在许多有…