多线程基础(一)线程基础信息、synchronized 锁概念

news/2024/11/19 17:37:37/

1. 基本概念:

程序:

        程序是一些保存在磁盘上的指令的有序集合,是静态的。程序包括:内存资源、IO资源、信号处理等。(如:XX.exe)

进程:

        进程是程序执行的过程,包括了动态创建、调度和消亡的整个过程,进程是程序资源管理的最小单位。(当:XX.exe执行后,就会创建进程)

线程:

        线程是操作操作系统能够进行运算调度的最小单位。线程被包含在进程之中,是进程中的实际运作单位,一个进程内可以包含多个线程,线程是资源调度的最小单位。        

协程/纤程:

        协程 Coroutines 是一种比线程更加轻量级的微线程。类比一个进程可以拥有多个线程,一个线程也可以拥有多个协程,因此协程又称微线程和纤程。

2. 线程的启动

public class theardTest {private static class T1 extends Thread{@Overridepublic void run(){for (int i=0;i<10;i++){try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("T1");}}}public static void main(String[] args) {new T1().run();// new T1().start();for (int i=0;i<10;i++){try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("Main");}}}

使用 new T1().run(); 执行结果:

T1
Main
Main
Main
Main
Main
Main
Main
Main
Main
Main

结论:使用 new T1().run(); 方式执行启动线程,会先执行 T1 中的方法,然后再执行主程序 Main方法。

使用 new T1().start(); 方式执行后:

Main
T1
T1
Main
T1
Main
Main
T1
Main
T1
Main

结论:使用 new T1().start(); T1 线程会和主线程抢占资源执行。

面试题:启动(实现)线程的三种方式:

1. 继承 Thread类
2. 实现 Runnable接口

3. 使用线程池。如: Executors.newCachedThreadPool();

 3. 线程的基本方法

// 当前线程睡眠1000毫秒
Thread.sleep(1000);// 当前线程回到等待队列,重新竞争cpu资源
Thread.yield();// 当 T1线程执行完毕后,才开始执行当前线程
T1.join();

注意:T1.join() 方法 可以确保线程执行的顺序。

4. 线程状态

new 新建状态 ——>start 进入线程 ——>runnable 可执行状态 ——>cpu选中 running 执行状态 ——>blocked 等待状态 ——> terminated 结束状态。

注:一般我们不用stop是停止线程。而是使用 intertupted方法去判断当前线程是否已经中断,去清除状态。

5. synchronized 锁

JVM规范里并没有说明synchronized必须要怎么实现,它只要给一个对象加上锁,才可以去执行锁里的代码,锁的是对象,不是代码。并且,锁定的方法和非锁定的方法会同时执行,这个要注意脏读问题。

并且 synchronized 是可以保证可见性、原子性、有序性的。而 volatile不保证原子性。

synchronized 包括 monitor enter 获取监视器锁、monitor exi 释放监视器锁,这个重要JVM指令,java中每个对象都关联着一个监视器,当线程获取了某一对象的监视器锁,再次获取时,只增加对应的计数,不在重新执行一次获取过程。

这就是 synchronized 的可重入锁,比如:m1方法加了 synchronized ,在m1中调用m2,m2方法也是加了synchronized锁的,但是,m1可以正常调用m2方法。

即:一个同步方法可以调用另外一个同步方法,一个线程已经拥有某个对象的锁,(调用其他方法时)再次申请锁的时候,仍然会得到该对象的锁,只增加对应的计数,这就是获得的锁是可重入的。

synchronized的锁升级

markword 记录这个线程ID(偏向锁)

如果线程争用:升级为 自旋锁(轻量级锁)

10次以后,升级为重量级锁 - OS(向操作系统申请锁)

锁只能升不能降(锁膨胀)。执行时间短《加锁代码),线程数少,用自旋,执行时间长,线程数多,用系统锁

使用锁的注意事项

1. 程序在执行过程中,如果出现异常,默认情况锁会被释放所以,在并发处理的过程中,有异常要多加小心,不然可能会发生不一致的情况。比如,在一个web app处理过程中,多个servlet线程共同访网同一个资源,这时如果异常处理不合适在第一个线程中抛出异常,其他线程就会进入同步代码区,有可能会访问到异常产生时的数据。因此要非常小心前处理同步业务逻超中的异常. 

2. synchronized(Object)不能使用String常量、Integer、Long类型等基础数据类型。

3. 锁的细粒度和粗粒度

