java多线程补充

 

 

java多线程

【狂神说Java】多线程详解哔哩哔哩bilibili

深入浅出线程Thread类的start()方法和run()方法 - 简书 (jianshu.com)

1.初见

java的线程是通过java.lang.Thread类来实现的。每个线程都是通过某个特定Thread对象所对应的方法run()来完成其操作的,方法run()称为线程体。通过调用Thread类的start()方法来启动一个线程。

 

狂神的两个小demo

demo01

第一个建立线程的方法

package com.kuang;

public class demo01 extends Thread{
   @Override
   //run方法可以理解为简单的一个函数并没有什么实际作用
   public void run() {
       for (int i = 0; i < 20; i++) {
           System.out.println("run-----" + i);
      }
  }

   public static void main(String[] args) {

       demo01 demo01 = new demo01();

       //start()方法是启动这个线程
       demo01.start();

       for (int i = 0; i < 2000; i++) {
           System.out.println("主线程---" + i);
      }


  }
}

demo02

用Runable接口来实现

package com.kuang;

public class demo02 implements Runnable{
   @Override
   public void run() {
       for (int i = 0; i < 20; i++) {
           System.out.println("线程" + i);
      }
  }

   public static void main(String[] args) {

       //线程代理
       demo02 demo02 = new demo02();
       Thread thread = new Thread(demo02);

       thread.start();
       for (int i = 0; i < 20; i++) {
           System.out.println("学习多线程 -------------" + i);
      }
  }
}

通过分析源码可以看出来

Thread继承了Runable的接口

image-20220408114207816

Runable接口实现了run()方法

但是你的Thread才是线程,不能直接用Runable直接测试run方法,还得通过Thread来进行测试

 

image-20220408113337056

demo03

发现并发问题

一共5张票,但是三个人抢的话,都会遍历一个遍

package com.kuang;

public class demo03 implements Runnable{

   /**
    * 通过买票发现并发问题
    */
   @Override
   public void run() {
       for (int i = 1; i <= 5; i++) {
           System.out.println(Thread.currentThread().getName() + "抢到了第" + i + "票");
      }
  }

   public static void main(String[] args) {
       demo03 demo03 = new demo03();

       new Thread(demo03,"小米").start();
       new Thread(demo03,"小红").start();
       new Thread(demo03,"黄牛").start();

  }
}

结果:

小红抢到了第1票
黄牛抢到了第1票
小米抢到了第1票
黄牛抢到了第2票
小红抢到了第2票
黄牛抢到了第3票
小米抢到了第2票
黄牛抢到了第4票
黄牛抢到了第5票
小红抢到了第3票
小米抢到了第3票
小红抢到了第4票
小米抢到了第4票
小红抢到了第5票
小米抢到了第5票

显然不符合要求

demo04

利用多线程解决龟兔赛跑

package com.kuang;

public class demo04 implements Runnable{
   @Override
   public void run() {
       for (int i = 0; i <= 100; i++) {
           System.out.println(Thread.currentThread().getName() + "跑了" + i + "米");

           if(win(i)){
               break;
          }
           if (Thread.currentThread().getName().equals("兔子") && i % 10 == 0 )
          {
               try {
                   //模拟兔子休息
                   Thread.sleep(20000);
              } catch (InterruptedException e) {
                   e.printStackTrace();
              }
          }

      }

  }

   public boolean win(int step)
  {
       if(step == 100)
      {
           System.out.println(Thread.currentThread().getName() + "获胜");
           return true;
      }
       return false;
  }

   public static void main(String[] args) {
       demo04 demo04 = new demo04();

       new Thread(demo04,"🐢").start();
       new Thread(demo04,"🐇").start();


  }
}

结果:

🐢跑了999米 🐢跑了1000米 🐇跑了996米 🐇跑了997米 🐇跑了998米 🐇跑了999米 🐇跑了1000米 🐢获胜

 

 

2.Lamda表达式

image-20220408150921758

演变过程!!

package com.kuang.lambda;

public class evolution {

   //2.2静态内部类
   static class Like2 implements llike {

       @Override
       public void lambda() {
           System.out.println("l like lambda2");
      }
  }

