线程安全问题:

简单来说,就是多个线程在操作同一个变量时引起的问题。

这里是用一个简单的例子说明一下:

以Runnable创建的线程为例:一个售票系统,count代表当前票数,卖出一张count--。

Runnable线程类:

public class Runnable_Exp implements Runnable{
private int count = 50;
@Override
public void run() {
while(true)
if(count >= 0){
System.out.println(Thread.currentThread().getName() +"现在的数:"+count);
count--;
}else break;
}
}

在main调用数个相同的Runnable线程:

Runnable_Exp rExp = new Runnable_Exp();
Thread trs1 = new Thread(rExp,"线程1");
Thread trs2 = new Thread(rExp,"线程2");
Thread trs3 = new Thread(rExp,"线程3");
Thread trs4 = new Thread(rExp,"线程4");
trs1.start();trs2.start();trs3.start();trs4.start();

运行:

我们会发现一个问题,票数会有相同的!

这就是线程安全的体现。此时我们就需要使用线程锁解决这个问题。

另外,我们之前学习的集合类中,有分线程安全与线程不安全的两类。

StringBuffer和Vector是线程安全的。

StringBuilder和ArrayList是线程不安全的。

在多线程中,我们不可以使用多个线程同时操作不安全的集合类


线程锁的使用:synchronized

Runnable中使用线程锁:

我们只需要修改一下Runnable线程类。

public class Runnable_Exp implements Runnable{
private int count = 500; private Object lock = new Object(); @Override
public void run() {
while(true)
synchronized (lock) {
if (count >= 0) {
System.out.println(Thread.currentThread().getName() + "现在的数:" + count);
count--;
} else break;
}
}
}

解析:

添加一个线程锁对象,一般使用Object即可。

private Object lock = new Object();

这个Object对象lock是四个线程共享的(且这个lock一定是共用的!)可以自己尝试一下把锁放进run()中。

synchronized (lock)

当线程一运行到上面这一行的时候,线程一就会占用lock对象(相当于上锁了),直到线程一执行完synchronized语句块内的全部代码,lock对象就会取消占用。

若线程二运行到synchronized语句时,发现lock对象已经被占用了,则会等待,直到lock被取消占用。

有可能多个线程一起抢占。

效率会比较低。毕竟是你执行完再到我执行。无法同步执行。这就是所谓“抢占式”执行。

Thread中使用线程锁:

Thread线程类中:

public class Thread_Exp extends Thread {
public Thread_Exp(){}
public Thread_Exp(String name){
super(name);
}
private static Object lock = new Object();
private static int count = 100;
public void run(){
while(true)
synchronized(lock){
if (count >= 0) {
System.out.println(Thread.currentThread().getName() + "现在的数:" + count);
count--;
} else break;
}
}
}

需要注意的是,这里的线程锁对象和count都需要设置为静态的!

Thread_Exp t1 = new Thread_Exp("线程1");
Thread_Exp t2 = new Thread_Exp("线程2");
Thread_Exp t3 = new Thread_Exp("线程3");
Thread_Exp t4 = new Thread_Exp("线程4");
t1.start();t2.start();t3.start();t4.start();

不创建Object对象可以实现相同的操作吗?

答案是可以的,我们可以不用创建Object类型的lock对象。

我们直接用synchronized(this)就可以了。

相当于检测当前对象是非被线程占用了(加锁)。

实例:

@Override
public void run() {
synchronized (this){
while(true)
if (count >= 0) {
System.out.println(Thread.currentThread().getName() + "现在的数:" + count);
count--;
} else break;
}
}

我们还可以直接使用synchronized关键字描述一个方法,具体请看下文。


使用synchronized方法简化上述代码:

同步方法,顾名思义,就是自带线程锁的方法。

只需要在方法的返回值前面添加synchronized关键字即可。

原先的代码我们可以写成:

使得run()方法更加简洁。

@Override
public void run() {
Count();
} public synchronized void Count(){
while(true)
if (count >= 0) {
System.out.println(Thread.currentThread().getName() + "现在的数:" + count);
count--;
} else break;
}

使用同步方法,这个方法体就会自动被加锁。


关于线程安全的方法:

上面我们说了StringBuffer方法是线程安全的,那我们是怎么知道的呢?

我们可以直接查看StringBuffer中方法的源代码。

例如attend方法:

@Override
@IntrinsicCandidate
public synchronized StringBuffer append(char c) {
toStringCache = null;
super.append(c);
return this;
}

我们可以看到这个方法是synchronized方法。

大家可以自行去研究一下各种Java自带的方法,是否是线程安全的。

