多线程编程

一、创建线程

  • 1.继承 Thread 类,重写 run() 方法

    栗子:

    public class TestThread extends Thread {
    
      @Override
      public void run() {
          System.out.println("Hello World");
      }
    
      
      public static void main(String[] args) {
          TestThread t = new TestThread();
          t.start();
      }
     }
    

    注意啦!!!注意啦!!!注意啦!!!

    调用 start() 方法并不会立即执行多线程的代码,而是将该线程变为可运行状态,什么时候运行多线程代码又操作系统决定。

  • 2.实现 Runnable 接口,实现该接口的 run() 方法

    栗子:

    public class MyRunnable implements Runnable{
    
      @Override
      public void run() {
          System.out.println("Hello World");
      }
    
      public static void main(String[] args) {
          Thread t = new Thread(new MyRunnable());
          t.start();
      }
    }
    
  • 3.实现 Callable 接口,实现该接口的 call() 方法

    Callable 接口和 Runnable 接口功能类似,它是 Executor 框架的功能类,它比 Runnable 更强大的地方在于:

    • Callable 可以在任务接受后提供一个返回值,Runnable 无法提供这个功能
    • Callable 中的 call() 可以抛出异常,Runnable 的 run() 遇到异常线程就停止了,不能抛出
    • 运行 Callable 能拿到一个 Future 对象,这个对象表示异步计算的结果。通过调用 Future 对象的 get() 方法,就能获取线程的计算结果。
      栗子:
    public class MyCallable implements Callable<Integer>{
    
      @Override
      public Integer call() throws Exception {
          return 10*1093;
      }
    
      public static void main(String[] args) {
          MyCallable c = new MyCallable();
          ExecutorService executorService = Executors.newSingleThreadExecutor();
          Future<Integer> future = executorService.submit(c);
          
          try {
              System.out.println(future.get());
          }catch (Exception e) {
              e.printStackTrace();
          }
      }
    }
    

推荐使用第二种方法,直接实现 Runnable 接口。

二、volatile

  • vloatile 无法保证变量的原子性
  • 可见性
  • 有序性

可用于双重检查的单例模式

public class Singleton {

    private volatile static Singleton instance = null;
    
    public  static Singleton getInstance() {
        if(instance == null) {
            synchronized (Singleton.class) {
                if(instance == null) {
                    instance = new Singleton();
                }
            }
        }
        
        return instance;
    }
}

三、java 中的阻塞队列

  • 1.ArrayBlockingQueue: 由数组结构组成的有界阻塞队列

  • 2.LinkedBlockingQueue: 基于链表的有界阻塞队列

  • 3.PriorityBlockingQueue: 支持优先级排序的无界阻塞队列

  • 4.DelayQueue: 使用优先级队列实现的无界阻塞队列,支持延时获取元素

  • **5.SynchronousQueue: ** 不存储元素的阻塞队列

  • **6.LinkedTransferQueue: ** 由链表组成无界阻塞队列

  • **7.LinkedBlockingDqueue: ** 由链表组成的双向阻塞队列

四、线程池

  • 1.FixedThreadPool: 可重用固定线程数的线程池。
public static ExecutorService newFixedThreadPool(int nThreads) {
        return new ThreadPoolExecutor(nThreads, nThreads,
                                      0L, TimeUnit.MILLISECONDS,
                                      new LinkedBlockingQueue<Runnable>());
 }

这种线程池只有核心线程,并且数量是固定的,没有非核心线程。

工作流程:当执行 execute() 时,如果线程池的核心线程数还没达到 corePoolSize ,就创建核心线程执行任务,否则就将任务丢进无界的阻塞队列中,当有核心线程空闲时再去执行任务。

  • 2.CachedThreadPool: 根据需要创建线程的线程池
public static ExecutorService newCachedThreadPool() {
        return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
                                      60L, TimeUnit.SECONDS,
                                      new SynchronousQueue<Runnable>());
}

这种线程池没有核心线程,非核心线程是无界的。空闲线程等待新任务的最长时间是 60s。在这里用了 SynchronousQueue 不存储元素的阻塞队列,每个插入操作必须等待另一个线程的移除操作,同样任何一个移除操作都必须等待另一个线程的插入操作。

  • 3. SingleThreadExecutor:使用单个工作线程的线程池
public static ExecutorService newSingleThreadExecutor() {
        return new FinalizableDelegatedExecutorService
            (new ThreadPoolExecutor(1, 1,
                                    0L, TimeUnit.MILLISECONDS,
                                    new LinkedBlockingQueue<Runnable>()));
}

这种线程池能按照顺序逐一执行任务。

  • 4. ScheduleThreadPool: 实现定时和周期性任务的线程池
public ScheduledThreadPoolExecutor(int corePoolSize) {
        super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
              new DelayedWorkQueue());
}

这里使用的是优先级队列实现的无界阻塞队列,线程启动时需要延时获取任务,所以能实现延时或者周期执行。

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

推荐阅读更多精彩内容

  • 第5章 多线程编程 5.1 线程基础 5.1.1 如何创建线程 在java要创建线程,一般有==两种方式==:1)...
    AndroidMaster阅读 1,769评论 0 12
  • 从哪说起呢? 单纯讲多线程编程真的不知道从哪下嘴。。 不如我直接引用一个最简单的问题,以这个作为切入点好了 在ma...
    Mr_Baymax阅读 2,676评论 1 17
  • “1”:是指1个目标。有目标,一节课才有方向,譬如船行海中,目标即是灯塔。无目标,无方向。 “2”:是指2个过程:...
    大熊bear阅读 662评论 0 0
  • 哎,我的名字叫做林白芍。一个月前,我在上山炼晶的时候遇到了一个少年。他受了伤,我救了他,他就一直赖在了我的家。哎,...
    你可爱炸的朵莉阅读 175评论 2 0
  • 吸引力法则有三个过程,提出要 求,信念,接受。首先,你要对宇宙提出要求,你想住在什么样的房子里,想要什么样的车,什...
    展翅大鹏阅读 424评论 0 0