   public static void main(String[] args) {
       //2.3直接调用实现类

       Like1 like1 = new Like1();
       like1.lambda();

       Like2 like2 = new Like2();
       like2.lambda();

       //2.4局部内部类
       class Like3 implements llike {
           @Override
           public void lambda() {
               System.out.println("l like lambda3");
          }
      }

       Like3 like3 = new Like3();
       like3.lambda();


       //2.5匿名内部类,没有名称要借助接口,直接用接口书写
       llike llike4 = new llike() {
           @Override
           public void lambda() {
               System.out.println("l like lambda4");
          }
      };
       llike4.lambda();

       //2.6lambda表达式
       llike like = ()->{
           System.out.println("l like lambda5");
      };
       like.lambda();



  }

   //1.定义一个接口
   interface llike {
       void lambda();
  }

   //2.1实现接口
   static class Like1 implements llike {
       @Override
       public void lambda() {
           System.out.println("l like lambda1");
      }
  }
}

第二个例子最简化:

package com.kuang.lambda;

public class test {
   
   public static void main(String[] args) {
       ILove iLove = a-> System.out.println("i love" + a);

       iLove.love(2);
  }
}

interface ILove
{
   void love(int a);
}

3.静态代理

代理模式是常用的Java设计模式,它的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。

package com.kuang.static01;

public class marry {

   public static void main(String[] args) {
//对比多线程,也是Runnable target静态代理
       new Thread(() -> System.out.println("asdasd")).start();

       weddingCompany weddingCompany = new weddingCompany(new you());
       
       weddingCompany.Marry();
  }

}
interface marryy{
   void Marry();
}

class you implements marryy{

   @Override
   public void Marry() {
       System.out.println("结婚啦");
  }
}

class weddingCompany implements marryy{

   private marryy target;

   public weddingCompany (marryy target){
       this.target = target;
  }

   @Override
   public void Marry() {
       this.target.Marry();
  }
}

image-20220408163329268

4.线程方法

 

线程停止:

    package com.kuang.test;
   
   public class stop implements Runnable{
   
       private boolean st = true;
   
       @Override
       public void run() {
           int i = 0;
           while (st)
          {
               i++;
               System.out.println("线程" + i);
          }
      }
   
       public void stopp()
      {
           this.st = false;
      }
   
       public static void main(String[] args) {
   
           stop stop = new stop();
           new Thread(stop).start();
           for (int i = 0; i < 1000; i++) {
   
               System.out.println("main" + i);
               if(i == 900){
                   System.out.println("--------------停止了-------------");
                   stop.stopp();
              }
   
          }
      }
  }

线程休眠:

package com.kuang.test;

import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.logging.SimpleFormatter;

public class testSleep {

   //倒计时
   public static void tendown() throws InterruptedException {
       int num = 10;
       while (true)
      {
           Thread.sleep(1000);
           System.out.println(num);
           num --;
           if(num <= 0)break;
      }
  }

   public static void main(String[] args) throws InterruptedException {
       //tendown();

       Date date = new Date(System.currentTimeMillis());
       while(true)
      {
           int num = 10;
           num --;
           Thread.sleep(1000);
           if(num <= 0)break;
           System.out.println(new SimpleDateFormat("HH:mm:ss").format(date));
           date = new Date(System.currentTimeMillis());
      }

  }

}

线程礼让:

package com.kuang.test;

public class yield {

   public static void main(String[] args) {
       myYield myYield = new myYield();

       new Thread(myYield,"a").start();
       new Thread(myYield,"b").start();
  }

}
class myYield implements Runnable{

   @Override
   public void run() {
       System.out.println(Thread.currentThread().getName() + "开始执行");
       Thread.yield();
       System.out.println(Thread.currentThread().getName() + "结束执行");
  }
}

礼让可能成功,可能不成功

结果1:

b开始执行 a开始执行 b结束执行 a结束执行

结果2:

a开始执行 b开始执行 b结束执行 a结束执行

线程插队:

package com.kuang.test;

public class testJoin implements Runnable{
   @Override
   public void run() {
       for (int i = 0; i < 300; i++) {
           System.out.println("vip来了" + i);
      }
  }

   public static void main(String[] args) throws InterruptedException {

       testJoin testJoin = new testJoin();
       Thread thread = new Thread(testJoin);
       thread.start();
       for (int i = 0; i < 1000; i++) {
           if(i == 100)
          {
               thread.join();
          }
           System.out.println("main" + i);
      }
  }
}

值得一提的是,小于100的时候这两个是并发处理的,但是的等于100后就会先执行线程里面方法,主线程就是在等待中

5.线程状态观察

1.创建状态

2.启动状态

3.运行状态

4.阻塞状态

5.死亡状态

image-20220408195045662

 

 

 

posted @ 2022-04-08 20:20  爽爽子的秃头生活  阅读(112)  评论(0)    收藏  举报