编程开发之--java多线程学习总结(1)问题引入与概念叙述
1、经典问题,火车站售票,公共票源箱,多个窗口同时取箱中车票销售
package com.lfy.ThreadsSynchronize; /**
* 解决办法分析:即我们不能同时让超过两个以上的线程进入到 if(num>0)的代码块中
* 1、使用 同步代码块 2、使用 同步方法 3、使用 锁机制
*/
public class TicketSell1 extends Thread{ //定义一共有 50 张票,注意声明为 static,表示几个窗口共享
private static int num = 50; //调用父类构造方法,给线程命名
public TicketSell1(String string) {
super(string);
} @Override
public void run() {
//票分 50 次卖完
for(int i = 0 ; i < 50 ;i ++){
if(num > 0){
try {
Thread.sleep(10);//模拟卖票需要一定的时间
} catch (InterruptedException e) {
// 由于父类的 run()方法没有抛出任何异常,根据继承的原则,子类抛出的异常不能大于父类, 故我们这里也不能抛出异常
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"卖出一张票,剩余"+(--num)+"张");
}
}
}
}
2、对于解决办法,java中专门提供了synchronized关键字处理多线程同步问题,有了synchronized关键字,多线程程序的运行结果将变得可以控制。synchronized关键字用于保护共享数据。synchronized实现同步的机制:synchronized依靠"锁"机制进行多线程同步,"锁"有2种,一种是对象锁,一种是类锁。synchronized关键字修饰普通方法时,获得的锁是对象锁,也就是this。而修饰静态方法时,锁是类锁,也就是类名.class。
阻塞:A、B线程同时运行,由于锁的控制,某时刻A线程还能继续执行,B线程被挂起等待了,就说B线程被阻塞了。
(1)对象锁,使用synchronized修饰多个普通方法,当不同线程调用同一个对象的不同被synchronized修饰过的方法时,第一个调用被synchronized修饰过的方法的线程会得到对象锁,其他线程处于阻塞状态,直至第一个得到锁的线程退出被synchronized修饰过的方法。举个例子:
package lfy;
public class TestSynchronized {
public synchronized void method1() throws InterruptedException {
System.out.println("method1 begin at:" + System.currentTimeMillis());
System.out.println("method1 begin to sleep 5s");
Thread.sleep(5000);
System.out.println("method1 end at:" + System.currentTimeMillis());
}
public synchronized void method2() throws InterruptedException {
for(int i=0;i<5;i++) {
System.out.println("method2 running");
Thread.sleep(200);
}
}
static TestSynchronized instance = new TestSynchronized();
public static void main(String[] args) {
Thread thread1 = new Thread(new Runnable() {
@Override
public void run() {
try {
instance.method1();
} catch (InterruptedException e) {
e.printStackTrace();
}
for(int i=1; i<4; i++) {
try {
Thread.sleep(200);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("Thread1 still alive");
}
System.out.println("Thread1 over");
}
});
Thread thread2 = new Thread(new Runnable() {
@Override
public void run() {
try {
instance.method2();
System.out.println("Thread2 over");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
thread1.start();
thread2.start();
}
}
运行结果:

解释:创建了一个对象实例instance,创建两个线程Thread1、Thread2,它们分别调用instance的mothod1、mothod2方法,由于线程1先启动并先访问到被synchronized修饰的mothod1,此时instance对mothod2上锁,线程2此时只能等待mothod2的锁被释放,才能执行mothod2方法。这就是对象锁机制。
(2)类锁,对所有对象调用被synchronized修饰的static方法进行锁定,没有被synchronized修饰的static方法不会被上锁。
package lfy;
public class TestSynchronized {
public synchronized static void method1() throws InterruptedException {
System.out.println("method1 begin at:" + System.currentTimeMillis());
System.out.println("method1 begin to sleep 5s");
Thread.sleep(5000);
System.out.println("method1 end at:" + System.currentTimeMillis());
}
public synchronized static void method2() throws InterruptedException {
for(int i=0;i<5;i++) {
System.out.println("method2 running");
Thread.sleep(200);
}
}
/**
* 没有static、synchronized修饰的普通方法
* @throws InterruptedException
*/
public void method3() throws InterruptedException {
System.out.println("method3 running");
Thread.sleep(200);
}
/**
* 只有synchronized修饰的普通方法
* @throws InterruptedException
*/
public synchronized void method4() throws InterruptedException {
System.out.println("method4 running");
Thread.sleep(200);
}
static TestSynchronized instance1 = new TestSynchronized();
static TestSynchronized instance2 = new TestSynchronized();
static TestSynchronized instance3 = new TestSynchronized();
static TestSynchronized instance4 = new TestSynchronized();
public static void main(String[] args) {
Thread thread1 = new Thread(new Runnable() {
@Override
public void run() {
try {
instance1.method1();
//TestSynchronized.mothod1();
} catch (InterruptedException e) {
e.printStackTrace();
}
for(int i=1; i<4; i++) {
try {
Thread.sleep(200);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("Thread1 still alive");
}
System.out.println("Thread1 over");
}
});
Thread thread2 = new Thread(new Runnable() {
@Override
public void run() {
try {
instance2.method2();
//TestSynchronized.mothod2();
System.out.println("Thread2 over");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
Thread thread3 = new Thread(new Runnable() {
@Override
public void run() {
try {
instance3.method3();
System.out.println("method3 over");
Thread.sleep(10000);
System.out.println("Thread3 still alive");
System.out.println("Thread3 now to over");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
Thread thread4 = new Thread(new Runnable() {
@Override
public void run() {
try {
instance4.method4();
System.out.println("method4 over");
Thread.sleep(10000);
System.out.println("Thread4 still alive");
System.out.println("Thread4 now to over");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
thread1.start();
thread2.start();
thread3.start();
thread4.start();
}
}
运行结果:

编程开发之--java多线程学习总结(1)问题引入与概念叙述的更多相关文章
- 编程开发之--java多线程学习总结(5)
4.对继承自Runnable的线程进行锁机制的使用 package com.lfy.ThreadsSynchronize; import java.util.concurrent.locks.Lock ...
- 编程开发之--java多线程学习总结(4)
3.使用锁机制lock,unlock package com.lfy.ThreadsSynchronize; import java.util.concurrent.locks.Lock; impor ...
- 编程开发之--java多线程学习总结(2)同步代码块
1.第一种解决办法:同步代码块,关键字synchronized package com.lfy.ThreadsSynchronize; /** * 1.使用同步代码块 * 语法: synchroniz ...
- 编程开发之--java多线程学习总结(6)
5.测试 package com.lfy.ThreadsSynchronize; public class Test { public static void main(String[] args) ...
- 编程开发之--java多线程学习总结(3)类锁
2.使用方法同步 package com.lfy.ThreadsSynchronize; /** * 1.使用同步方法 * 语法:即用 synchronized 关键字修饰方法(注意是在1个对象中用锁 ...
- Java多线程学习(转载)
Java多线程学习(转载) 时间:2015-03-14 13:53:14 阅读:137413 评论:4 收藏:3 [点我收藏+] 转载 :http://blog ...
- Java多线程学习(三)volatile关键字
转载请备注地址:https://blog.csdn.net/qq_34337272/article/details/79680693 系列文章传送门: Java多线程学习(一)Java多线程入门 Ja ...
- Java多线程学习(一)Java多线程入门
转载请备注地址:https://blog.csdn.net/qq_34337272/article/details/79640870 系列文章传送门: Java多线程学习(一)Java多线程入门 Ja ...
- 转:Java多线程学习(总结很详细!!!)
Java多线程学习(总结很详细!!!) 此文只能说是java多线程的一个入门,其实Java里头线程完全可以写一本书了,但是如果最基本的你都学掌握好,又怎么能更上一个台阶呢? 本文主要讲java中多线程 ...
随机推荐
- python socket编程入门(编写server实例)-乾颐堂
python 编写server的步骤: 1. 第一步是创建socket对象.调用socket构造函数.如: socket = socket.socket( family, type ) family参 ...
- 不要怂,就是GAN (生成式对抗网络) (五):无约束条件的 GAN 代码与网络的 Graph
GAN 这个领域发展太快,日新月异,各种 GAN 层出不穷,前几天看到一篇关于 Wasserstein GAN 的文章,讲的很好,在此把它分享出来一起学习:https://zhuanlan.zhihu ...
- python进行数据清理之pandas中的drop用法
好久好久没有更新博客了,之前自学的估计也都忘记差不多了.由于毕业选择从事的行业与自己的兴趣爱好完全两条路,心情也难过了很久,既然入职了就要好好干,仍要保持自己的兴趣,利用业余时间重拾之前的乐趣. 从基 ...
- Mybatis的select查询的三种方式
1.首先建立一个测试的dao public interface IStudentDao { // 根据姓名查询 List<Student> selectStudentsByName(Str ...
- nohup、&、setsid、fork和fg、bg究竟有啥区别?
目录 目录 1 1. 名词解释 1 2. 什么是守护进程? 2 2.1. 后台运行 2 2.2. 解释关系 2 3. 以"&"方式运行有何问题? 2 4. fork 2 5 ...
- Linux 基础教程 43-su和sudo命令
在使用Linux系统中,有时候还需要做身份切换,这是为什么? 使用普通账号:系统日常操作的好习惯 虽然使用root对系统进行各种操作不受权限等方面的限制,但却存在重大的安全隐患,假如有人不 ...
- HRBUST1315 火影忍者之~大战之后 2017-03-06 16:14 54人阅读 评论(0) 收藏
火影忍者之-大战之后 经历了大战的木叶村现在急需重建,人手又少,所以需要尽可能多的接受外来的任务,以赚取报酬,重建村庄,假设你现在是木叶的一名高级忍者,有一大堆的任务等着你来做,但毕竟个人时间有限,所 ...
- 深海划水队项目--七天冲刺之day6
站立式会议:由于有位项目组成员回家了,所以由微信群在线讨论代替. 昨天已完成的任务:界面优化,实现方块的移动,旋转和下降. 今天已完成的任务:设置游戏按键,检查重合.检查是否超出边界.检查是否可以下落 ...
- (1)-使用json所要用到的jar包下载
json是个非常重要的数据结构,在web开发中应用十分广泛.我觉得每个人都应该好好的去研究一下json的底层实现,需要准备下面的六个jar包:commons-lang-1.0.4.jar common ...
- 微信公众开发URL和token填写详解
微信公众开发URL和token填写详解 方法/步骤 作为一名微信公众号开发者,别人进入你的微信公众号,肯定会看见某些网页,或者给你发某些信息,你需要实时自动回复,所以你需要一个24小时为用户服 ...