java初入多线程10

线程阻塞工具类 :LockSupport

  • LockSupport 是一个非常实用的线程阻塞工具, 可以在线程内任意位置,让线程阻塞。 这个类补充了 使用resume 导致线程无法继续执行的情况,和wait方法相比 ,不需要获得某个对象的锁 ,也不会抛出 中断异常。
public class LockSupportDemo {

    public static Object U = new Object() ;
    static ChangeObjectThread t1 = new ChangeObjectThread("t1");
    static ChangeObjectThread t2 = new ChangeObjectThread("t2");
    public static class ChangeObjectThread extends Thread{
        public ChangeObjectThread (String name){
            super.setName(name);
        }
        
        @Override
        public void run() {
            synchronized(U){
                System.out.println("in " + getName());
                LockSupport.park();
                
            }
        }
    }
    
    public static void main(String[] args) throws Exception {
        t1.start();
        Thread.sleep(100);
        t2.start();
        LockSupport.unpark(t1);
        LockSupport.unpark(t2);
        t1.join();
        t2.join();
    }
}
  • 在该代码中,我们使用的是方法park(),和unpark(); 该类为每个线程准备了一个许可, 如果许可可用,那么park()函数就会立即返回,消费掉这个许可,。 如果许可不可用,就会发生阻塞。这个优点是即使 unpark ()发生在park上 ,也能顺利的保证park操作正确执行。
  • 除了有定时阻塞的功能之外, park()还支持 中断影响,但是与其他中断函数不一样,park()函数 不会抛出中断异常, 只是默默返回。 但是我们可以用 Thread.interupted() 等方法获得中断标记。代码演示如下......
public class LockSupportIntDemo {

    public static Object U = new Object() ;
    static ChangeObjectThread t1 = new ChangeObjectThread("t1");
    static ChangeObjectThread t2 = new ChangeObjectThread("t2");
    
    public static class ChangeObjectThread extends Thread{
        public ChangeObjectThread (String name){
            super.setName(name);
        }
        
        @Override
        public void run() {
            synchronized(U){
                System.out.println("in " + getName());
                LockSupport.park();
                if(Thread.interrupted()){
                    System.out.println(getName() + "被中断了");
                }
            }
            System.out.println(getName() + "执行结束了");
        }
    }

    public static void main(String[] args) throws Exception {
        t1.start();
        Thread.sleep(100);
        t2.start();
        t1.interrupt();
        LockSupport.unpark(t2);
    }
}
临界

线程复用:线程池

  • 线程池是为了方便我们对创建的线程进行复用。节约了创建和销毁对象的时间。
线程池的使用
jdk 对线程池的使用。
  1. jdk 提供了一套Executor 框架, 本质就是线程池的使用。
Executor框架
  • 从图中可以看到ThreadPoolExecutor 表示一个线程池, Executors 类则扮演者线程池工厂的角色。 通过Executors 可以获得拥有特定功能的线程池。 我们在uml图上 ,可以看到Executor 接口 ,我们通过该接口 让Runnable 的对象可以被ThreadPoolExecutor 线程池调度。
  • Executor 框架提供了各种类型线程池。工厂方法如下:
  1. newFixedThreadPool(int nThreads); 该方法返回的是一个固定线程数量的线程池,该线程池的线程数量始终不变。 当有新任务提交过来的时候 ,线程池中有空闲线程,则立即执行。若没有 ,则会把新的任务放到一个队列中。等待有空闲线程的时候就会执行队列中的任务。
  2. newSingleThreadExecutor(): 方法会返回一个只有一个线程的线程池。若多余一个任务被提交到该线程池。任务同样会保存在任务队列中,等待线程空闲,按照先入先出的顺序 执行队列中的任务。
  3. newCachedThreadPool() 方法: 方法会返回一个根据实际情况调整线程数量的线程池。线程池的里面的线程数量不确定。在任务提交过来的时候,若有空闲线程 则直接复用空闲线程,若所有线程都在使用,那么会创建新的线程处理任务。所有线程结束后,将返回线程池进行复用。
  4. newSingThreadScheduleExecutor(): 该方法会返回一个ScheduledExecutorService对象,线程池大小为1, 这个线程池是在给定任务时间执行某任务的功能,或者是周期性执行任务
  5. newSheduledThreadPool(): 方法会返回ScheduledExecutorService对象,该线程池可以指定线程数量。
  • 对计划任务线程池的使用:newSheduledThreadPool():
  1. 主要方法如下 对该线程里面的设置。
/**下面两种方法 会在给定时间进行一次任务调度。**/
    public ScheduledFuture<?> schedule(Runnable command,
       long delay, TimeUnit unit);

    public <V> ScheduledFuture<V> schedule(Callable<V> callable,    long delay, TimeUnit unit);
/*** 该方法是对任务周期性的调度  fixRate 是任务调度频率是一致的**/
    public ScheduledFuture<?> scheduleAtFixedRate(Runnable command,   long initialDelay,  long period, TimeUnit unit);
/***同样 周期性的调度 FixedDelay 是在任务完成一周周期后 再经过某个时间长度 才会执行任务**/
    public ScheduledFuture<?> scheduleWithFixedDelay(Runnable command,long initialDelay,
 long delay,  TimeUnit unit);
//代码演示如下:
public class ScheduledExecutorServiceDemo {

    
    public static void main(String[] args) {
        ScheduledExecutorService  service = Executors.newScheduledThreadPool(10);
        service.scheduleAtFixedRate(new Runnable() {
            
            @Override
            public void run() {
                try {
                    Thread.sleep(1000);
                    System.out.println(System.currentTimeMillis() / 1000);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }, 0, 2, TimeUnit.SECONDS);
    }
}
每隔两秒
  1. 还有一种情况如果任务执行时间超过我们所规定的时间 ,会怎么样,是聚集还是? 答案就是 会持续的当当前任务执行完成后立即执行下面的任务,。还有一个如果任务出现异常,那么该任务会被中断 后续的任务也会被中断,那么需要我们最好异常处理。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,088评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,715评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,361评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,099评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,987评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,063评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,486评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,175评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,440评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,518评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,305评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,190评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,550评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,152评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,451评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,637评论 2 335

推荐阅读更多精彩内容