Java笔记(十九)……多线程
概述
进程:
是一个正在执行中的程序
每一个进程执行都有一个执行顺序,该执行顺序是一个执行路径,或者叫一个控制单元
线程:
就是进程中的一个独立的控制单元,线程在控制着进程的执行
一个进程中至少有一个线程
Java JVM启动的时候会有一个进程java.exe,该进程中至少有一个线程负责Java程序的执行,而且这个线程运行的代码存在于main方法中,该线程称之为主线程,与C类似,java.exe相当于所有进程的父进程,而且JVM启动了不止一个线程,还有负责垃圾回收的线程
创建线程
第一种创建方式:继承Thread类
1: class Sell extends Thread
2: {
3: private int tickets = 100;
4: Sell()
5: {
6: //启动线程
7: start();
8: }
9: //复写Thread类中的run方法,将线程运行所需的代码写到run方法中
10: public void run()
11: {
12: while(tickets > 0)
13: {
14: System.out.println("tickets = "+tickets--);
15: }
16: }
17: }
第二种创建方式:实现Runnable接口
1: class Sell implements Runnable
2: {
3: private int tickets = 100;
4:
5: //实现Runnable接口的run方法
6: public void run()
7: {
8: while(tickets > 0)
9: {
10: System.out.println("tickets = "+tickets--);
11: }
12: }
13: }
14: class ThreadDemo
15: {
16: public static void main(String[] args)
17: {
18: //将Runnable接口的子类Sell传递给Thread的构造函数,并启动线程
19: new Thread(new Sell()).start();
20: }
21: }
实现方式与继承方式的区别
其实Thread类同样也是实现了Runnable接口,所以我们要的只是run方法,那么我们只要实现Runnable接口即可,这样避免了单继承的局限性,我们在实现Runnable接口的同时还可以继承其他的类,扩展了功能
线程权限问题

线程安全问题--同步
当多条语句在操作同一个线程共享数据时,一个线程对多条语句只执行了一部分,还没有执行完,另一个线程参与进来执行,导致共享数据的错误
1:
2: class Sell implements Runnable
3: {
4: private int tickets = 10;
5:
6: //实现Runnable接口的run方法
7: public void run()
8: {
9: while(tickets > 0)
10: {
11: try{Thread.sleep(500);}catch(Exception e){}
12: System.out.println(Thread.currentThread().getName()+"..tickets = "+(--tickets));
13: }
14: }
15: }

当我们让线程操作共享数据时,暂停一会,可以发现,每个线程很容易-1剩余票数的错误
解决方法
对多条操作共享数据的语句,只能让一个线程都执行完,在执行过程中,其他线程不能参与进来执行
Java对多线程的安全问题提供了专业的解决方式,就是同步代码块或者同步函数
synchronized(对象)
{
需要被同步的代码
}
synchronized void show()
{
函数内容
}
1: class Sell implements Runnable
2: {
3: private int tickets = 10;
4:
5: //实现Runnable接口的run方法
6: public void run()
7: {
8: //加入锁,使代码同步
9: synchronized(this)
10: {
11: while(tickets > 0)
12: {
13: try{Thread.sleep(500);}catch(Exception e){}
14: System.out.println(Thread.currentThread().getName()+"..tickets = "+(--tickets));
15: }
16: }
17: }
18: }

我们可以看到,即使中途有睡眠过程,也不再出现错误票数,当然这个例子有一些问题,虽然保证了安全,却只能让一个线程完成操作,解决方法很简单,在while循环内设置判断语句,再加锁即可
对象如同锁,持有锁的线程可以在同步中执行
没有持有锁的线程即使获取了cpu的执行权,也无法执行同步代码
如果同步函数被静态修饰后,如何使用锁?
通过验证,发现不再是this,因为静态方法中也不可以定义this
静态进入内存时,内存中没有本类对象,但是一定有该类的对应的字节码文件对象 类名.class,该对象的类型是Class
所以静态的同步方法是,使用该方法所在类的字节码对象作为锁,即类名.class
1: class Tickets
2: {
3: static int tickets = 10;
4: public static void sell()
5: {
6: //类对象作为锁
7: synchronized(Selling.class)
8: {
9: if(tickets > 0)
10: {
11: try{Thread.sleep(500);}catch(Exception e){}
12: System.out.println(Thread.currentThread().getName()+"..tickets = "+(--tickets));
13: }
14: }
15: }
16: }
17:
18: class Selling implements Runnable
19: {
20: //实现Runnable接口的run方法
21: public void run()
22: {
23: while(Tickets.tickets> 0)
24: {
25: Tickets.sell();
26: }
27: }
28: }
同步的前提:
- 必须要有两个或两个以上的线程
- 必须要多个线程使用同一个锁
同步的利弊
好处:解决了多线程的安全问题
弊端:多个线程需要判断锁,较为消耗资源
加入同步之后线程的状态如下

