Day09
Java.Thread
1.定义
进程是具有一定独立功能的程序关于某个数据集合上的一次运行活动,进程是系统进行资源分配和调度的一个独立单位.
线程是进程的一个实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位.线程自己基本上不拥有系统资源,只拥有一点在运行中必不可少的资源(如程序计数器,一组寄存器和栈),但是它可与同属一个进程的其他的线程共享进程所拥有的全部资源.
2.关系 一个线程可以创建和撤销另一个线程;同一个进程中的多个线程之间可以并发执行.
相对进程而言,线程是一个更加接近于执行体的概念,它可以与同进程中的其他线程共享数据,但拥有自己的栈空间,拥有独立的执行序列。
3.区别 进程和线程的主要差别在于它们是不同的操作系统资源管理方式。进程有独立的地址空间,一个进程崩溃后,在保护模式下不会对其它进程产生影响,而线程只是一个进程中的不同执行路径。线程有自己的堆栈和局部变量,但线程之间没有单独的地址空间,一个线程死掉就等于整个进程死掉,所以多进程的程序要比多线程的程序健壮,但在进程切换时,耗费资源较大,效率要差一些。但对于一些要求同时进行并且又要共享某些变量的并发操作,只能用线程,不能用进程。
根本区别:进程是操作系统资源分配的基本单位,而线程是任务调度和执行的基本单位
1) 简而言之,一个程序至少有一个进程,一个进程至少有一个线程.
2) 线程的划分尺度小于进程,使得多线程程序的并发性高。
3) 另外,进程在执行过程中拥有独立的内存单元,而多个线程共享内存,从而极大地提高了程序的运行效率。
4) 线程在执行过程中与进程还是有区别的。每个独立的线程有一个程序运行的入口、顺序执行序列和程序的出口。但是线程不能够独立执行,必须依存在应用程序中,由应用程序提供多个线程执行控制。
5) 从逻辑角度来看,多线程的意义在于一个应用程序中,有多个执行部分可以同时执行。但操作系统并没有将多个线程看做多个独立的应用,来实现进程的调度和管理以及资源分配。这就是进程和线程的重要区别。
4.优缺点 线程和进程在使用上各有优缺点:线程执行开销小,但不利于资源的管理和保护;而进程正相反。同时,线程适合于在SMP机器上运行,而进程则可以跨机器迁移。
线程创建
Thread class(重点)
-
自定义线程类继承Thread
-
重写 run() 方法,编写线程执行体
-
创建线程对象,调用 start() 方法启动线程
package demo01;
public class TestThread extends Thread{
网络图片下载练习
package demo01;
import org.apache.commons.io.FileUtils;
import java.io.File;
import java.io.IOException;
import java.net.URL;
public class TestThread02 extends Thread{
private String url;
private String name;
public TestThread02(String url,String name){
this.name=name;
this.url=url;
}
//下载图片线程的执行体
Runnable 接口(重点)
package demo01;
//实现runnable接口,重写run方法,执行线程需要丢入runnable接口实现类,调用start方法
public class TestThread03 implements Runnable{
并发问题
package demo01;
import oop.demo07.Test;
//多线程同时操作同一个对象
public class TestThread04 implements Runnable{
private int ticketNums = 10;
龟兔赛跑练习
package demo01;
public class Race implements Runnable{
private static String winner;
静态代理
package demo01;
//静态代理模式
//真实对象和代理对象都要实现同一个接口
public class StaticProxy {
public static void main(String[] args) {
WeddingCompany weddingCompany = new WeddingCompany(new You());
weddingCompany.HappyMarry();
}
}
interface Marry{
void HappyMarry();
}
//真实角色,你去结婚
class You implements Marry{
lamda表达式
其实质属于函数式编程的概念
package demo01;
//推导lamda表达式
public class Lamda {
//3.静态内部类
static class Like2 implements ILike{
练习
package demo01;
public class Lamda02 {
public static void main(String[] args) {
ILove love =(int a)->{
System.out.println("i love you--->"+a);
};
love.love(520);
//简化版本
love = a->System.out.println("i love you--->"+a);
love.love(521);
//只有一行代码的情况下,才可简化花括号
//前提是接口必须是函数接口
}
}
interface ILove{
void love(int a);
}
线程停止
package demo01;
import java.sql.SQLOutput;
public class TestStop implements Runnable{
//1.设置一个标识位
private boolean flag = true;
线程休眠---sleep
模拟网络延时:放大问题的发生性
package demo01;
//模拟倒计时
public class TestSleep {
public static void main(String[] args) {
try {
tenDown();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public static void tenDown() throws InterruptedException {
int num = 10;
while(true){
Thread.sleep(1000);
System.out.println(num--);
if (num<=0){
break;
}
}
}
}
package demo01;
import java.text.SimpleDateFormat;
import java.util.Date;
public class TestSleep {
public static void main(String[] args) {
//打印当前系统时间
Date startTime = new Date(System.currentTimeMillis());//获取当前系统时间
while(true){
try {
Thread.sleep(1000);
System.out.println(new SimpleDateFormat("HH:mm:ss").format(startTime));
startTime = new Date(System.currentTimeMillis());
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
//模拟倒计时
public static void tenDown() throws InterruptedException {
int num = 10;
while(true){
Thread.sleep(1000);
System.out.println(num--);
if (num<=0){
break;
}
}
}
}
线程礼让---yield
package demo01;
//测试礼让线程
public class TestYield {
public static void main(String[] args) {
MyYield myYield = new MyYield();
new Thread(myYield,"A").start();
new Thread(myYield,"B").start();
}
}
class MyYield implements Runnable{
线程强制执行---join
package demo01;
public class TestJoin implements Runnable{
观察线程状态
package demo01;
public class TestState {
public static void main(String[] args) throws InterruptedException {
Thread thread = new Thread(()->{
for (int i = 0; i < 5; i++) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}System.out.println("//////");
});
//观察状态
Thread.State state = thread.getState();
System.out.println(state);//NEW
//观察启动后
thread.start();
state = thread.getState();
System.out.println(state);//Run
while (state!=Thread.State.TERMINATED){
Thread.sleep(1000);
state = thread.getState();//更新线程状态
System.out.println(state);
}
}
}
线程优先级
package demo01;
import java.sql.SQLOutput;
public class TestPriority {
public static void main(String[] args) {
System.out.println(Thread.currentThread().getName() + "--->" + Thread.currentThread().getPriority());
MyPriority myPriority = new MyPriority();
Thread t1 = new Thread(myPriority);
Thread t2 = new Thread(myPriority);
Thread t3 = new Thread(myPriority);
Thread t4 = new Thread(myPriority);
Thread t5 = new Thread(myPriority);
t1.start();
t2.setPriority(4);
t2.start();
t3.setPriority(2);
t3.start();
t4.setPriority(1);
t4.start();
t5.setPriority(10);
t5.start();
}
}
class MyPriority implements Runnable{
线程同步安全
三大不安全案例
package syn;
//不安全的买票
//线程不安全,有负数
public class UnsafeBuyTicket {
public static void main(String[] args) {
BuyTicket station = new BuyTicket();
new Thread(station,"A").start();
new Thread(station,"B").start();
new Thread(station,"C").start();
}
}
class BuyTicket implements Runnable{
int ticketNums = 10;
boolean flag = true;
package syn;
//不安全的取钱
public class UnsafeBank {
public static void main(String[] args) {
Account account = new Account(100,"jijin");
Drawing you = new Drawing(account,50,"you");
Drawing girlFriend = new Drawing(account,100,"girlFriend");
you.start();
girlFriend.start();
}
}
//账户
class Account{
int money;
String name;
public Account(int money, String name) {
this.money = money;
this.name = name;
}
}
class Drawing extends Thread{
Account account;
int drawingMoney;//取了多少钱
int nowMoney;//手里有多少钱
public Drawing(Account account,int drawingMoney,String name){
super(name);
this.account = account;
this.drawingMoney = drawingMoney;
}
//取钱
package syn;
import java.util.ArrayList;
import java.util.List;
public class UnsafeList {
public static void main(String[] args) {
List<String> list = new ArrayList<String>();
for (int i = 0; i < 10000; i++) {
new Thread(()->{
list.add(Thread.currentThread().getName());
}).start();
}
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(list.size());
}
}
同步方法及同步块
同步方法
//private synchronized void buy() throws InterruptedException {}
同步块
//synchronized (account){}
同步块
/*
new Thread(()->{
synchronized (list){
list.add(Thread.currentThread().getName());
}
}).start();}
*/
CopyOnWriteArrayList
package syn;
import java.util.concurrent.CopyOnWriteArrayList;
public class TestJUC {
public static void main(String[] args) {
CopyOnWriteArrayList<String> list = new CopyOnWriteArrayList<String>();
for (int i = 0; i < 10000; i++) {
new Thread(()->{
list.add(Thread.currentThread().getName());
}).start();
}
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(list.size());
}
}
死锁
多个线程互相抱着对方的资源,然后形成僵持。
package syn;
public class DeadLock {
public static void main(String[] args) {
MakeUp g1 = new MakeUp(0,"HGN");
MakeUp g2 = new MakeUp(1,"BXGZ");
g1.start();
g2.start();
}
}
class Lipstick{
}
class Mirror{
}
class MakeUp extends Thread{
static Lipstick lipstick =new Lipstick();
static Mirror mirror =new Mirror();
int choice;
String girlsName;
MakeUp(int choice,String girlsName){
this.choice = choice;
this.girlsName = girlsName;
}
Lock锁
//定义lock锁
private final ReentrantLock lock = new ReentrantLock();
try {
lock.lock(); //加锁
} finally {
lock.unlock(); //解锁
}
package syn;
import java.util.concurrent.locks.ReentrantLock;
//不安全的买票
//线程不安全,有负数
public class UnsafeBuyTicket {
public static void main(String[] args) {
BuyTicket station = new BuyTicket();
new Thread(station,"A").start();
new Thread(station,"B").start();
new Thread(station,"C").start();
}
}
class BuyTicket implements Runnable{
//定义lock锁
private final ReentrantLock lock = new ReentrantLock();
int ticketNums = 10;
boolean flag = true;
生产者消费者问题
管程法
package syn;
//测试:生产者消费者模型--->利用缓冲区解决(管程法)
//生产者,消费者,产品,缓冲区
public class TestPC {
public static void main(String[] args) {
SynContainer container = new SynContainer();
new Productor(container).start();
new Consumer(container).start();
}
}
//生产者
class Productor extends Thread{
SynContainer container;
public Productor(SynContainer container){
this.container = container;
}
//生产
