1.
多线程基础
1.1 认识进程和线程
1.1.1 什么是进程
1.1.1.1 进程是正在进项的程序,是资源分配的一个基本单位,有内存分配;
1.1.2 什么是线程
1.1.2.1 线程是进程的一个执行单位,也是进程的执行顺序;
1.1.2.2 一个进程至少有一个线程,可以由两个或以上的线程;
1.1.3 JVM至少有几个线程
1.1.3.1 至少有一个或两个线程,main方法和垃圾回收线程;
1.1.4 什么是多线程
1.1.4.1 2个或以上的线程去执行
1.1.5 多线程的作用
1.1.5.1 提高效率,线程之间切换的小号是可以接受的;
1.1.5.2 单线程容易阻塞;
1.1.5.3 多个线程同时执行代码;
1.1.5.4 一个线程挂了,还有其他的线程在,程序还在执行
1.1.6 进程和线程是谁创建的
都是操作系统创建的
1.1.7 如何自定义一个线程
1.1.7.1 继承Thread类,重写run方法
1.1.7.2 生成对象
1.1.7.3 开启start
1.1.8 所有实现的接口
1.1.8.1 Runnable接口
1.1.8.2 new线程对象的时候,创建了线程
1.1.8.3 start方法,沟通了操作系统,告诉操作系统有一个线程要开启,也调用了run
1.2 代码示例
1.2.1 在main主线程里,再度开启其他线程
1.2.1.1 测试类:
public class Test {
public static void main(String[] args) throws ParseException {
Test001 t = new Test001();//这里代表着产生了一个线程
Test001 t1 = new Test001();//这里代表着产生了一个线程
System.out.println("开启线程之前");
//t1.run();//代表着调用了对象的方法
t.start();//这里代表开启了一个线程,是沟通了操作系统
t1.start();//这里代表开启了一个线程,是沟通了操作系统
try {
Thread.sleep(15);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("开启线程之后");
System.out.println("main over");
}
}
1.2.1.2 重写run方法类,继承Thread类:
class Test001 extends Thread{
@Override
public void run() {
for (int i = 0; i < 100; i++) {
try {
sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("run .. "+i);
}
}
}
1.2.2 创建一个线程,和主程序交替执行
1.2.2.1 测试类
public class ThreadTest {
public static void main(String[] args) {
Test t = new Test();
t.start();
for (int i = 0; i < 10; i++) {
System.out.println("main>>>"+i);
}
System.out.println("main - gg");
}
}
1.2.2.2 方法类
class Test extends Thread{
int i = 0;
@Override
public void run() {
while(true){
if(i<10){
System.out.println("run方法..." + i++);
}else{
System.out.println("run - OVER");
break;
}
}
}
}
1.2.3 模拟买票功能
1.2.3.1 测试类
public class TestRunnable01 {
public static void main(String[] args) {
Runnable x1 = new XinXiaoMi();
Thread t1 = new Thread(x1);
Thread t2 = new Thread(x1);
Thread t3 = new Thread(x1);
t1.setName("小红");
t2.setName("小蓝");
t3.setName("小绿");
t1.start();
t2.start();
t3.start();
}
}
1.2.3.2 方法类
class XinXiaoMi implements Runnable{
static int i =100;//共享的
Object o = new Object();
@Override
public void run() {
boolean b = true;
while(b){
synchronized (o) {
if(i>0){//i = 1
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"...卖出了手机,编号是..." + i--);
}
else{
break;
}
}
}
}
}
1.2.4 小明和老婆同时存钱,每次存取500元,每人存5次
1.2.4.1 测试类:
public class TestRunnable02 {
public static void main(String[] args) {
test t = new test();
Thread t1 = new Thread(t);
Thread t2 = new Thread(t);
t1.setName("小明");
t2.setName("陈学冬");
t1.start();
t2.start();
}
}
1.2.4.2 方法类:
class test implements Runnable{
int con = 0;//初始定义了账户余额为0元
Object o = new Object();
@Override
public void run() {
synchronized (o) {
for (int i = 0; i < 5; i++) {
con += 500;
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+" 存了500大洋---此时账户余额为 。。"+con);
}
}
}
}
1.2.4 小明和老婆同时存钱,每次存取500元,每人存5次; 同时实现方法的同步
1.2.4.1 测试类
public class TestM {
public static void main(String[] args) {
test00 t = new test00();
Thread t1 = new Thread(t);
Thread t2 = new Thread(t);
t1.setName("小明");
t2.setName("baby");
t2.start();
t1.start();
}
}
1.2.4.2 方法类
class test00 implements Runnable{
int con = 0;//初始定义了账户余额为0元
Object o = new Object();
@Override
public void run() {
System.out.println(Thread.currentThread().getName()+" 到达房间门口");
this.show();//一次就好,许你地老天荒
}
public synchronized void show(){
System.out.println(Thread.currentThread().getName()+" 锁住了房间");
for (int i = 0; i < 5; i++) {
con += 500;
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+" 存入了500,此时余额是 "+con);
}
System.out.println(Thread.currentThread().getName()+" 离开了房间,打开了锁");
}
}
1.2.5 小明和老婆同时存钱,每次存取500元,每人存5次; 同时实现方法的同步;同时实现交替存储
1.2.5.1 测试类
public class TestM2 {
public static void main(String[] args) {
test000 t = new test000();
Thread t1 = new Thread(t);
Thread t2 = new Thread(t);
t1.setName("小明");
t2.setName("baby");
t2.start();
t1.start();
}
}
1.2.5.2 方法类
class test000 implements Runnable{
int con = 0;//初始定义了账户余额为0元
Object o = new Object();
@Override
public void run() {
for (int i = 0; i < 5; i++) {
//等待区
show();
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public synchronized void show(){
System.out.println(Thread.currentThread().getName()+" 上锁了");
con += 500;
System.out.println(Thread.currentThread().getName()+" 存入了 500 , 此时余额为 "+con);
System.out.println(Thread.currentThread().getName()+" 锁开了");
}
}
1.2.6 死锁的示例
1.2.6.1 测试类:
/* Runnablere =new Runnable() {
public void run() {
}
};
*/
public class Test01 {
public static void main(String[] args) {
TestSY ts1 = new TestSY();
Thread th1 = new Thread(ts1);
Thread th2= new Thread(ts1);
th1.start();
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
TestSY.b = false;
th2.start();
}
}
1.2.6.2 方法类:
class TestSY implements Runnable{
static boolean b = true;
Object o = new Object();
Object o1 = new Object();
@Override
public void run() {
if( b ){
synchronized (o) {
System.out.println("if 拿到了 o 的锁");
try {
Thread.sleep(10000);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (o1) {
System.out.println("if-----");
}
}
}else{
synchronized (o1) {
System.out.println("else 拿到了 o1 的锁");
try {
Thread.sleep(10000);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (o) {
System.out.println("else-----");
}
}
}
}
}
1.3 cpu调度
cpu的时间片,cpu控制着执行;
cpu会自行分配资源给线程,得到资源的线程可以执行,没有得到资源的线程没有办法执行;
多个cpu才是真正意义上的同时执行多线程,因为运行速度特别多,会给我们产生错觉,认为是同时执行的;
1.4 线程的状态
1.4.1 新生
1.4.1.1 new对象的时候代表着新生;
1.4.2 就绪
1.4.2.1 start方法代表着就绪,这个时候在等待着CPU的资源;如果有资源就进入执行状态,没有资源就继续等待;
1.4.3 执行
1.4.3.1 拿到时间片,正在执行;时间片被拿走了就离开了执行状态;
1.4.4 阻塞
1.4.4.1 睡觉sleep,让资源;时间结束了就回到就绪状态,等待资源;
1.4.5 死亡
1.4.5.1 线程执行结束;非正常死亡方式;
1.5 两种创建线程方式的比较
1.5.1 继承方式创建简单,使用的方法也多;
1.5.2 java是单继承的,使用集成方式过于死板;
1.5.3 继承是不共享资源的,实现方式是共享的;
1.5.4 主要由于单继承的机制,推荐使用Runnable接口实现这种方式
1.5.4.1 实现接口,重写方法
1.5.4.2 new一个对象
1.5.4.3 把new的对象放进线程(Thread)对象里
1.5.4.4 用线程对象开启线程
1.5.4.5 注意:run方法不能抛异常
1.6 同步代码块原理解析
1.6.1 java提供了锁的机制synchronized ,可以锁住一段代码,在任何时期这段代码里面最多只有一个线程在执行。每个要进来的线程都会先判断此处有没有上锁。
1.6.1.1 如果没有,就进去并且上锁,再执行完之后才会离开,同时打开锁,其他的线程才可以进来
1.6.1.2 如果被锁,再门口等待,进不去; 这里锁,锁的是对象。具体锁那个对象呢。具体情况具体分析
1.6.2 同步的优缺点
1.6.2.1 同步的前提
1.6.2.1.1 两个或两个以上的线程去操作共享数据时
1.6.2.2 同步的好处
1.6.2.2.1 保证了线程的安全性
1.6.2.3 同步的缺陷
1.6.2.3.1 消耗增加,但是影响不大
1.6.2.3.2 使用不当容易造成死锁
===========================================================================================
1.7 线程之间的通讯
1.7.1 生产者消费者阻塞队列
1.7.1.1 阻塞队列(BlockingQueue)
1.7.1.1.1 栈和队列是在程序中被广泛使用的两种线性数据结构
1.7.1.1.2 Java5提供了阻塞队列的接口BlockingQueue,阻塞队列的概念是:一个指定长度的队列,如果队列满了,添加新元素的操作会被阻塞等待,直到有空位为止;同时,当队列为空的时候,请求队列元素的操作同样会阻塞等待,直到有可用元素为止;
1.7.1.1.3 Java阻塞队列应用于生产者消费者模式、消息传递、并行任务执行和相关并发设计的大多数常见使用上下文; 程序的两个线程通过交替向BlockingQueue中放入元素、取出元素,即可很好的控制线程的通信;
1.7.1.1.4 BlockingQueue提供了两个支持阻塞的方法:
1.7.1.1.4.1 put(E e), 尝试把E元素放入BlockingQueue中,如果该队列的元素已满,则阻塞该线程;
1.7.1.1.4.1 take( ), 尝试从BlockingQueue的头部取出元素,如果该队列的元素为空,则阻塞该线程;
1.7.1.1.5