大数据最新学习多线程,创建多线程的三种方式_实现多线程的三种方法(2)

本文介绍了Java中的抢占式线程调度机制,包括优先级设定、如何通过继承Thread类创建多线程,以及如何使用start方法、run方法、优先级和随机性。此外,还探讨了如何获取和设置线程名称,以及线程的睡眠操作。最后强调了系统化学习的重要性,邀请读者加入技术交流社区共同成长。
摘要由CSDN通过智能技术生成

抢占式调度:优先让优先级高的线程使用CPU,如果线程的优先级相同,那么会随机选择一个(线程随机性),Java使用的为抢占式调度。

用户可在任务管理器中的详细信息列表设置线程的优先级

主线程

主线程:执行main方法的线程,程序的执行从main方法开始,从上到下逐行执行

public class Person {
    private String name;

    public Person(String name) {
        this.name = name;
    }

    public void run(){
        for (int i = 0; i < 20; i++) {
            System.out.println(name+"run"+i);
        }
    }
}

public class Demo01 {
    public static void main(String[] args) {
        Person p1 = new Person("aaa");
        p1.run();

        Person p2 = new Person("bbb");
        p2.run();
    }
}

java程序进入到内存中,JVM会先执行main方法,JVM会通过操作系统,开辟一条main方法通向cpu的路径,cpu通过这条路径执行main方法。这条路径的名字就叫主线程(main线程)。

创建多线程程序

创建多线程程序的第一种方式:继承thread类
java.lang.Thread:就是一个描述线程的类,类想实现多线程,就必须继承Thread类。
线程是程序中执行的线程。 Java虚拟机允许应用程序同时运行多个执行线程。

java程序属于抢占式调度:优先让优先级高的线程使用CPU,如果线程的优先级相同,那么会随机选择一个线程执行(线程随机性)

实现步骤

创建Thread类的子类,继承Thread类
在Thread类的子类中,重写Thread类的run方法,设置线程任务(开启新的线程的目的)
创建Thread类的子类对象
调用Thread类中的start方法开启新线程,执行线程任务run方法
void start(): 导致此线程开始执行, Java虚拟机调用此线程的run方法。结果是两个线程同时运行:当前线程(主线程:执行main方法中的代码)和另一个线程(新的线程:执行其run方法中的方法)。不止一次启动线程永远不合法。 特别是,一旦完成执行,线程可能无法重新启动。—>start方法只能执行一次

示例:通过继承thread类,创建多线程程序

//1.创建Thread类的子类,继承Thread类
public class MyThread extends Thread{
    //2.在Thread类的子类中,重写Thread类的run方法,设置线程任务(开启新的线程的目的)

    @Override
    public void run() {
        //新线程执行run方法
        for (int i = 0; i < 20; i++) {
            System.out.println("run-->"+i);
        }
    }
}

public class Demo01Thread {
    public static void main(String[] args) {
        //3.创建Thread类的子类对象
        MyThread mt = new MyThread();
        //mt.run(); 不会开启新线程
        mt.start();
        //main线程会继续执行for循环
        for (int i = 0; i < 20; i++) {
            System.out.println("main-->"+i);
        }

    }
}

新的线程和main线程优先级相同,cpu会随机执行某一个线程,就会出现随机性打印结果

run-->0
main-->0
main-->1
run-->1
main-->2
run-->2
main-->3
run-->3
main-->4
run-->4
run-->5
run-->6
run-->7
run-->8
run-->9
main-->5
run-->10
main-->6
run-->11
main-->7
run-->12
main-->8
run-->13
main-->9
run-->14
main-->10
run-->15
main-->11
main-->12
main-->13
main-->14
main-->15
main-->16
main-->17
run-->16
main-->18
run-->17
run-->18
run-->19
main-->19

随机性打印结果原理:
1:JVM首先运行main方法,找到操作系统开辟一条通向cpu的路径,这个路径就是主线程,cpu通过这个路径就可以执行main
2:执行MyThread mt = new MyThread();,开辟一条新的通向cpu的路径。mt.start();用来执行run方法
3:现在有两条路径通向cpu,对于cpu而言就有了选择的权力,可以随机选择一个来执行(主线程和新线程的优先级相同),所以就有了随机性打印结果。
4:内存图解
在这里插入图片描述
获取线程的名称

主线程:main
新线程:Thread-0,Thread-1…Thread-n

使用Thread类中的getName方法
String getName() 返回此线程的名称。

public class MyThread extends Thread{
    @Override
    public void run() {
        //1.使用Thread类中的getName方法
        String name = getName();
        System.out.println(name);
  }
}

public class Demo01Thread {
    public static void main(String[] args) {
        //线程1
        MyThread myThread = new MyThread();
        myThread.start();//Thread-0
        //线程2
        new MyThread().start();//Thread-1
    }
}

可以先获取当前正在执行的线程,再通过getName获取线程名称
static Thread currentThread() 返回对当前正在执行的线程对象的引用。

public class MyThread extends Thread{
    @Override
    public void run() {
       //2.可以先获取当前正在执行的线程,再通过getName获取线程名称
        Thread currentThread = Thread.currentThread();
        String name = currentThread.getName();
        System.out.println(name);
    }
}

public class Demo01Thread {
    public static void main(String[] args) {
        //线程1
        MyThread myThread = new MyThread();
        myThread.start();//Thread-0
        //线程2
        new MyThread().start();//Thread-1

        //获取主线程名称
        System.out.println(Thread.currentThread().getName());
    }
}

设置线程的名称

可以使用Thread的方法setName
void setName​(String name) 将此线程的名称更改为等于参数 name 。

public class MyThread extends Thread{
 @Override
    public void run() {
        System.out.println(Thread.currentThread().getName());
    }
}

public class Demo01 {
    public static void main(String[] args) {
        MyThread mt = new MyThread();
        mt.setName("aaa");
        mt.start();
    }
}

添加一个带参构造方法,参数传递线程的名称,调用父类的带参构造,把名字传递给父类

public class MyThread extends Thread{
    public MyThread() {
    }

    public MyThread(String name) {
        super(name);
    }

    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName());
    }
}

public class Demo01 {
    public static void main(String[] args) {
        MyThread mt2 = new MyThread("bbb");
        mt2.start();//bbb
    }
}

Thread类中的sleep方法
static void sleep​(long millis) 导致当前正在执行的线程休眠(暂时停止执行)指定的毫秒数,具体取决于系统计时器和调度程序的精度和准确性。

img
img

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

-Dnutfn0k-1714790862082)]
[外链图片转存中…(img-G1j38Nmj-1714790862082)]

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 19
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值