java EE初阶 — 如何进行多线程编程

1.java如何进行多线程编程

Thread 是java操作多线程最核心类。
使用Thread不需要import导入别的包,因为它是在java.lang下面

1.1 最基础的多线程代码

package thread;

class MyThread extends Thread{
    //在此类中重写run方法
    @Override
    public void run() {
        System.out.println("hello world");
    }
}

public class ThreadDemo1 {

    public static void main(String[] args) {
        Thread thread = new MyThread();
        thread.start();//线程中的特殊方法 - 创建一个线程
    }
}

thread包中创建一个对象,Mythread子类继承Thread父类
重写Thread类中的run方法输出hello world。
start是线程中的特殊方法,在这里的作用是创建一个新的线程,而不是调用run。
新的线程负责调用run方法来输出hello world。


新的进程调用操作系统的API,通过操作系统内核创建新线程的PCB。
并且把要执行的指令交给这个PCB。
当这个PCB被调度到CPU上的时候,也就执行到了线程run方法中的代码了。

如果run方法执行完毕,这个新的线程会销毁。

1.2 线程的优势

改动一下上面的代码:

package thread;

class MyThread extends Thread{
    //在此类中重写run方法
    @Override
    public void run() {
        while (true) {
            System.out.println("hello world");
        }
    }
}

public class ThreadDemo1 {

    public static void main(String[] args) {
        Thread thread = new MyThread();
        thread.start();//线程中的特殊方法 - 创建一个线程
        while (true) {
            System.out.println("hello java");
        }
    }
}

如果此时的代码时非线程的,程序会死循环的输出 hello world,并且没有机会执行 hello java
但是实际情况是,这两条输出语句都会执行,并且都是死循环

为了方便观察,调用sleep设置睡眠时间为1秒。

package thread;

class MyThread extends Thread{
    //在此类中重写run方法
    @Override
    public void run() {
        while (true) {
            System.out.println("hello world");
            //睡一会儿
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

public class ThreadDemo1 {

    public static void main(String[] args) {
        Thread thread = new MyThread();
        thread.start();//线程中的特殊方法 - 创建一个线程
        while (true) {
            System.out.println("hello java");
            //睡一会儿
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}


先执行 hello world 还是先执行 hello java 是不确定的。
操作系统调度线程的时候,是“抢占式执行的”,具体哪个先哪个后,是不确定的。
这要取决与操作系统的调度器具体的实现策略。

虽然有优先级,但是在应用程序层面上无法修改。
从应用程序(代码)的角度,看到的效果,就好像是线程的调度效果好像是随机的一样。

内核里本身并不是随机的,但是干预的元素太多了,并且应用程序这一层也无法感知到细节,就只能认为是随机的了。

抢占式执行,随机访问就是线程安全问题的罪魁祸首。


如果其中一条语句无法输入,不会影响另一条语句。

把输出 hello world 的语句给屏蔽了,但是不会影响输出 hello java

class MyThread extends Thread{
    //在此类中重写run方法
    @Override
    public void run() {
        while (true) {
            //System.out.println("hello world");
            //睡一会儿
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}


就像是,张三和李四去学校,张三不来了并不会决定李四来不来。


start 和 run 之间的区别:

start是真正上创建了一个进程(从系统这里创建的),线程是独立的执行流。

run只是描述了线程要干的活是什么,如果直接在main中调用 run 。
此时没有创建新线程,全是main线程一个人干活。

2.java中创建线程的方法

2.1 继承Thread 重写run


第一种方法就是前面介绍的方法。

2.2 实现Runnable 接口

package thread;

//Runnable 的作用是描述一个要“执行的任务”,run方法就是任务的执行细节
class MyRunnable implements Runnable {
    @Override
    public void run() {
        System.out.println("hello 张三");
    }
}

public class ThreadDemo2 {
    public static void main(String[] args) {
        //这只是描述了任务
        Runnable runnable = new MyRunnable();
        //把任务交给线程来执行
        Thread thread = new Thread(runnable);
        thread.start();
    }
}

如果未来要更改代码,不用多线程,
而是使用多进程,或者线程池、协程…此时的代码改动比较小。

2.3 使用匿名的内部类,继承 Thread

package thread;

public class ThreadDemo3 {
    public static void main(String[] args) {
        Thread thread = new Thread() {
            @Override
            public void run() {
                System.out.println("haha");
            }
        };
        thread.start();
    }
}

上面方法先是创建了一个Thread的子类(子类没有名字),所以才叫做匿名。
然后是创建了子类的实例,并且让thread引用指向该实例。

2.4 使用匿名类。继承 Runnable

package thread;

public class ThreadDemo4 {
    public static void main(String[] args) {
        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("hehehehe");
            }
        });
        thread.start();
    }
}

这个写法本质和2相同。
只不过是把实现 Runnable 任务交给了匿名内部类的语法。

此处是创建了一个类,实现 Runnable ,同时创建了类的实例,并且传给 Thread 的构造方法。

2.5 使用 Lambda 表达式(最简答、最推荐)

package thread;

public class ThreadDemo5 {
    public static void main(String[] args) {
        Thread thread = new Thread(() -> {
            System.out.println("hellohello");
        });
        thread.start();
    }
}

把任务用 Lambda 表达式来描述
直接把 Lambda 传给 Thread 构造方法。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

与大师约会

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值