多线程基础(一)

一、进程和线程的概念

  • 进程:运行中的某个程序或者应用,至少包含一个线程
  • 线程:进程中负责执行的一个或者多个执行单元,归属于进程,且多个线程共享进程的资源
  • 并发机制:多个线程同时运行,CPU给每个线程分配时间片,获得时间的线程运行,其他线程等待,由于时间片很短,从宏观上看是线程都在运行,微观上看是线程走走停停

二、线程的创建

1、继承Thread类,重写run()
public class MyThread extends Thread{
    private String name;
    MyThread(String name){
        this.name=name;
    }
    
    @Override
    public void run(){
        System.out.println("name"+name+",此线程的id为"+Thread.currentThread().getId());
    }
    
    public static void main(String[] args) {
        System.out.println("当前主线程id为:"+Thread.currentThread().getId());
        MyThread myThread1 = new MyThread("线程1");
        myThread1.start();
        MyThread myThread2 = new MyThread("线程2");
        myThread2.run();
        
    }
}

输出如下:

image.png

小结:

  • start()和run()方法的不同。start是启动线程,交给CPU去获取时间片并调用run方法。而用run方法相当于交给主线程去执行run,并不会创建新的线程。
  • 线程1是先创建的,但是输出结果在后,说明线程的创建并不会阻塞主线程的执行
2、实现Runnable接口
public class MyRunnable implements Runnable {
    private String name;
    MyRunnable(String name){
        this.name = name;
    }
    public void run() {
        System.out.println("启动"+name+":id为"+Thread.currentThread().getId());
    }
    
    public static void main(String[] args) {
        MyRunnable my = new MyRunnable("runnable线程");
        Thread thread = new Thread(my);
        thread.start();
    }
}
3、最简洁的启动线程的方式
new Thread(new Runnable(){
  public void run(){
        xxxxxx
  }
};).start();

三、线程的状态

  • 创建(new)状态: 准备好了一个多线程的对象
  • 就绪(runnable)状态: 调用了start()方法, 等待CPU进行调度
  • 运行(running)状态: 执行run()方法
  • 阻塞(blocked)状态: 暂时停止执行, 可能将资源交给其它线程使用
  • 终止(dead)状态: 线程销毁
image.png

1、当线程进入就绪状态后,要等CPU分配到时间片之后,线程便真正进入运行状态。
2、线程在运行状态过程中,可能有多个原因导致当前线程不继续运行下去,比如用户主动让线程睡眠(睡眠一定的时间之后再重新执行)、用户主动让线程等待,或者被同步块给阻塞,此时就对应着多个状态:time waiting(睡眠或等待一定的事件)、waiting(等待被唤醒)、blocked(阻塞)。
3、当由于突然中断或者子任务执行完毕,线程就会被消亡。

注:sleep和wait的区别:

  • sleep是Thread类的方法,wait是Object类中定义的方法.
  • Thread.sleep不会导致锁行为的改变, 如果当前线程是拥有锁的, 那么
    Thread.sleep不会让线程释放锁.
  • Thread.sleep和Object.wait都会暂停当前的线程. OS会将执行时间分配给其它线程. 区别是, 调用wait后, 需要别的线程执行notify/notifyAll才能够重新获得CPU执行时间.

四、线程API

1、sleep():线程休眠,且不会释放锁。
public class MyThreadAPI{
    private int i = 0;
    private Object object = new Object();
    
