Java 多线程基础

线程概念

  1. 与进程的关系
    进程是操作系统分配内存及其他系统资源的基本单元,线程之间相互独立。线程是操作系统分配CPU的基本单元,但共享内存和其他系统资源。线程切换速度比进程要快。
  2. 操作系统使用时间片轮转算法来给线程分配CPU,不同的系统策略不同。
  3. 线程有状态:新建,就绪,阻塞,运行,死亡。
    具体的线程调用start()就是就绪状态,等待JVM的调度,等获得CPU就进入运行状态,此时可能发生阻塞(资源不足)或者就绪(时间片用完)。


    java thread status
  4. 线程的优先级priority
    • 优先级范围为:1(THREAD.MIN_PRIORITY)-10(THREAD.MAX_PRIORITY),默认为5(THREAD.NORM_PRIORITY)
    • 线程创建时,子线程继承父线程的优先级,可以通过setPriority()来改变线程的优先级。

多线程的相关操作

继承Thread类创建线程类,重写run方法。

class myThread extends Thread{
public void run(){}
}

public class ThreadRun extends Thread{

    public void run(){
        System.out.println("other thread:"+Thread.currentThread().getName());
        for(int i=0;i<10;i++){
            System.out.println(i);
        }
    }
    public static void main(String[] args){
        System.out.println("main thread:"+Thread.currentThread().getName());
        ThreadRun t = new ThreadRun();
        ThreadRun s = new ThreadRun();
        t.start();
        s.start();
    }
}

实现Runnable接口创建线程类,覆盖run方法。

接口原型:public interface Runnable{ public void run(){} }
封装:class MyThread implements Runnable{ public void run(){} }

运行方法:
创建一个Runable对象:
Runnable r = new MyRunnable();
将Runable对象传给Thread对象
Thread t = new Thread(r);
启动多线程运行:
t.start();

class MyThread implements Runnable {
    @Override
    public void run() {
        System.out.println("mythread start:" + Thread.currentThread().getName());
        for (int i = 0; i < 10; i++) {
            System.out.println(i);
        }
        System.out.println("mythread end:" + Thread.currentThread().getName());
    }

}
public class RunnableThreadTest {

    public static void main(String[] args) {
        System.out.println("mainTHread start:" + Thread.currentThread().getName());
        Thread t = new Thread(new MyThread(), "mythread"); // create thread.
        t.start();
        System.out.println("mainTHread end:" + Thread.currentThread().getName());
    }
}

匿名线程类

注意匿名类的写法,愤青最后运行的是那个run方法。

public class RunWhatThread {
    public static void main(String[] args) {
        Thread th = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("runnable");

            }
        }) {
            @Override
            public void run() {
                //super.run();
                System.out.println("thread");
            }
        };
        th.start();
    }

}

通过线程工厂创建线程

接口原型:public interface ThreadFactory{ Thread newThread( Runnable r){} }

实现此接口,就可以使用工厂模式来创建线程,只需一条语句,使创建线程更方便。

import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.util.concurrent.ThreadFactory;

public class ThreadFactoryDemo {
    public static void main(String[] args) {
        MyThreadFactory factory = new MyThreadFactory("thread factory");
        factory.newThread(new Runnable() {
            @Override
            public void run() {
                System.out.println("in runnable");
            }
        }).start();
    }

}
class MyThreadFactory implements ThreadFactory {
    private int count;
    private String name;
    private List<String> stats;

    public MyThreadFactory(String name) {
        count = 0;
        this.name = name;
        stats = new ArrayList<>();
    }

    public Thread newThread(Runnable r) {
        Thread t = new Thread(r, name + "-thread-" + count);
        count++;
        stats.add(String.format("create thread %d with name %s on %s\n", t.getId(), t.getName(), new Date()));
        return t;
    }
}

后台Daemon线程

后台线程相对前台线程而言,当主线程结束时,后台线程也会自动结束。
默认情况下,创建的线程均为“前台”线程,需要调用setDeamon方法设置为“背景”线程:thread.setDaemon(true);

线程的暂停

  1. 静态方法 Thread.sleep(long mills),需捕获InterruptedException 异常
  2. 使用TimeUnit枚举类的相应方法,TimeUnit.SECONDS.sleep(long s)
import java.util.concurrent.TimeUnit;
public class ThreadSleep {