如何编写多线程
- 明确哪些代码是多线程运行代码
- 明确共享数据
- 明确多线程运行代码中哪些语句是操作共享数据的
死锁
死锁其实就是同步中嵌套同步
1:
2: class A implements Runnable
3: {
4: String a = "a";
5:
6: public void run()
7: {
8: while(true)
9: {
10: //A抢占A锁
11: synchronized(A.class)
12: {
13: System.out.println("A get 1");
14: //A抢占B锁
15: synchronized(B.class)
16: {
17: System.out.println("A get 2");
18: System.out.println("A:"+a);
19: }
20: }
21: }
22: }
23: }
24:
25: class B implements Runnable
26: {
27: String b = "b";
28:
29: public void run()
30: {
31: while(true)
32: {
33: //B抢占B锁
34: synchronized(B.class)
35: {
36: System.out.println("B get 1");
37: //B抢占A锁
38: synchronized(A.class)
39: {
40: System.out.println("B get 2");
41: System.out.println("B:"+b);
42: }
43: }
44: }
45: }
46: }
47: class DeadlockDemo
48: {
49: public static void main(String[] args)
50: {
51: new Thread(new A()).start();
52: new Thread(new B()).start();
53: }
54: }
结果如下,A和B都在等待对方释放资源(对应的锁)

Java笔记(十九)……多线程的更多相关文章
- python3.4学习笔记(十九) 同一台机器同时安装 python2.7 和 python3.4的解决方法
python3.4学习笔记(十九) 同一台机器同时安装 python2.7 和 python3.4的解决方法 同一台机器同时安装 python2.7 和 python3.4不会冲突.安装在不同目录,然 ...
- “全栈2019”Java第九十九章:局部内部类与继承详解
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- “全栈2019”Java第二十九章:数组详解(中篇)
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- “全栈2019”Java第十九章:关系运算符、条件运算符和三元运算符
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- (C/C++学习笔记) 十九. 模板
十九. 模板 ● 模板的基本概念 模板(template) 函数模板:可以用来创建一个通用功能的函数,以支持多种不同形参,进一步简化重载函数的函数体设计. 语法: template <<模 ...
- Java基础学习笔记十九 IO
File IO概述 回想之前写过的程序,数据都是在内存中,一旦程序运行结束,这些数据都没有了,等下次再想使用这些数据,可是已经没有了.那怎么办呢?能不能把运算完的数据都保存下来,下次程序启动的时候,再 ...
- Java基础学习笔记十九 File
IO概述 回想之前写过的程序,数据都是在内存中,一旦程序运行结束,这些数据都没有了,等下次再想使用这些数据,可是已经没有了.那怎么办呢?能不能把运算完的数据都保存下来,下次程序启动的时候,再把这些数据 ...
- Java学习笔记十九:Java中的访问控制修饰符
Java中的访问控制修饰符 一:Java修饰符的种类: 访问修饰符 非访问修饰符 修饰符用来定义类.方法或者变量,通常放在语句的最前端.我们通过下面的例子来说明: public class Hello ...
- swift 笔记 (十九) ——
协议
协议(Protocols) 协议仅是用定义某些任务或者是功能必须的方法和属性. 类似于java里的interface的作用.但协议并不会实现详细的功能. 我猜这个名字源于OO中提到的"契约& ...
随机推荐
- python3编码问题
继续收集python3编码问题相关资料 资料来源 鹏程的新浪博客(转载)http://blog.sina.com.cn/s/blog_6d7cf9e50102vo90.html 这篇鹏程老师写的关 ...
- 工作流(Workflow)学习---基础知识整理
工作流定义: 工作流是将一组任务组织起来以完成某个经营过程:定义了任务的触发顺序和触发条件,每个任务可以由一个或多个软件系统完成,也可以由一个或一组人完成,还可以由一个或多个人与软件系统协作完成. 工 ...
- 基于BOOST 实现并发服务器框架
一:设计思路 本服务器框架使用 UDP 传输协议,程序柱线程等待客户端数据,并将数组存取队列缓冲区.另外可开启多个工作线程,工作线程可以依据具体项目实现不同的功能 ,例如可以将队列缓冲区中的数据逐个取 ...
- 使用grub硬盘重装ubuntu
电脑一直是双系统,Windows8 + Ubuntu13.04,昨天瞎搞,更新了一下QT,结果就不能开机了...真心难. 老系统坏了也好,正好可以装个Ubuntu14.04LTS.手边没有U盘,可以把 ...
- Static File Middleware
[ASP.NET Core] Static File Middleware 前言 本篇文章介绍ASP.NET Core里,用来处理静态档案的Middleware,为自己留个纪录也希望能帮助到有需要 ...
- CentOS搭建OpenVPN服务(简易版)
OpenVPN服务端配置 1. 安装OpenVPN软件包 默认的Centos软件源里面没有OpenVPN的软件包,我们可以添加rpmforge的repo,从而实现yum安装openvpn 针对Cent ...
- Bengio最新博文:深度学习展望
Bengio最新博文:深度学习展望 人类一直梦想着创造有智能的机器.早在第一台可编程计算机问世前100多年,发明家就对如何能让由连杆和齿轮组成的设备也变得更加智能这一命题充满好奇.后来,20世纪40年 ...
- java 使用正则表达式从网页上提取网站标题
如何从网页上抓取有价值的东西?看懂了下面的程序(非常简单),想从网页上抓取什么信息(标题.内容.Email.价格等)就能抓取什么信息. package catchhtml; import java.i ...
- ZOJ 3609 Modular Inverse
点我看题目 题意 : 这个题是求逆元的,怎么说呢,题目看着很别扭....就是给你a和m,让你求一个最小的x满足a-1≡x (mod m).或者ax≡1 (mod m).通俗点说呢,就是找一个最小的x, ...
- HDU4535+公式
错排公式. 用64位! /* */ #include<stdio.h> #include<string.h> #include<stdlib.h> #include ...