线程安全问题:

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

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

以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. Node.js amqplib 连接 Rabbitmq 学习笔记

    var amqp = require('amqplib'); connect([url, [socketOptions]]) var amqp = require('amqplib/callback_ ...

  2. 【Redis】事件驱动框架源码分析

    aeEventLoop初始化 在server.c文件的initServer函数中,对aeEventLoop进行了初始化: 调用aeCreateEventLoop函数创建aeEventLoop结构体,对 ...

  3. 【Spring】事务的执行原理(二)

    前置知识 事务的执行步骤如下: 获取事务管理器 创建事务 执行目标方法 捕捉异常,如果出现异常进行回滚 提交事务 public abstract class TransactionAspectSupp ...

  4. JS:条件语句1

    条件语句: 1.if...else if (condition1) { 当条件 1 为 true 时执行 } else { 当条件 1 不为 true 时执行 } if (condition1) { ...

  5. Python教程:随机验证码生成和join 字符串

    函数:string.join() Python中有join()和os.path.join()两个函数,具体作用如下: join(): 连接字符串数组.将字符串.元组.列表中的元素以指定的字符(分隔符) ...

  6. BigDecimal加减乘除及setScale的用法小结

    Bigdecimal初始化: BigDecimal num = new BigDecimal(2.225667);//这种写法不允许,会造成精度损失. BigDecimal num = new Big ...

  7. kubernetes之ingress探索实践

    3.Ingress实践 3.1.什么是Ingress? 在ingress之前,我们想要访问k8s集群中的pod服务时,是通过部署一个service,将service的type设置为NodePort或者 ...

  8. JDBC:获取自增长键值的序号

    1.改变的地方  实践: package com.dgd.test; import java.io.FileInputStream; import java.io.FileNotFoundExcept ...

  9. Java编程作业

    1.编程题 设计一个用户类User,类中的变量有用户名.密码和记录用户数量的变量,定义3个构造方法:无参的.为用户名赋值的.为用户名和密码赋值的,还有获取和设置密码的方法和返回类信息的方法. pack ...

  10. 经典面试题:==和equals的区别

    1.== 既可以比较基本类型也可以比较引用类型.对于基本类型就是比较值,对于引用类型就是比较内存地址 2.equals的话,它是属于java.lang.Object类里面的方法,如果该方法没有被重写过 ...