【Java】学习路径47-线程锁synchronized的更多相关文章

  1. 【Java】学习路径48-线程锁ReentrantLock

    与上一章学习的线程锁synchronized类似,都是为了解决线程安全的问题. 使用方法: 新建一个ReentrantLock对象.(如果使用Thread多线程,则需要声明static静态) 然后在需 ...

  2. Java学习路径及练手项目合集

    Java 在编程语言排行榜中一直位列前排,可知 Java 语言的受欢迎程度了. 实验楼上的[Java 学习路径]中将首先完成 Java基础.JDK.JDBC.正则表达式等基础实验,然后进阶到 J2SE ...

  3. java学习笔记15--多线程编程基础2

    本文地址:http://www.cnblogs.com/archimedes/p/java-study-note15.html,转载请注明源地址. 线程的生命周期 1.线程的生命周期 线程从产生到消亡 ...

  4. Java学习路径(抛光砖)

    这就是我刚刚在五孔问答中找到的Java学习路线图抛光砖价格.我个人认为,这条Java学习路线是可以的.它是2018年相对较新的Java学习路线,更符合企业就业标准. Java学习路径的第一阶段:Jav ...

  5. java学习笔记14--多线程编程基础1

    本文地址:http://www.cnblogs.com/archimedes/p/java-study-note14.html,转载请注明源地址. 多线程编程基础 多进程 一个独立程序的每一次运行称为 ...

  6. 深入理解java:2.2. 同步锁Synchronized及其实现原理

    同步的基本思想 为了保证共享数据在同一时刻只被一个线程使用,我们有一种很简单的实现思想,就是 在共享数据里保存一个锁 ,当没有线程访问时,锁是空的. 当有第一个线程访问时,就 在锁里保存这个线程的标识 ...

  7. 【转】Java学习---深入理解线程池

    [原文]https://www.toutiao.com/i6566022142666736131/ 我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很 ...

  8. Java学习路径:不走弯路,这是一条捷径

    1.如何学习编程? JAVA是一种平台.也是一种程序设计语言,怎样学好程序设计不只适用于JAVA,对C++等其它程序设计语言也一样管用.有编程高手觉得,JAVA也好C也好没什么分别,拿来就用.为什么他 ...

  9. 【Java并发系列04】线程锁synchronized和Lock和volatile和Condition

    img { border: solid 1px } 一.前言 多线程怎么防止竞争资源,即防止对同一资源进行并发操作,那就是使用加锁机制.这是Java并发编程中必须要理解的一个知识点.其实使用起来还是比 ...

随机推荐

  1. 测试人生 | 薪资翻倍涨至50W是种什么样的体验?

    本文为霍格沃兹测试开发学社优秀学员跳槽笔记,测试开发进阶学习文末加群. 本人已经工作7年了,做的都是功能测试以及写一些简单的自动化脚本,加上之前没有学习的意识,导致专业技术水平与工作年限不匹配,在上家 ...

  2. 如何利用 RPA 实现自动化获客?

    大家好,我是二哥.前高级技术专家 & 增长黑客,现一枚爱折腾的小小创业者,专注于 RPA & SaaS 软件这块.这次给大家带来如何利用 RPA 实现自动化获客 一.RPA 是什么?难 ...

  3. 最简单的离散概率分布,伯努利分布 《考研概率论学习之我见》 -by zobol

    上文讲了离散型随机变量的分布,我们从最简单的离散型分布伯努利分布讲起,伯努利分布很简单,但是在现实生活中使用的很频繁.很多从事体力工作的人,在生活中也是经常自觉地"发现"伯努利分布 ...

  4. 阿里云体验有奖:使用PolarDB-X与Flink搭建实时数据大屏

    体验简介 场景将提供一台配置了CentOS 8.5操作系统的ECS实例(云服务器).通过本教程的操作带您体验如何使用PolarDB-X与Flink搭建一个实时数据链路,模拟阿里巴巴双十一GMV大屏. ...

  5. centos7解决无法上网的问题

    问题:centos7出现无法进行联网,如下图所示,执行该命令: ping qq.com 出现如下情况: 解决方法: 首先cd到需要修改文件的所在目录下: [root@localhost ~]# cd ...

  6. 《A Neural Algorithm of Artistic Style》理解

    在美术中,特别是绘画,人类掌握了通过在图像的内容和风格间建立复杂的相互作用从而创造独特的视觉体验的技巧.到目前为止,这个过程的算法基础是未知的,也没有现存的人工系统拥有这样的能力.然而在视觉感知的其他 ...

  7. Tapdata 数据库实时同步的技术要点

    Tapdata 是由深圳钛铂数据有限公司研发的一款实时数据处理及服务的平台产品,企业可以使用 Tapdata 快速构建数据中台和实时数仓, Tapdata 提供了一站式的解决方案,包括实时数据采集.数 ...

  8. FileNameFilter过滤器的使用和Lambda优化程序--IO概述(概念&分类)

    FileNameFilter过滤器的使用和Lambda优化程序 public class Demo02Filter { public static void main(String[] args) { ...

  9. 如何用WebGPU流畅渲染百万级2D物体?

    大家好~本文使用WebGPU和光线追踪算法,从0开始实现和逐步优化Demo,展示了从渲染500个2D物体都吃力到流畅渲染4百万个2D物体的优化过程和思路 目录 需求 成果 1.选择渲染的算法 2.实现 ...

  10. 20220722-Java中this关键字

    this关键字知识总结 学习资源:B站韩顺平老师Java入门教学 代码示例1 public class This01 { public static void main(String[] args) ...