多线程编程-- part 3 多线程同步->synchronized关键字
多线程同时访问一个资源,可以会产生不可预料的结果,所以为这个资源加锁,访问资源的第一个线程为其加锁后,其他线程便不能在使用那个资源,直到锁被解除。
举个例子:
存款1000元,能取出800的时候我就取800,当我同时用两个线程调用这个取钱操作时,有时可以取出1600元
static class HelloRunable implements Runnable{
         private int money = 1000;
         //取出800元
          int getMoney() {
             System.out.println("开始" + money);
             if(money > 800) {
                 try {
                     Thread.sleep(100);
                 } catch (InterruptedException e) {
                     e.printStackTrace();
                 }
                 money -= 800;
                 System.out.println("我取了800元" + money);
             }
             System.out.println("结束" + money);
             return money;
         }
        @Override
        public void run() {
            money = getMoney();
        }
    }
    public static void main(String[] args) {
        HelloRunable helloRunable = new HelloRunable();
        Thread t = new Thread(helloRunable);
        Thread t1 = new Thread(helloRunable);
        t.start();
        t1.start();
    }
  
synchronized:
所以我们引入了同步机制,在取钱方法前面加入了synchronized关键字,该方法称为同步方法,输出结果为

java中每个对象都有一个锁,或者叫做监视器(monitor),当一个线程访问某个对象的synchronized方法时,将该对象上锁,其他任何线程都无法再去访问该对象的synchronized方法,直到之前的那个线程执行完方法之后,才会将对象锁释放,其他线程才可以在用该对象的synchronized方法。 注:这是给对象上锁,如果是不同的对象则该对象之间没有限制关系。
如果一个对象有多个synchronized方法,某一时刻某个线程已经进入到了synchronized方法,那么在该方法没有执行完前,其他线程是无法访问该对象的任何synchronized方法的。
例如:
public static void main(String[] args)
{
Example example = new Example(); Thread t1 = new Thread1(example);
Thread t2 = new Thread2(example); t1.start();
t2.start();
} static class Example
{
public synchronized void execute()
{
for (int i = 0; i < 20; ++i)
{
try
{
Thread.sleep((long) Math.random() * 1000);
}
catch (InterruptedException e)
{
e.printStackTrace();
}
System.out.println("Hello: " + i);
}
} public void execute2()
{
for (int i = 0; i < 20; ++i)
{
try
{
Thread.sleep((long) Math.random() * 1000);
}
catch (InterruptedException e)
{
e.printStackTrace();
}
System.out.println("World: " + i);
}
} } static class Thread1 extends Thread
{
private Example example; public Thread1(Example example)
{
this.example = example;
} @Override
public void run()
{
example.execute();
} } static class Thread2 extends Thread
{
private Example example; public Thread2(Example example)
{
this.example = example;
} @Override
public void run()
{
example.execute2();
} }
  
当加上synchronized后,当同步方法未执行完时另一个同步方法不会执行

静态的同步方法
当一个synchronized关键字修饰的方法同时被static修饰,非静态的同步方法会将对象上锁,但是静态方法不属于对象,而是属于类,他会将这个方法所在的类的Class对象上锁。
public static void main(String[] args)
{
Example example = new Example(); Thread t1 = new Thread1(example); example = new Example();
Thread t2 = new Thread2(example); t1.start();
t2.start();
} static class Example
{
public static synchronized void execute()
{
for (int i = 0; i < 20; ++i)
{
try
{
Thread.sleep((long) Math.random() * 1000);
}
catch (InterruptedException e)
{
e.printStackTrace();
}
System.out.println("Hello: " + i);
}
} public static synchronized void execute2()
{
for (int i = 0; i < 20; ++i)
{
try
{
Thread.sleep((long) Math.random() * 1000);
}
catch (InterruptedException e)
{
e.printStackTrace();
}
System.out.println("World: " + i);
}
} } static class Thread1 extends Thread
{
private Example example; public Thread1(Example example)
{
this.example = example;
} @Override
public void run()
{
example.execute();
} } static class Thread2 extends Thread
{
private Example example; public Thread2(Example example)
{
this.example = example;
} @Override
public void run()
{
example.execute2();
} }

不加静态方法方法:

当某个synchronized方法是static的,那么当线程访问该方法时,他锁的并不是synchronized方法所在的对象,而是synchronized方法所在类的对应的class对象(java中,一个类无论有多少对象,这些对象会对应唯一的class对象,因此当线程分别访问一个类的两个对象的的两个static synchronized方法时,他们的执行顺序也是有顺序的,也就是说一个线程先去执行,执行完释放锁,另一个线程再去执行)
synchronized代码块
synchronized(object) {
}
表示的是在执行过程中会将object对象上锁(这个对象可以使任意类的对象,也可以使用this),可以自行规定上锁对象
public class test {
    public static void main(String[] args) {
        Example example = new Example();
        Thread t1 = new Thread1(example);
        Thread t2 = new Thread2(example);
        t1.start();
        t2.start();
    }
}
    class Example {
        private Object object = new Object();
        public void execute() {
            System.out.println("我开始了");
            synchronized (object) {
                for (int i = 0; i < 20; ++i) {
                    try {
                        Thread.sleep((long) Math.random() * 1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println("Hello: " + i);
                }
            }
        }
        public void execute2() {
            System.out.println("我也开始了");
            synchronized (object) {
                for (int i = 0; i < 20; ++i) {
                    try {
                        Thread.sleep((long) Math.random() * 1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println("World: " + i);
                }
            }
        }
    }
    class Thread1 extends Thread {
        private Example example;
        public Thread1(Example example) {
            this.example = example;
        }
        @Override
        public void run() {
            example.execute();
        }
    }
    class Thread2 extends Thread {
        private Example example;
        public Thread2(Example example) {
            this.example = example;
        }
        @Override
        public void run() {
            example.execute2();
        }
    }

可以看到除了synchronized代码块里的方法,其他是不会同步执行的。
synchronized方法是一种粗粒度的并发控制,某一时刻,只能有一个线程执行synchronized方法。
synchronized块则是细粒度的并发控制,只会将块中的代码同步,方法内的其他代码是可以被多个线程同时访问到的
多线程编程-- part 3 多线程同步->synchronized关键字的更多相关文章
- “全栈2019”Java多线程第十六章:同步synchronized关键字详解
		
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...
 - Java多线程同步 synchronized 关键字的使用
		
代表这个方法加锁,相当于不管哪一个线程A每次运行到这个方法时,都要检查有没有其它正在用这个方法的线程B(或者C D等),有的话要等正在使用这个方法的线程B(或者C D)运行完这个方法后再运行此线程A, ...
 - 多线程学习-基础( 九)线程同步Synchronized关键字
		
一.线程同步1.synchronized关键字的作用域有二种:(1)某个对象实例内:synchronized aMethod(){}可以防止多个线程同时访问这个对象的synchronized方法(如果 ...
 - java 多线程:Thread 并发线程: 方法同步synchronized关键字,与static的结合
		
1.方法内的变量是安全的 方法内定义的变量,每个变量对应单独的内存变量地址,多个线程之间相互不影响.多个线程之间的变量根本没有一毛钱关系 public class ThreadFuncVarSafe ...
 - Java多线程系列--“基础篇”04之 synchronized关键字
		
概要 本章,会对synchronized关键字进行介绍.涉及到的内容包括:1. synchronized原理2. synchronized基本规则3. synchronized方法 和 synchro ...
 - 2.3多线程(java学习笔记)synchronized关键字
		
一.为什么要用synchronized关键字 首先多线程中多个线程运行面临共享数据同步的问题. 多线程正常使用共享数据时需要经过以下步骤: 1.线程A从共享数据区中复制出数据副本,然后处理. 2.线程 ...
 - Java多线程编程核心技术---Java多线程技能
		
基本概念 进程是操作系统结构的基础,是一次程序的执行,是一个程序及其数据结构在处理机上顺序执行时所发生的活动,是程序在一个数据集合上运行的过程,是系统进行资源分配和调度的独立单位.线程可以理解成是在进 ...
 - Java多线程编程核心技术(三)多线程通信
		
线程是操作系统中独立的个体,但这些个体如果不经过特殊的处理就不能成为一个整体.线程间的通信就是成为整体的必用方案之一,可以说,使线程间进行通信后,系统之间的交互性会更强大,在大大提高CPU利用率的同时 ...
 - (1)Java多线程编程核心——Java多线程技能
		
1.为什么要使用多线程?多线程的优点? 提高CPU的利用率 2.什么是多线程? 3.Java实现多线程编程的两种方式? a.继承Thread类 public class MyThread01 exte ...
 
随机推荐
- Centos安装MySql、Java、Tomcat
			
一.安装MySql 安装MySql yum install -y mysql-server mysql mysql-devel 启动MySql服务 service mysqld start 为root ...
 - hibernate的反转引擎生成两个实体类的问题
			
在使用myeclipse中自带的hibernate 进行jsp开发时候遇到了这个问题.使用hibernate的反转引擎从数据库生成生成实体类,一个表生成了两个类,xx.java和xxId.java . ...
 - 怎样配置mysql_installer_community_V5.6........_setup的数据库?
			
直接访问查看如下路径就可以根据方法匹配 http://jingyan.baidu.com/album/c35dbcb0f1b1448916fcbcc7.html
 - Timer,TimerTask通过程序计数器实现的定时任务
			
1.程序计数器 程序计数器(Program Counter Register)是一块较小的内存空间,它的作用可以看 做是当前线程所执行的字节码的行号指示器.在虚拟机的概念模型里(仅是概念模型, 各种虚 ...
 - Android系统--输入系统(七)Reader_Dispatcher线程启动分析
			
Android系统--输入系统(七)Reader_Dispatcher线程启动分析 1. Reader/Dispatcher的引入 对于输入系统来说,将会创建两个线程: Reader线程(读取事件) ...
 - java面试题(一)
			
1.面向对象的特征有哪些方面? 答:面向对象的特征主要有以下几个方面: - 抽象:抽象是将一类对象的共同特征总结出来构造类的过程,包括数据抽象和行为抽象两方面.抽象只关注对象有哪些属性和行为,并不关注 ...
 - 剑指Offer面试题1
			
面试题1: 题目描述: 如下为类型CMyString的声明,请为该类型添加赋值运算符函数. class CMyString { public: CMyString(char* pData = NULL ...
 - Google Code Jam 2016 Round 1B Problem C. Technobabble
			
题目链接:https://code.google.com/codejam/contest/11254486/dashboard#s=p2 大意是教授的学生每个人在纸条上写一个自己的topic,每个to ...
 - BogoMIPS与calibrate_delay
			
在分析Arm+linux启动信息的时候.发现有一个信息竟然耗费了2s的时间,这简直是不能忍受的.这个耗时大鳄是什么东西哪,请看分析信息: [ 0.000000] console [ttyMT0] ...
 - [Git]04 如何使用标签
			
 Git也可以对某一时间点上的版本打上标签.人们在发布某个软件版本(比如 v1.0 等等)的时候,经常这么做. 本节我们一起来学习如何如何新建标签,列出所有可用的标签,以及各种不同类型标签之间的差 ...