    public static void main(String[] args) {
        Thread t = new Thread(new Runnable() {
            private int count = 0;

            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    try {
                        //Thread.sleep(500); //sleep 单位是毫秒
                        TimeUnit.SECONDS.sleep(1);   //单位可以自定义,more convinent
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    count++;
                    System.out.println(count);
                }
            }
        });
        t.start();
    }

}

线程终止

  1. 当执行完毕,发生未捕获的异常,其他线程请求终止该线程时,线程会发生终止。
  2. 可以通过interrupt()向进程发送中断请求,此时内部的中断标志会被设置,通常需要定时检测这一标志。
import java.util.concurrent.TimeUnit;

public class PrimerGenerator extends Thread {

    @Override
    public void run() {
        long number = 0L;
        while (true) {
            if (isPrimer(number)) {
                System.out.println(number + " is a primer");
            }
            // 定时检查线程的中断标志
            if (isInterrupted()) {
                System.out.println("thread already is interrupted");
                return;
            }
            number++;
        }
    }

    private boolean isPrimer(long n) {
        if (n < 2) {
            return false;
        }
        for (int i = 2; i < n; i++) {
            if (n % i == 0)
                return false;
        }
        return true;
    }

    public static void main(String[] args) {
        Thread task = new PrimerGenerator();
        task.start();
        try {
            TimeUnit.SECONDS.sleep(5);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        task.interrupt();
    }
}

定时任务

使用Timer类的schedule方法可以定时执行一个任务。此任务是TimerTask类型的实例,注意TimerTask派生自Runnable。因此,会在另外一个线程中运行这个定时任务。

import java.util.Timer;
import java.util.TimerTask;

public class UseTimer {
    public static void main(String[] args) {
        //TimerTask定时任务,继承Runnable接口
        TimerTask task = new TimerTask() {
            private int count = 0;

            @Override
            public void run() {
                count++;
                System.out.println(count + ":invoked.");
            }
        };
        
        Timer timer = new Timer();
        //第一个参数是TimerTask表示待执行的任务,第二个参数是第一次调用的时间,第二个参数表示第一次调用之后,从第二次开始每隔多长的时间调用一次 run()方法。
        timer.schedule(task, 2000, 3000);
    }
}

线程组

  1. 将一批线程组做成统一的组管理,每个组有各自的名字,可有父子关系。
  2. ThreadGroup parent=new ThreadGroup(“parent”)
    ThreadGroup child=new ThreadGroup(parent,”Child”)
  3. 向线程组增加新线程
    Thread t1=new Thread(parent,”t1”)
  4. 线程有未捕获的异常,需要通过Thread.UncaughtExceptionHandler接口来处理, 此接口定义了以下方法:void uncaughtException(Thread t, Throwable e)
  5. 如果没有设定线程的未捕获异常处理对象,将会使用线程的ThreadGroup对象,它实现了Thread.UncaughtExceptionHandler接口。使用线程组可以统一管理未捕获异常。

public class ThreadGroupDemo {

    public static void main(String[] args) {
        ThreadGroup threadGroup = new ThreadGroup("group1"){
            public void uncaughtException(Thread t,Throwable e){
                System.out.println(t.getName()+":"+e.getMessage());
            }
        };

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

推荐阅读更多精彩内容

  • 前言 多线程并发编程是Java编程中重要的一块内容,也是面试重点覆盖区域,所以学好多线程并发编程对我们来说极其重要...
    嘟爷MD阅读 7,316评论 21 272
  • 进程:正在执行中的程序,其实是应用程序在内存中运行的那片空间。 线程:进程中的一个执行单元,负责进程中程序的执行。...
    七弦桐语阅读 466评论 2 7
  • 写在前面的话: 这篇博客是我从这里“转载”的,为什么转载两个字加“”呢?因为这绝不是简单的复制粘贴,我花了五六个小...
    SmartSean阅读 4,738评论 12 45
  • 纷繁复杂的世界里,我们常常熬夜,并习惯于寻找用各种各种的理由来让我们自己觉得我们并没有浪费时间,殊不知,浪费时间与...
    奇奇怪怪小技巧阅读 224评论 0 3
  • 我是一个很喜欢回忆的人,回忆过去,尤其喜欢在故园度过的为数不多的,但仿佛构建了我整个童年的旧时光。可我每次...
    闫小尧思密达阅读 487评论 0 2