概念
进程是指可执行程序并存放在计算机存储器的一个指令序列,它是一个动态执行的过程。
线程是比进程还要小的运行单位,一个进程包含多个线程
线程可以看成一个子程序
线程的创建
- 创建一个Thread类,或者一个Thread子类的对象
- 创建一个实现Runnable接口的类的对象
- 两种方式
- 通过继承Thread类创建线程(MyThread)- new MyThread().start()
- 通过实现Runnable接口创建线程(MyRunnable)- new Thread(new MyRunnable()).start()
Thread类
是一个线程类,位于java.lang包下
构造方法和说明
- Thread(),创建一个线程对象
- Thread(String name),创建一个具有指定名称的线程对象
- Thread(Runnable target),创建一个基于Runnable接口实现类的线程对象
- Thread(Runnable terget ,String name),创建一个基于Runnable接口实现类,并且具有指定名称的线程对象
package com.dodoke.thread;
class MyThread extends Thread{
public void run() {
System.out.println(getName()+"该线程正在执行!");
}
}
public class ThreadTest {
public static void main(String[] args) {
// System.out.println("主线程1");
MyThread mt=new MyThread();
mt.start();//启动线程,不是调用run方法,但是运行的还是run方法,一个线程只能启动一次
mt.start();//编译正常,运行时抛出异常,不合逻辑的线程状态,不能多次调用,只能启动一次
// System.out.println("主线程2");
}
}
package com.dodoke.thread1;
class MyThread extends Thread{
public MyThread(String name) {
super(name);
}
public void run() {
for(int i=1;i<=10;i++) {
System.out.println(getName()+"正在运行"+i);
}
}
}
public class ThreadTest {
public static void main(String[] args) {
MyThread mt1=new MyThread("线程1");
MyThread mt2=new MyThread("线程2");
mt1.start();
mt2.start();//运行结果都是随机的
//线程获取cpu的使用权时随机的,线程运行的结果会有很多种不同的情况
}
}
常用方法
- public void run()--线程相关的代码写在该方法中,一般需要重写
- public void start()--启动线程的方法
- public static void sleep(long m)--线程休眠m毫秒的方法
- public void join()--优先执行调用join()方法的线程
Runnable接口
- 只有一个方法run()
- Runnable是Java中用以实现线程的接口
- 任何实现线程功能的类都必须实现该接口
package com.dodoke.runnable;
class PrintRunnable implements Runnable{
int i=1;//将i变成成员变量,运行结果少了
@Override
// run方法中的代码可以被多个线程共享,多个线程就是Thread类的实例
public void run() {
// int i=1;
while(i<=10) {
System.out.println(Thread.currentThread().getName()+"正在运行"+(i++));
}
}
}
public class Test {
public static void main(String[] args) {
PrintRunnable pr=new PrintRunnable();//先定义runnable的实现类对象
Thread t1=new Thread(pr);//创建线程类的对象
t1.start();//启动线程
// PrintRunnable pr1=new PrintRunnable();
// pr1和pr2的运行结果没什么变化
// Thread t2=new Thread(pr1);
Thread t2=new Thread(pr);
t2.start();
}
}
线程创建
通过继承Thread类的方法创建该线程类,重写run()方法
通过实现Runnable接口的方法创建
为什么实现Runnable接口--Java不支持多继承,不打算重写Thread类的其他方法
线程的状态
新建(New)、可运行(Runnable)、正在运行(Running)、阻塞(Blocked)、终止(Dead)
线程的生命周期
sleep方法的应用
- Thread类的方法----- public static void sleep(long millis)
- 作用:在指定的毫秒数内让正在执行的线程休眠(暂停执行)
- 参数为休眠的时间,单位是毫秒
package com.dodoke.sleep;
class MyThread implements Runnable{
@Override
public void run() {
for(int i=1;i<=30;i++) {
System.out.println(Thread.currentThread().getName()+"执行第"+i+"次!");
try {
Thread.sleep(1000);//1000毫秒
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
public class SleepDemo {
public static void main(String[] args) {
MyThread mt=new MyThread();
Thread t=new Thread(mt);
t.start();
Thread t1=new Thread(mt);
t1.start();
}
}
join方法的应用
- Thread类的方法----- public final void join()
- 作用:等待调用该方法的线程结束后才能执行
- Thread类的方法----- public final void join(long millis)
- 作用:等待该线程终止的最长时间为millis毫秒
package com.dodoke.join;
class MyThread extends Thread{
public void run() {
for(int i=1;i<=500;i++) {
System.out.println(getName()+"正在执行"+i+"次!");
}
}
}
public class JoinDemo {
public static void main(String[] args) {
MyThread mt=new MyThread();
mt.start();
try {
mt.join(1);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
for(int i=1;i<=20;i++) {
System.out.println("主线程运行第"+i+"次!");
}
}
}
线程优先级
- Java为线程类提供了10个优先级
- 优先级可以用整数1-10表示,超出范围会抛出异常
- 主线程默认优先级为5
- 优先级常量
- MAX_PRIORITY:线程的最高优先级10
- MIN_PRIORITY:线程的最低优先级1
- NORM_PRIORITY:线程的默认优先级5
- 优先级的方法
- public int getPriority()-- 获取线程优先级的方法
- public void setPriority(int newPriority)-- 设置线程优先级的方法
package com.dodoke.priority;
class MyThread extends Thread{
private String name;
public MyThread(String name) {
this.name=name;
}
public void run() {
for(int i=1;i<=50;i++) {
System.out.println("线程"+name+"正在运行"+i);
}
}
}
public class PriorityDemo {
public static void main(String[] args) {
//获取主线程的优先级
int mainPriority=Thread.currentThread().getPriority();
// System.out.println("主线程的优先级为:"+mainPriority);
MyThread mt1=new MyThread("线程1");
MyThread mt2=new MyThread("线程2");
// mt1.setPriority(10);
mt1.setPriority(Thread.MAX_PRIORITY);
mt2.setPriority(Thread.MIN_PRIORITY);
mt2.start();
mt1.start();
// System.out.println("线程1的优先级为:"+mt1.getPriority());
}
}
多线程运行问题
- 各个线程是通过竞争CPU时间而获得运行机会的
- 各线程什么时候得到CPU时间,占用多久,是不可预测的
- 一个正在运行着的线程在什么地方被暂停是不确定的
同步
synchronized关键字用在成员方法,静态方法,语句块
public synchronized void saveAccount(){}
public static synchronized void saveAccount(){}
synchronized (obj){......}
线程间通信
- wait()方法:中断方法的执行,使线程等待
- notify()方法:唤醒处于等待的某一个线程,使其结束等待
- notifyAll()方法:唤醒所有处于等待的线程,使它们结束等待
银行存取款问题
package com.dodoke.bank;
public class Bank {
private String account;//账号
private int balance;//账户余额
public Bank(String account, int balance) {
this.account = account;
this.balance = balance;
}
public String getAccount() {
return account;
}
public void setAccount(String account) {
this.account = account;
}
public int getBalance() {
return balance;
}
public void setBalance(int balance) {
this.balance = balance;
}
@Override
public String toString() {
return "Bank [账号:" + account + ", 余额:" + balance + "]";
}
//存款
public synchronized void saveAccount() {
//可以在不同的位置处添加sleep方法
//获取当前的账号余额
int balance=getBalance();
try {
Thread.sleep(1000);
}catch(InterruptedException e) {
e.printStackTrace();
}
//修改余额,存100元
balance+=100;
//修改账户余额
setBalance(balance);
//输出存款后的账户余额
System.out.println("存款后的账户余额为:"+balance);
}
public void drawAccount() {
synchronized(this) {
//可以在不同的位置处添加sleep方法
//获得当前的账户余额
int balance=getBalance();
//修改余额,取200
balance=balance-200;
try {
Thread.sleep(1000);
}catch(InterruptedException e){
e.printStackTrace();
}
//修改账户余额
setBalance(balance);
System.out.println("取款后的账户余额为:"+balance);
}
}
}
package com.dodoke.bank;
//取款
public class DrawAccount implements Runnable{
Bank bank;
public DrawAccount(Bank bank) {
this.bank=bank;
}
public void run() {
bank.drawAccount();
}
}
package com.dodoke.bank;
//存款
public class SaveAccount implements Runnable{
Bank bank;
public SaveAccount(Bank bank) {
this.bank=bank;
}
public void run() {
bank.saveAccount();
}
}
package com.dodoke.bank;
public class Test {
public static void main(String[] args) {
//创建账户,给定余额为1000
Bank bank=new Bank("1001",1000);
//创建线程对象
SaveAccount sa=new SaveAccount(bank);
DrawAccount da=new DrawAccount(bank);
Thread save=new Thread(sa);
Thread draw=new Thread(da);
save.start();
draw.start();
try {
save.join();
draw.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(bank);
}
}
生产和消费的问题(生产一个消费一个)
package com.dodoke.queue;
public class Consumer implements Runnable{
Queue queue;
public Consumer(Queue queue) {
this.queue = queue;
}
@Override
public void run() {
while(true) {
queue.get();
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
package com.dodoke.queue;
public class Producer implements Runnable{
Queue queue;
public Producer(Queue queue){
this.queue=queue;
}
public void run() {
int i=0;
while(true) {
queue.set(i++);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
package com.dodoke.queue;
public class Queue {
private int n;
boolean flag=false;
public synchronized int get() {
if(flag==false) {
try {
wait();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
System.out.println("消费:"+n);
flag=false;//消费完毕,容器中没有数据
notifyAll();
return n;
}
public synchronized void set(int n) {
if(flag) {
try {
wait();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
System.out.println("生产:"+n);
this.n = n;
flag=true;//生产完毕,容器中已经有数据
notifyAll();
}
}
package com.dodoke.queue;
public class Test {
public static void main(String[] args) {
Queue queue=new Queue();
new Thread(new Producer(queue)).start();
new Thread(new Consumer(queue)).start();
}
}