        实际业务逻辑中,如果只有很少的代码(一行或者几行)需要加锁,则在这些代码上加锁,这就是采用细粒度的锁,可以使线程争用时向变短,从而提高效率。

// 业务逻辑中只有下面这句需要sync,这时不应该给整个方法上锁
// 采用细粒度的锁,可以使线程争用时问变短,从而楚高效率
synchronized(this) {count++;
}

        如果实际业务中,这个方法下有多个代码段都需要使用sync锁,则需要在该方法上加锁,使用 粗粒度锁,避免频繁调用 sync锁。

4. 锁定某对象o,如果o的属性发生改变,不形响锁的使用但是如果o变成另外一个对象,则锁定的对象发生改变,应该避免将锁定对象的引用变成另外的对象。即:需要在该对象前加上 final 属性。


http://www.ppmy.cn/news/75772.html

相关文章

XPCIE1032 — 高速高精,超快交互速率的PCIe EtherCAT实时运动控制卡

产品导读 XPCIE1032是一款基于PCI Express的EtherCAT总线运动控制卡&#xff0c;可选4-16轴运动控制&#xff0c;支持多路高速数字输入输出&#xff0c;可轻松实现多轴同步控制和高速数据传输。 XPCIE1032集成了强大的运动控制功能&#xff0c;结合MotionRT7运动控制实时软核…

redis 数据库概述

一 概述 redis是一种nosql数据库,他的数据是保存在内存中&#xff0c;同时redis可以定时把内存数据同步到磁盘&#xff0c;即可以将数据持久化&#xff0c;并且他比memcached支持更多的数据结构(string,list列表[队列和栈],set[集合],sorted set[有序集合],hash(hash表))。相关…

SpringBoot中使用枚举类、switch、常量类(声明并初始化map)实现类策略者模式,接口返回常量数据

场景 SpringBoot中策略模式工厂模式业务实例(接口传参-枚举类查询策略映射关系-执行不同策略)规避大量if-else&#xff1a; SpringBoot中策略模式工厂模式业务实例(接口传参-枚举类查询策略映射关系-执行不同策略)规避大量if-else_霸道流氓气质的博客-CSDN博客 SpringBootVa…

learn_C_deep_12 (深度理解“取整“、“取余“、“取模“运算、掌握运算符优先级 )

目录 关于“取整” "取整"规则 1、向零取整 2、向-∞取整 3、向∞取整 4、四舍五入 关于"取模和取余" 运算符优先级 关于“取整” #include <stdio.h> int main() {//本质是向0取整int i -2.9;int j 2.9;printf("%d\n", i); /…

数据结构与算法之逻辑回归详解

逻辑回归&#xff08;Logistic Regression&#xff09;是一种广义线性模型&#xff0c;在分类问题中广泛应用&#xff0c;特别是二分类问题。逻辑回归可以通过把输入的特征与权重线性组合&#xff0c;再通过逻辑函数&#xff08;sigmoid函数&#xff09;将结果映射到0~1之间&am…

云HIS住院业务模块常见问题及解决方案

一&#xff1a;住院业务 1.患者办理住院时分配了错误的病区怎么办&#xff1f; 操作员误操作将患者分配了错误的病区分为以下两种情况&#xff1a; &#xff08;1&#xff09;、患者刚刚入院&#xff0c;未分配床位、主治医师与管床护士&#xff1a;这种情况比较好处理&#xf…

SpringCloud 性能优化

文章目录 Springcloud的性能问题应用服务组件调优Servlet 容器 优化Feign 配置优化 Gateway组件调优Zuul配置 优化hystrix配置 优化ribbon 优化 Springcloud的性能问题 Springcloud 原始的配置&#xff0c;性能是很低的&#xff0c;大家可以使用Jmeter测试一下&#xff0c;QPS…

go-gf框架查询结果数组里套数组 数据和方法

数据 type ObtainClassificationRes struct {Id int json:"id" dc:"用户名"Name string json:"name" dc:"标题"Introduce string json:"introduce" dc:"介绍"Picture string…