JavaWeb笔记03:多线程的创建、构造方法和属性

多线程的使用

t.start 将线程放到就绪队列中,当线程被分配到CPU资源时,才会运行起来。

t 线程被调度下来的5中情况
1:被高优先级抢占
2:执行结束
3:等待IO
4:主动放弃
5:时间片耗尽了

注意:

  • 多线程,可能提升执行的速度(每多一个线程(调度单位),将相当于增加了抢到CPU的机会)。线程太多时,再加也无意义了,因为等待调度的都是咱们的线程了。
  • 多线程的创建也耗费时间,如果耗费时间大于节省的时间,则不必采用多线程。
  • 有些场景,需要创建多线程来提升效率,比如IO时。
创建线程
  1. 创建一个线程对象
  2. 调用start方法,把线程放到就绪队列中

创建线程对象的两种方式:
1) Thread直接创建

class MyThread extends Thread{
	@Override
	public void run(){
	//需要执行的代码
	}
}
Thread t = new MyThread();

2) 通过实现Runnable接口

class MyRunnable implements Runnable{
	@Override
	public void run(){
	//需要执行的代码
	}
}
Runnable target = new MyRunnable();
Thread  t = new Thread(target);

Thread 代表的是线程对象
Runnable 相当于一个需要线程执行的任务清单对象,它需要将Runnable对象交给Thread。

任务线程一体——Thread

Thread t = new MyThread();
Runnable target = t;

任务线程分离——Runnable

Thread p= new Thread(t); 等价于
Thread p=new Thread(target);

注意:

  • Thread也实现了Runnable接口,因此Thread对象也可以看成一个Runnable对象。
  • t.join: 主线程放弃抢占CPU,即主线程阻塞,直到t线程结束。
  • b.join()会抛出IntrrupedException异常

匿名类直接创建线程对象

  1. Thread
Thread a = new Thread(){
	@Override
	public void run(){
	//需要执行的代码
	}
}
  1. Runnable
Thread d = new Thread(new Runnable() {
    @Override
    public void run() {
        sum();
    }
});

这个可以变形,为lamda表达式,alt+enter

Thread d = new Thread(() -> sum());
线程构造方法和属性

线程的构造方法:

Thread(): 创建线程对象
Thread(Runnable target) :使用Runnable 对象创建线程对象
Thread(String name) :创建线程对象,并命名
Thread(Runnable target, String name) :使用Runnable 对象创建线程对象,并命名
Thread(ThreadGroup group, Runnable target):线程可以被用来分组管理,分好的组即使线程组,了解即可

线程命名使用:

Thread t3 = new Thread("这是我的名字");
Thread t4 = new Thread(new MyRunnable(), "这是我的名字");

线程的属性

ID getId()
名称 getName()
状态 getState()
优先级 getPriority()
是否后台线程 isDaemon()
是否存活 isAlive()
是否被中断 isInterrupted()

线程调试:
Thread t = Thread.currentThread(): 获取当前进程(被哪个线程执行到,就在代表哪个线程)

线程监视 cmd > jconsole
where jconsole 获取jconsole的路径

通过jconsole观察属性:
1.属性的解释
2.属性的值

注意:

  • 线程分为:后台线程、前台(非后台)线程,非后台线程结束时,标志着整个线程结束。
  • JVM启动时就有好多线程,包括后台线程如垃圾回收线程GC(专门负责垃圾回收)等。
  • join中就是通过isAlive()判断线程是否结束来进行等待的。

不同线程的状态:

public class 演示不同线程下的状态 {
    private static void print() {
        Thread current = Thread.currentThread();
        // 之所以全部带着 id,是希望能分辨出我的打印是哪个线程打的
        System.out.println(current.getId() + ":" + current.getName());
        System.out.println(current.getId() + ":" + current.getPriority());
        System.out.println(current.getId() + ":" + current.getState());
        System.out.println(current.getId() + ":" + current.isDaemon());
        System.out.println(current.getId() + ":" + current.isAlive());
        System.out.println(current.getId() + ":" + current.isInterrupted());
    }
    public static void main(String[] args) {
        Thread t = new Thread(() -> {
            print();    // 在子线程中打印
        });
        t.start();
        print();        // 在主线程中打印
    }
}

线程优先级设置:

public class Main {
    private static class A extends Thread {
        @Override
        public void run() {
            System.out.println(" A");
        }
    }
    private static class B extends Thread {
        @Override
        public void run() {
            System.out.println(" B");
        }
    }
    public static void main(String[] args) {
        A a = new A();
        a.start();
        B b = new B();
        //b.setPriority(Thread.MAX_PRIORITY); // 优先级只是个建议
        b.start();
        System.out.println(" main");
    }
}

在执行顺序上,大概率的执行顺序是主线程 main > A > B ,因为主线程已经在运行中,除非main的时间片耗尽。当然任何情况都是可能发生的。

如果想提高某个线程的优先级,可以考虑使用b.setPriority(Thread.MAX_PRIORITY);来提升线程的优先级。

发布了54 篇原创文章 · 获赞 6 · 访问量 4795

猜你喜欢

转载自blog.csdn.net/glpghz/article/details/104671969