Java线程的基础用法教程

发布时间:2021-10-14 15:25:37 作者:iii
来源:亿速云 阅读:128

这篇文章主要讲解了“Java线程的基础用法教程”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java线程的基础用法教程”吧!

线程

线程和进程

进程是操作系统分配资源的最小单位,而线程是程序执行的最小单位,他们都是可以并发执行的。一个进程至少有一个线程,这些线程共享进程的资源空间。

线程简介

每个线程都有一个优先级,高优先级的线程比低优先级的线程先执行。优先级的取值范围是1到10的整数,默认是5。每个线程有可能被标记为一个守护线程。当一个线程创建另外一个新的线程对象,新的线程的优先级等于创建他的线程的优先级;如果新的线程对象是一个守护线程当且仅当创建他的线程是一个守护线程。

线程分类

Java线程分为守护线程(Daemon Thread)用户线程(User Thread)。守护线程和用户线程基本上是一样的,唯一的区别是如果用户线程全部退出运行了,不管有没有守护线程虚拟机都会退出。守护线程的作用是为其他的线程的运行提供服务,最典型的守护线程就是GC(垃圾回收期)。

创建线程

创建线程的方式

创建一个线程类有三种方式:

Thread

Thread简介

Thread是创建线程最关键的一个类,这个词本身也代表线程,Thread类实现了Runnable接口。

代码示例

public class ThreadDemo extends Thread {
	public void run() {
		for (int i = 0; i < 60; i++) {
			System.out.println(getName() + ":" + i);
		}
	}
}
public class Demo{
    public static void main(String[] args) {
        ThreadDemo t1 = new ThreadDemo();
		ThreadDemo t2 = new ThreadDemo();
		t1.start();
		t2.start();
    }
}

Runnable

Runnable简介

Runnable是提供线程的接口,有一个抽象方法public abstract void run()。实现了这个接口的类必须实现它的run方法。

代码示例

public class Runnable implements Runnable{
    public void run() {
       public void run() {
           for (int i = 0; i < 60; i++) {
				System.out.println(Thread.currentThread().getName() + ":" + i);
           }
	   }
	}
}
public class Demo{
    public static void main(String[] args) {
        RunnableDemo run = new RunnableDemo();
        Thread t1 = new Thread(run);
        Thread t2 = new Thread(run);
        t1.start();
        t2.start();
    }
}

Callable和Future

Callable和Future简介

Thread和Runnable创建线程不能获取线程的返回值。从Java1.5开始,就提供了Callable和Future,通过他们可以在任务执行完毕之后得到任务执行结果。

代码案例

public class MyCallable implements Callable<Integer> {
    public Integer call() {
        int sum = 0;
        for (int i = 0; i <= 100; i++) {
            sum += i;
        }
        return new Integer(sum);
    }
}
public class Demo{
    public static void main(String[] args) {
        MyCallable callable = new MyCallable();
        FutureTask<Integer> result = new FutureTask<Integer>(callable);
        new Thread(result).start();
        try {
            Integer value = result.get();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
    }
}

线程生命周期

线程状态

在Thread类中有一个内部枚举类State代表了线程的状态,一个线程从创建到销毁就是一个完整的生命周期。

public enum State {
    /**
     * 线程被创建,还没有开始运行
     */
    NEW,
    /**
     * 线程运行状态,运行状态的线程是正在被Java虚拟机执行,但是也可能正在等待操作系统的其他资源例如处理器
     */
    RUNNABLE,
    /**
     * 线程阻塞状态,等待监视器锁。处于阻塞状态线程是在等待监视器锁为了:进入同步代码块/方法或者被调用后重	  		 * 新进入同步代码/方法
     */
    BLOCKED,
    /**
     * 线程等待状态,一个线程处于等待状态由于调用了以下这几种方法:Object.wait;Thread.join;LockSupp
     * ort.park。处于等待的线程正在等待另一个线程执行一个特定的操作。
     */
    WAITING,
    /**
     * 线程超时等待状态,一个线程处于超时等待状态在一个特定的等待时间,由于调用了以下几个方法Thread.slee
     * p;Object.wait(long);Thread.join(long);LockSupport.parkNanos;LockSupport.parkUntil。
     */
    TIMED_WAITING,
    /**
     * 线程结束状态,线程已经执行完成了。
     */
    TERMINATED;
}

线程状态转换

线程状态转换图

线程从创建后就在几个状态中切换。下面是一个线程状态转换图,调用不同的方法就可以切换线程线程的状态。

Java线程的基础用法教程

运行状态&无限等待

调用Object.wait();Thread.join();LockSupport.park()方法可以让线程从运行状态进入到无限等待状态。

运行状态&超时等待

调用Object.wait(long);Thread.join(long);LockSupport.park(long)方法可以让线程从运行状态进入到等待状态,直到到达等待时间或者主动唤醒。

public class Demo {
    public static void main(String[] args) {
        Demo demo = new Demo();
        Thread t1 = new Thread(() -> {
            synchronized (demo) {
                for (int i = 0; i < 1000; i++) {
                    if (i == 500) {
                        try {
                            demo.wait(100);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    }
                    System.out.println("------t1------: " + i);
                }
            }
        });
        Thread t2 = new Thread(() -> {
            synchronized (demo) {
                for (int i = 0; i < 1000; i++) {
                    System.out.println("------t2------: " + i);
                }
            }
        });
        t1.start();
        t2.start();
    }
}
public class Demo {
    public static void main(String[] args) throws Exception {
        System.out.println("main start");
        Thread t1 = new Thread(() -> {
            for (int i = 0; i < 1000; i++) {
                System.out.println("----t1----: " + i);
            }
        });
        t1.start();
        t1.join(1);
        System.out.println("main end");
    }
}
public class Demo {
    public static void main(String[] args) {
        System.out.println("main start");
        Thread t1 = new Thread(() -> {
            System.out.println("t1 start");
            LockSupport.parkNanos(3000000000L);
            System.out.println("t1 end");
        });
        t1.start();
        System.out.println("main end");
    }
}
public class Demo {
    public static void main(String[] args) throws Exception{
        System.out.println("main start");
        Thread t1 = new Thread(() -> {
            System.out.println("t1 start");
            String dateTimeStr = "2021-04-04 14:57:00";
            DateTimeFormatter df = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
            LocalDateTime dateTime = LocalDateTime.parse(dateTimeStr, df);
            LockSupport.parkUntil(dateTime.toInstant(ZoneOffset.of("+8")).toEpochMilli());
            System.out.println("t1 end");
        });
        t1.start();
        System.out.println("main end");
    }
}

无限等待&阻塞状态

对象调用wait方法后线程会进入无限等待状态,当对象调用notify或者notifyAll时,线程将从无限等待状态进入阻塞状态。

阻塞状态到运行状态

线程处于阻塞状态,如果获取到锁对象,就进入运行状态。

感谢各位的阅读,以上就是“Java线程的基础用法教程”的内容了,经过本文的学习后,相信大家对Java线程的基础用法教程这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是亿速云,小编将为大家推送更多相关知识点的文章,欢迎关注!

推荐阅读:
  1. expect的基础用法
  2. Python教程 | Requests的基本用法

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

java

上一篇:php页面缓存ob系列函数有哪些

下一篇:signal函数是什么

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》