多线程基础2-同步代码块、同步方法
来源:互联网 发布:网络理财平台 编辑:程序博客网 时间:2024/06/07 22:29
- 实现方式同步代码块
- 继承方式同步代码块
- Runnable同步方法
- 继承方式同步方法
- 单利模式线程安全
实现方式同步代码块
package safly;//使用实现Runnable接口的方式,售票/* * 此程序存在线程的安全问题:打印车票时,会出现重票、错票 * 1.线程安全问题存在的原因? * 由于一个线程在操作共享数据过程中,未执行完毕的情况下,另外的线程参与进来,导致共享数据存在了安全问题。 * * 2.如何来解决线程的安全问题? * 必须让一个线程操作共享数据完毕以后,其它线程才有机会参与共享数据的操作。 * * 3.java如何实现线程的安全:线程的同步机制 * * 方式一:同步代码块 * synchronized(同步监视器){ * //需要被同步的代码块(即为操作共享数据的代码) * } * 1.共享数据:多个线程共同操作的同一个数据(变量) * 2.同步监视器:由一个类的对象来充当。哪个线程获取此监视器,谁就执行大括号里被同步的代码。俗称:锁 * 要求:所有的线程必须共用同一把锁! * 注:在实现的方式中,考虑同步的话,可以使用this来充当锁。但是在继承的方式中,慎用this! * * 方式二:同步方法 * * */class Window2 implements Runnable { int ticket = 20;// 共享数据// Object obj = new Object(); 也可以 public void run() {// Animal a = new Animal();//局部变量不可以 while (true) { synchronized (this) {//this表示当前对象,本题中即为w if (ticket > 0) { System.out.println(Thread.currentThread().getName() + "售票,票号为:" + ticket--); }else{ break; } } } }}public class Demo { public static void main(String[] args) { Window2 w = new Window2(); Thread t1 = new Thread(w); Thread t2 = new Thread(w); Thread t3 = new Thread(w); t1.setName("窗口1"); t2.setName("窗口2"); t3.setName("窗口3"); t1.start(); t2.start(); t3.start(); }}class Animal{}
输出如下:
窗口3售票,票号为:20窗口3售票,票号为:19窗口3售票,票号为:18窗口3售票,票号为:17窗口3售票,票号为:16窗口3售票,票号为:15窗口3售票,票号为:14窗口3售票,票号为:13窗口3售票,票号为:12窗口1售票,票号为:11窗口1售票,票号为:10窗口3售票,票号为:9窗口3售票,票号为:8窗口3售票,票号为:7窗口2售票,票号为:6窗口2售票,票号为:5窗口2售票,票号为:4窗口2售票,票号为:3窗口2售票,票号为:2窗口2售票,票号为:1
继承方式同步代码块
package safly;//模拟火车站售票窗口,开启三个窗口售票,总票数为100张//存在线程的安全问题--->使用同步代码块处理。class Window3 extends Thread { static int ticket = 20; static Object obj = new Object(); public void run() { while (true) { // synchronized (this) {//在本问题中,this表示:w1,w2,w3 不能使用 synchronized (obj) { if (ticket > 0) { System.out.println(Thread.currentThread().getName() + "售票,票号为:" + ticket--); }else{ break; } } } }}public class Demo { public static void main(String[] args) { Window3 w1 = new Window3(); Window3 w2 = new Window3(); Window3 w3 = new Window3(); w1.setName("窗口1"); w2.setName("窗口2"); w3.setName("窗口3"); w1.start(); w2.start(); w3.start(); }}
输出如下:
窗口1售票,票号为:20窗口3售票,票号为:19窗口3售票,票号为:18窗口3售票,票号为:17窗口2售票,票号为:16窗口2售票,票号为:15窗口2售票,票号为:14窗口2售票,票号为:13窗口2售票,票号为:12窗口2售票,票号为:11窗口2售票,票号为:10窗口2售票,票号为:9窗口2售票,票号为:8窗口2售票,票号为:7窗口2售票,票号为:6窗口2售票,票号为:5窗口2售票,票号为:4窗口2售票,票号为:3窗口2售票,票号为:2窗口2售票,票号为:1
需要注意是synchronized (this) {//在本问题中,this表示:w1,w2,w3 不能使用
只能使用静态static Object obj = new Object(); 共同的一把锁
Runnable同步方法
package safly;//使用实现Runnable接口的方式,售票/* * 此程序存在线程的安全问题:打印车票时,会出现重票、错票 * 1.线程安全问题存在的原因? * 由于一个线程在操作共享数据过程中,未执行完毕的情况下,另外的线程参与进来,导致共享数据存在了安全问题。 * * 2.如何来解决线程的安全问题? * 必须让一个线程操作共享数据完毕以后,其它线程才有机会参与共享数据的操作。 * * 3.java如何实现线程的安全:线程的同步机制 * * 方式一:同步代码块 * synchronized(同步监视器){ * //需要被同步的代码块(即为操作共享数据的代码) * } * 1.共享数据:多个线程共同操作的同一个数据(变量) * 2.同步监视器:由一个类的对象来充当。哪个线程获取此监视器,谁就执行大括号里被同步的代码。俗称:锁 * 要求:所有的线程必须共用同一把锁! * 注:在实现的方式中,考虑同步的话,可以使用this来充当锁。但是在继承的方式中,慎用this! * * 方式二:同步方法 * 将操作共享数据的方法声明为synchronized。即此方法为同步方法,能够保证当其中一个线程执行 * 此方法时,其它线程在外等待直至此线程执行完此方法。 * >同步方法的锁:this * * 4.线程的同步的弊端:由于同一个时间只能有一个线程访问共享数据,效率变低了。 * */class Window4 implements Runnable { int ticket = 20;// 共享数据 public void run() { while (true) { show(); } } public synchronized void show() { if (ticket > 0) { System.out.println(Thread.currentThread().getName() + "售票,票号为:" + ticket--); } }}public class Demo { public static void main(String[] args) { Window4 w = new Window4(); Thread t1 = new Thread(w); Thread t2 = new Thread(w); Thread t3 = new Thread(w); t1.setName("窗口1"); t2.setName("窗口2"); t3.setName("窗口3"); t1.start(); t2.start(); t3.start(); }}
输出如下:
窗口3售票,票号为:20窗口3售票,票号为:19窗口3售票,票号为:18窗口3售票,票号为:17窗口3售票,票号为:16窗口3售票,票号为:15窗口3售票,票号为:14窗口3售票,票号为:13窗口3售票,票号为:12窗口3售票,票号为:11窗口3售票,票号为:10窗口3售票,票号为:9窗口3售票,票号为:8窗口3售票,票号为:7窗口3售票,票号为:6窗口3售票,票号为:5窗口3售票,票号为:4窗口3售票,票号为:3窗口3售票,票号为:2窗口3售票,票号为:1
继承方式同步方法
package safly;//模拟火车站售票窗口,开启三个窗口售票,总票数为100张//存在线程的安全问题--->使用同步代码块处理。class Window3 extends Thread { static int ticket = 20; public void run() { while (true) { show(); } } public synchronized void show() {// this充当的锁 if (ticket > 0) { System.out.println(Thread.currentThread().getName() + "售票,票号为:" + ticket--); } }}public class Demo { public static void main(String[] args) { Window3 w1 = new Window3(); Window3 w2 = new Window3(); Window3 w3 = new Window3(); w1.setName("窗口1"); w2.setName("窗口2"); w3.setName("窗口3"); w1.start(); w2.start(); w3.start(); }}
输出的代码跟上面一样~~
单利模式线程安全
package safly;//关于懒汉式的线程安全问题:使用同步机制//对于一般的方法内,使用同步代码块,可以考虑使用this。//对于静态方法而言,使用当前类本身充当锁。class Singleton { private Singleton() { } private static Singleton instance = null; public static Singleton getInstance() { if (instance == null) { synchronized (Singleton.class) { if (instance == null) { instance = new Singleton(); } } } return instance; }}public class Demo { public static void main(String[] args) { Singleton s1 = Singleton.getInstance(); Singleton s2 = Singleton.getInstance(); System.out.println(s1 == s2); // Class clazz = Singleton.class; }}
阅读全文
0 0
- 多线程基础2-同步代码块、同步方法
- Java_基础—多线程(同步代码块和同步方法)
- 多线程基础之同步代码块
- 多线程 同步代码块
- 多线程、同步代码块
- 多线程-同步代码块
- 多线程14__Java同步代码块,同步方法
- java多线程——同步方法和同步代码块
- 多线程---java同步方法和同步代码块的区别
- 多线程三(同步代码块)
- 多线程安全问题----同步代码块
- 同步代码块,同步代码方法,锁
- JAVA基础 day11 多线程 同步代码块 死锁问题
- 同步代码块2
- java同步方法和同步代码块
- Synchronized 同步方法和同步代码块
- 同步方法和同步代码块
- 锁,同步代码块,同步方法 区别
- Linux内核之内存寻址
- OpenGL学习系列导航
- 莫让懒惰蹉跎了青春
- 微信公众号实现简易的物联网控制(一)
- SIM800C EAT开发
- 多线程基础2-同步代码块、同步方法
- JAVA-日期类
- 临时对象
- ajax 调用后端传递的map类型时
- uvalive6837 There is No Alternative(暴力+kruskal)
- CentOS下配置jdk
- 容器部署解决方案Docker
- OpenGL基础知识
- nodejs在window下安装以及配置淘宝镜像