    public static void main(String[] args) {
        MyThreadAPI api = new MyThreadAPI();
        MyThreadSleep thread1 = api.new MyThreadSleep() ;
        MyThreadSleep thread2 = api.new MyThreadSleep() ;
        thread1.start();
        thread2.start();
    }
    class MyThreadSleep extends Thread{
        public void run(){
            synchronized (object) {
                i++;
                System.out.println(Thread.currentThread().getName()+"睡眠前i:"+i);
                System.out.println(Thread.currentThread().getName()+"进入睡眠");
                try {
                    Thread.currentThread().sleep(3000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName()+"睡眠结束");
                i++;
                System.out.println(Thread.currentThread().getName()+"睡眠后i:"+i);
            }
        }
    }
}

输出结果:


image.png
2、yield(),让出当前时间片,与sleep不同的是不可选择时间
public class MyThreadYeild extends Thread {
    public void run(){
        long start = System.currentTimeMillis();
        int count = 0;
        for(int i = 1;i<50000000;i++){
            count++;
            Thread.yield();
        }
        System.out.println("当前count为:"+count);
        long end = System.currentTimeMillis();
        System.out.println("耗时:"+(end-start)+"毫秒");
    }
    public static void main(String[] args) {
        MyThreadYeild thread = new MyThreadYeild();
        thread.start();
    }
}

输出可发现:不写thread.yield()方法时,执行时间较短

线程的sleep()方法和yield()方法有什么区别?
① sleep()方法给其他线程运行机会时不考虑线程的优先级,因此会给低优先级的线程以运行的机会;yield()方法只会给相同优先级或更高优先级的线程以运行的机会;
② 线程执行sleep()方法后转入阻塞(blocked)状态,而执行yield()方法后转入就绪(ready)状态;
③ sleep()方法声明抛出InterruptedException,而yield()方法没有声明任何异常;
④ sleep()方法比yield()方法(跟操作系统CPU调度相关)具有更好的可移植性。

3、join()方法

主线程创建并启动了线程,如果子线程中要进行大量耗时运算,主线程往往将早于子线程结束之前结束。这时,如果主线程想等待子线程执行完成之后再结束,比如子线程处理一个数据,主线程要取得这个数据中的值,就要用到join()方法了。方法join()的作用是等待线程对象销毁。

public class MyThread4 extends Thread {
    public MyThread4(String name){
        super(name);
    }
    public void run() {
        for (int i = 0; i < 5; i++) {
            System.out.println(getName() + "  " + i);
        }
    }
    public static void main(String[] args) throws InterruptedException {
        // 启动子进程
        for (int i = 0; i < 10; i++) {
            if (i == 5) {
                MyThread4 th = new MyThread4("joined thread");
                th.start();
                th.join();
            }
        System.out.println(Thread.currentThread().getName() + "  " + i);
        }
    }
}

输出如下:


image.png
4、interrupt线程中断

public void interrupt(); 中断线程。
public static boolean interrupted(); 是一个静态方法,用于测试当前线程是否已经中断,并将线程的中断状态 清除。所以如果线程已经中断,调用两次interrupted,第二次时会返回false,因为第一次返回true后会清除中断状态。

五、守护线程

线程分为用户线程和守护线程,当用户线程结束,守护线程会被强制终止。GC就是守护线程,默认线程是用户线程,在启动前可设置为守护

参考:
http://www.importnew.com/21136.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,509评论 6 504
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,806评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,875评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,441评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,488评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,365评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,190评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,062评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,500评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,706评论 3 335
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,834评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,559评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,167评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,779评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,912评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,958评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,779评论 2 354

推荐阅读更多精彩内容

  • 线程的简介 几乎每种操作系统都支持进程的概念。进程就是在某种程度上相互隔离的、独立运行的程序。线程化是允许多个活动...
    小人物灌篮阅读 677评论 2 4
  • 前言 多线程并发编程是Java编程中重要的一块内容,也是面试重点覆盖区域,所以学好多线程并发编程对我们来说极其重要...
    嘟爷MD阅读 7,315评论 21 272
  • 本文主要讲了java中多线程的使用方法、线程同步、线程数据传递、线程状态及相应的一些线程函数用法、概述等。 首先讲...
    李欣阳阅读 2,454评论 1 15
  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 2,957评论 1 18
  • 写在前面的话: 这篇博客是我从这里“转载”的,为什么转载两个字加“”呢?因为这绝不是简单的复制粘贴,我花了五六个小...
    SmartSean阅读 4,730评论 12 45