【linux线程(一)】什么是线程 怎样操作线程

97 阅读6分钟

Linux线程

1. 前言

可能大家对多线程这三个字早有耳闻,
那么到底什么是线程?为什么要有它?
它和进程之间有什么联系?

本章重点:

本篇文章着重讲解线程的基本概率,
以及进程和线程的对比,最后会讲解
在Linux下如何创建,控制,终止,等待线程


2. 什么是线程?

程序中的一个执行路线就叫做线程

一个进程至少要有一个执行线程,单个进程本身就是一个执行流,所以单个进程某种意义上也是一个线程(是主线程).线程在进程内部运行,本质是在进程地址空间内运行.在Linux系统中,在CPU眼中,看到的PCB都要比传统的进程更加轻量化

在Linux系统下,线程是轻量化的进程,我们知道进程有PCB结构和程序地址空间,那么线程的话只有PCB,它的地址空间是和主线程共享的,如下图:

在这里插入图片描述

综上所述,线程就是一个没有独立的地址空间的PCB结构,线程的资源是从最开始的主线程,也就是进程来的.而站在CPU的视角,CPU调度的是PCB结构,CPU只认PCB,它并不关心此PCB是进程的还是线程的,所以线程被称为系统调度的基本单位.而在Linux操作系统下,线程就是轻量化的进程


3. 线程和进程的区别和联系

通过上面对线程的描述,我们可以窥探到:

  1. 线程是担任系统调度的基本实体
  2. 进程是担任系统资源分配的基本实体

虽然线程共享进程数据,但也拥有自己的数据:

  • 线程ID
  • 栈区资源
  • 信号屏蔽字
  • 调度优先级

在这里插入图片描述

使用指令: ps -al查看线程ID

在这里插入图片描述

主线程的PID和LWP相同,CPU调度时是在看LWP,而不是PID,线程的PID和主线程相同,自己独有LWP

在这里插入图片描述


4. Linux下如何操作线程?

在这里插入图片描述

由于pthred是第三方库
所以编译时要加上-lpthread的字段

  1. 如何创建线程

在这里插入图片描述

第一个参数需要传入一个pthread_t类型的变量,这个变量我们需要先定义.所谓的pthread_t,实际上本质就是unsigned long int类型,第二个参数我们一般设置为空.第三个参数要传入线程启动后要执行的函数的地址,这个函数的返回值和参数都是void*,最后一个参数是传入给函数形参的,要强转为void*

一般的编码格式:

void \*route(void \*arg) {
	while(1) {
		printf("I'am a thread \n");
		sleep(1);
	}
}
int main()
{
	pthread\_t tid;
	int ret;
	if ( (ret=pthread\_create(&tid, NULL, rout, NULL)) != 0 ) {
		fprintf(stderr, "pthread\_create : %s\n", strerror(ret));
		exit(-1); //如果创建线程失败会走进此语句
	}
	int i;
	for(; ; ) {
		printf("I'am main thread\n");
		sleep(1);
	}
	return 0;
}

  1. 如何终止线程

在这里插入图片描述

一般使用return返回的来终止线程

  1. 如何进行线程等待

在这里插入图片描述

假如不进行线程等待,可能会出现类似于僵尸进程的问题

一般编码格式:

void \*thread\_run( void \* arg )
{
	pthread\_detach(pthread\_self());
	printf("%s\n", (char\*)arg);
	return NULL;
}
int main( void )
{
	pthread\_t tid;
	if ( pthread\_create(&tid, NULL, thread_run, "thread1 run...") != 0 ) {
		printf("create thread error\n");
		return 1;
	}
	int ret = 0;
	sleep(1);//很重要,要让线程先分离,再等待
	if ( pthread\_join(tid, NULL ) == 0 ) {
		printf("pthread wait success\n");
		ret = 0;
	} else {
		printf("pthread wait failed\n");
		ret = 1;
	}
	return ret;
}

  1. 如何进行线程分离

假如创建一个线程后,不想关心它的返回值,此时使用join等待实际上是一种负担,所以使用线程分离后,可以把创建的线程和主线程分离,这样就不会影响到主线程

在这里插入图片描述

pthread_self()函数可以返回当前线程的线程ID,所以假设我们想要当前线程与主线程脱离关系,可以这样写: pthread_detach(pthread_self());

一般编码格式:

为了做好运维面试路上的助攻手,特整理了上百道 【运维技术栈面试题集锦】 ,让你面试不慌心不跳,高薪offer怀里抱!

这次整理的面试题,小到shell、MySQL,大到K8s等云原生技术栈,不仅适合运维新人入行面试需要,还适用于想提升进阶跳槽加薪的运维朋友。

本份面试集锦涵盖了

  • 174 道运维工程师面试题
  • 128道k8s面试题
  • 108道shell脚本面试题
  • 200道Linux面试题
  • 51道docker面试题
  • 35道Jenkis面试题
  • 78道MongoDB面试题
  • 17道ansible面试题
  • 60道dubbo面试题
  • 53道kafka面试
  • 18道mysql面试题
  • 40道nginx面试题
  • 77道redis面试题
  • 28道zookeeper

总计 1000+ 道面试题, 内容 又全含金量又高

  • 174道运维工程师面试题

1、什么是运维?

2、在工作中,运维人员经常需要跟运营人员打交道,请问运营人员是做什么工作的?

3、现在给你三百台服务器,你怎么对他们进行管理?

4、简述raid0 raid1raid5二种工作模式的工作原理及特点

5、LVS、Nginx、HAproxy有什么区别?工作中你怎么选择?

6、Squid、Varinsh和Nginx有什么区别,工作中你怎么选择?

7、Tomcat和Resin有什么区别,工作中你怎么选择?

8、什么是中间件?什么是jdk?

9、讲述一下Tomcat8005、8009、8080三个端口的含义?

10、什么叫CDN?

11、什么叫网站灰度发布?

12、简述DNS进行域名解析的过程?

13、RabbitMQ是什么东西?

14、讲一下Keepalived的工作原理?

15、讲述一下LVS三种模式的工作过程?

16、mysql的innodb如何定位锁问题,mysql如何减少主从复制延迟?

17、如何重置mysql root密码? 详情docs.qq.com/doc/DSmdCdUNwcEJDTXFK