当前位置: 首页 > news >正文

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

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类型等基础数据类型。

http://www.lryc.cn/news/69179.html

相关文章:

  • JAVA期末考内容知识点的梳理
  • 为什么要使用Thrift与Protocol Buffers?
  • oa是什么意思?oa系统哪个好用?
  • Linq和C# Lambda表达式
  • 蓝桥:前端开发笔面必刷题——Day2 数组(三)
  • 人工智能专栏第四讲——人工智能的未来展望与机遇
  • Unity阴影(Shadow)、Shadowmap
  • 编程语言的四种错误处理方法,你知道几种?
  • ContOS7单机安装Hadoop
  • 抓取动态网页的数据的具体操作方法
  • Windows 和 Linux 环境下 ProtoBuf 的安装
  • 商用密码应用安全性测评方案编制流程
  • Elasticsearch 集群部署插件管理及副本分片概念介绍
  • Liunx 套接字编程(2)TCP接口通信程序
  • 8年开发经验,浅谈 API 管理
  • 【软考备战·四月模考】希赛网四月模考软件设计师上午题
  • MySQL中的@i:=@i+1用法详解
  • web安全第一天 ,域名,dns
  • 【Linux】Linux编辑神器vim的使用
  • vulnhub渗透测试靶场练习1
  • Uart,RS232,RS485串口通讯协议学习
  • UML中的assembly关系
  • [Python]缓存cachetools与TTLCache简介
  • 现在的00后,真是卷死了呀,辞职信已经写好了·····
  • 【wpf】列表类,用相对源时,如何绑定到子项
  • 头歌计算机组成原理实验—运算器设计(3)第3关:4位快速加法器设计
  • Java中synchronized的优化
  • 软件测试技术课程:软件测试流程
  • 【Redis】聊一下缓存双写一致性
  • Java学习笔记-04