范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文
国学影视

并发编程解惑之线程

  主要内容:  线程的分类  线程的状态  线程间通信  线程的封闭  异步线程  J.U.C 包和线程池  一、线程与进程
  进程是资源分配的最小单位,每个进程都有独立的代码和数据空间,一个进程包含 1 到 n 个线程。线程是 CPU 调度的最小单位,每个线程有独立的运行栈和程序计数器,线程切换开销小。
  Java 程序总是从主类的 main 方法开始执行,main 方法就是  Java 程序默认的主线程,而在 main 方法中再创建的线程就是其他线程。在  Java 中,每次程序启动至少启动 2 个线程。一个是 main 线程,一个是垃圾收集线程。每次使用  Java 命令启动一个  Java 程序,就相当于启动一个 JVM 实例,而每个 JVM 实例就是在操作系统中启动的一个进程。 二、线程的创建方式
  多线程可以通过继承或实现接口的方式创建。 2.1 直接继承 Thread 类实现多线程
  Thread 类是 JDK 中定义的用于控制线程对象的类,该类中封装了线程执行体 run() 方法。需要强调的一点是,线程执行先后与创建顺序无关。
  /**  * 类MyThread    */    public class MyThread extends Thread {     int nTime;     String strThread;     public MyThread(int nTime, String strThread) {         this.nTime = nTime;         this.strThread = strThread;    }     //线程执行体    public void run() {        while (true) {            try {                 System.out.println("Thread name:" + strThread + " ");     //线程睡眠,睡眠完成后继续执行                Thread.sleep(nTime);             } catch (Exception e) {                 e.printStackTrace();            }        }    }     //main方法是主线程    static public void main(String args[]) {    //额外创建了三个MyThread线程        MyThread aThread = new MyThread(1000, "aThread");         aThread.start();         MyThread bThread = new MyThread(2000, "bThread");         bThread.start();         MyThread cThread = new MyThread(3000, "cThread ");     //调用线程对象的start方法,线程会以多线程的方式并发执行,    //如果直接调用run方法,线程是直接执行普通方法,并不是并发运行        cThread.start();     }    } 2.2 通过 Runnable 接口实现多线程/**   * 类MyRunnable    */    public class MyRunnable implements  Runnable{         int nTime;         String strThread;         public MyRunnable(int nTime, String strThread) {             this.nTime = nTime;             this.strThread = strThread;         }     @Override    public void run() {        while (true) {            try {                 System.out.println("Thread name:" + strThread + " ");     //线程睡眠,睡眠完成后继续执行                Thread.sleep(nTime);             } catch (Exception e) {                 e.printStackTrace();             }        }    }         //main方法是主线程        static public void main(String args[]) {             //对象实例的运行线程            Thread aRunnable = new Thread(new MyRunnable(1000, "aRunnable"));             aRunnable.start();             Thread bRunnable = new Thread(new MyRunnable(2000, "bRunnable"));             bRunnable.start();             Thread cRunnable = new Thread(new MyRunnable(3000, "cRunnable"));             cRunnable.start();         }     }
  通过 Runnable 方式创建线程相比通过继承 Thread 类创建线程的优势是避免了单继承的局限性。若一个 boy 类继承了 person 类,boy 类就无法通过继承 Thread 类的方式来实现多线程。
  使用 Runnable 接口创建线程的过程:先是创建对象实例 MyRunnable,然后将对象 My Runnable 作为 Thread 构造方法的入参,来构造出线程。对于 new Thread(Runnable target) 创建的使用同一入参目标对象的线程,可以共享该入参目标对象 MyRunnable 的成员变量和方法,但 run() 方法中的局部变量相互独立,互不干扰。
  上面代码是 new 了三个不同的 My Runnable 对象,如果只想使用同一个对象,可以只 new 一个 MyRunnable 对象给三个 new Thread 使用。
  实现 Runnable 接口比继承 Thread 类所具有的优势: 适合多个相同的程序代码的线程去处理同一个资源(使用同一个目标对象),一份代码,多份数据,代码和数据分离 可以避免  Java 中的单继承的限制 线程池只能放入实现 Runable 或 callable 类线程,不能直接放入继承 Thread 类的线程 三、线程的状态
  线程有新建、可运行、阻塞、等待、定时等待、死亡 6 种状态。一个具有生命的线程,总是处于这 6 种状态之一。 每个线程可以独立于其他线程运行,也可和其他线程协同运行。线程被创建后,调用 start() 方法启动线程,该线程便从新建态进入就绪状态。 3.1 新建状态
  NEW 状态(新建状态) 实例化一个线程之后,并且这个线程没有开始执行,这个时候的状态就是 NEW 状态: Thread thread = new Thread();   System.out.println(thread.getState());   // NEW 状态(thread.getState()获取线程状态)。  3.2 就绪状态
  RUNNABLE 状态(就绪状态): static public void main(String args[]) {  Thread thread = new Thread(   // 创建runnable入参对象 new Runnable() {   @Override  public void run()  {   System.out.println(thread.getState());  }  },   // 线程名称  "RUNNABLE-Thread");   // 线程启动后就是runnable状态,正在运行的状态 thread.start();  } 3.3 阻塞状态
  阻塞状态有 3 种: 等待阻塞:运行的线程执行了 wait() 方法,释放持有的锁,JVM 会把该线程放入等待队列(等待池)中。 同步阻塞:运行线程获取同步锁时,而同步锁被其他线程持有,则 JVM 会把线程放入同步队列(锁池)中。 其他阻塞:运行的线程执行 sleep() 或 join() 方法,JVM 会把该线程置为阻塞状态。当 sleep() 状态超时、join() 等待的线程结束了或者超时,线程会重新转入就绪状态。当运行线程发出了 I/O 请求时,一样会变成阻塞状态,直到 I/O 处理完毕,才变成就绪状态。 3.4 等待状态
  如果一个线程调用了一个对象的 wait 方法, 那么这个线程就会处于等待状态(waiting 状态)直到另外一个线程调用这个对象的 notify 或者 notifyAll 方法后才会解除这个状态。 static public void main(String args[]) {  final Object lock = new Object();  Thread threadA = new Thread(new Runnable() {      @Override     public void run() {          synchronized (lock) {             try {  //当前线程释放持有的lock锁,并等待lock锁的分配                 lock.wait(); //等待结束                 System.out.println("wait over");              } catch (InterruptedException e) {                 e.printStackTrace();             }         }     } }, "WAITING-Thread-A");  Thread threadB = new Thread(new Runnable() {      @Override     public void run() {          synchronized (lock) {  //当前线程唤醒等待队列里的线程,让线程进入就绪队列申请lock锁 //但是本线程仍然继续拥有lock这个同步锁,本线程仍然继续执行              lock.notifyAll();             try {  // 当前线程睡眠2000毫秒                 Thread.sleep(2000);              } catch (InterruptedException e) {                 e.printStackTrace();             }         }     } }, "WAITING-Thread-B");  threadA.start();  threadB.start();  } 3.5 终结状态
  run() 里的代码执行完毕后,线程进入终结状态(TERMINATED 状态)。 3.6 小结
  线程状态有 6 种:新建、可运行、阻塞、等待、定时等待、死亡。
  四、线程常用的 APIThread.currentThread():获取当前线程 thread.isAlive():某个线程实例是否存活。 Thread.sleep():sleep 方法是 static 方法,线程类和线程实例调用,效果一样 thread.interrupt():将某个线程的中断标志位设置为 true,并没有中断线程,它只是向线程发送一个中断信号。 Thread.interrupted():判断当前线程是否中断,如果发现是 true,表明线程是中断,返回 true,返回前将标志位设置为 false thread.isInterrupted():判断线程是否中断,不改变标志位 Object.wait():让获得 Object 锁的 thread 线程等待 Object.notify():唤醒获得 Object 锁的 thread 线程 Object.wait() 与 Object.notify() 必须要与 synchronized (Object) 一起使用 thread.join():等待 thread 线程终止
  我们看下 join 方法的使用:     public class JoinTest {          public static void main(String[] args) {              System.out.println(Thread.currentThread().getName() + "主线程运行开始!");              MyThreads m1 = new MyThreads("A");              MyThreads m2 = new MyThreads("B");              m1.start();              m2.start();              try {                  //等待A线程运行结束后,main线程和B线程才能继续执行                 m1.join();              } catch (InterruptedException e) {                 e.printStackTrace();             }             try {                  //等待B线程运行结束后,main线程才能继续执行                 m2.join();              } catch (InterruptedException e) {                 e.printStackTrace();             }              System.out.println(Thread.currentThread().getName() + "主线程运行结束!");          }      }      class MyThreads extends Thread {          private String name;          public MyThreads(String name) {             super(name);             this.name = name;         }          public void run() {              System.out.println(Thread.currentThread().getName() + " 子线程运行开始!");              try {                  sleep((int) Math.random() * 10);              } catch (InterruptedException e) {                 e.printStackTrace();             }              System.out.println(Thread.currentThread().getName() + " 子线程运行结束!");         }      }
  运行结果:     main主线程运行开始!     A 子线程运行开始!     B 子线程运行开始!     A 子线程运行结束!     B 子线程运行结束!     main主线程运行结束! thread.yield(): yield() 是让当前运行中的线程回到就绪状态(可运行状态),以允许具相同优先级的其他线程获得运行机会。使得相同优先级的线程之间能适当的轮流执行。但实际中无法保证 yield() 达到让步目的,让步的线程还是有可能被调度到继续执行。
  我们来看下 yield 方法的使用:     public class YieldTest {          public static void main(String[] args) {              ThreadInstance y1 = new ThreadInstance("A");              ThreadInstance y2 = new ThreadInstance("B");              y1.start();              y2.start();          }      }      class ThreadInstance extends Thread {          public ThreadInstance(String name) {              super(name);          }          @Override         public void run() {              for (int i = 1; i <= 5; i++) {                  System.out.println("" + this.getName() + "-----" + i);                  // 当i等于10时,就把对CPU的占用释放,让自己或者同优先级的其他线程运行,谁抢到CPU时间片谁就执行                 if (i == 3) {                      this.yield();                  }             }         }      }
  运行结果:     B-----1     B-----2     B-----3     B-----4     B-----5     A-----1     A-----2     A-----3     A-----4     A-----5 五、线程间的通信
  线程与线程之间是无法直接通信的,A 线程无法直接通知 B 线程,Java 中线程之间交换信息是通过共享的内存来实现的,控制共享资源的读写的访问,使得多个线程轮流执行对共享数据的操作,线程之间通信是通过对共享资源上锁或释放锁来实现的。线程排队轮流执行共享资源,这称为线程的同步。 5.1 线程的通信方式
  Java 提供了很多同步操作(也就是线程间的通信方式),同步可使用 synchronized 关键字、Object 类的 wait/notifyAll 方法、ReentrantLock 锁、无锁同步 CAS 等方式来实现。 5.2 ReentrantLock 锁
  ReentrantLock 是 JDK 内置的一个锁对象,用于线程同步(线程通信),需要用户手动释放锁。 public class ReentrantLockTest { // 创建锁对象     private ReentrantLock lock = new ReentrantLock();      public void work() {         lock.lock();//对下面的操作上锁,只有拿到锁,才能继续执行         try {             System.out.println(Thread.currentThread().getName() );             try {                 Thread.sleep(6000);             } catch (InterruptedException e) {              }         } finally {         // 就算出现异常,也确保能释放锁             lock.unlock();         }     }      public static void main(String[] args) {         ReentrantLockTest reentrantLockTest = new ReentrantLockTest();         Thread thread1 = new Thread(new Runnable() {             @Override             public void run() {             // 创建线程1执行同步方法work,需要拿到锁才能执行try里的代码                 reentrantLockTest.work();             }         });         Thread thread2 = new Thread(new Runnable() {             @Override             public void run() {            // 创建线程2执行同步方法work,需要拿到锁才能执行try里的代码                 reentrantLockTest.work();             }         });         //启动两个线程         thread1.start();         thread2.start();     }  }
  运行结果: Thread-0  // 隔了6秒钟 输入下面 Thread-1
  这表明同一时间段只能有 1 个线程执行 work 方法,因为 work 方法里的代码需要获取到锁才能执行,这就实现了多个线程间的通信,线程 0 获取锁,先执行,线程 1 等待,线程 0 释放锁,线程 1 继续执行。 5.3 synchronized 内置锁
  synchronized 是一种语法级别的同步方式,称为内置锁。该锁会在代码执行完毕后由 JVM 释放。 public class SynchronizedTest {  //synchronized放在返回值后,对work方法上锁,锁住的是SynchronizedTest该类的对象实例     public synchronized void work() {         try {              nextLock();              Thread.sleep(6000);          } catch (InterruptedException e) {              Thread.currentThread().interrupt();         }     }      public static void main(String[] args) {          SynchronizedTest reentrantLockTest = new SynchronizedTest();          Thread thread1 = new Thread(new Runnable() {             @Override             public void run() {                  reentrantLockTest.work();             }         });         Thread thread2 = new Thread(new Runnable() {             @Override             public void run() {                  reentrantLockTest.work();             }         });          //启动两个线程,这两个线程都调用同一个对象实例reentrantLockTest的work方法          //这两次调用需要竞争的是同一个锁对象reentrantLockTest,故能实现线程间的同步          thread1.start();          thread2.start();     }  }
  输出结果跟 ReentrantLock 一样。 Thread-0 等待了6秒后 Thread-1 5.4 wait/notifyAll 方式
  Java 中的 Object 类默认是所有类的父类,该类拥有 wait、 notify、notifyAll 方法,其他对象会自动继承 Object 类,可调用 Object 类的这些方法实现线程间的通信。 public class WaitNotifyAllTest {      public synchronized void doWait() {         //进入到方法内部,表明线程获取到了锁,锁就是WaitNotifyAllTest对应的对象实例,this代表该实例         try {          // 此时释放了锁,其他线程可获得锁并执行              this.wait();         } catch (InterruptedException e) {             Thread.currentThread().interrupt();         }      }      public synchronized void doNotify() {      //进入到方法,表明获取到了锁         try {              Thread.sleep(6000);              //通知其他线程竞争锁,此时锁还未被释放             this.notifyAll();          } catch (InterruptedException e) {              Thread.currentThread().interrupt();         }          //方法结束,锁被自动释放,其他线程终于可以申请锁了     }      public static void main(String[] args) {          WaitNotifyAllTest waitNotifyAllTest = new WaitNotifyAllTest();          Thread thread1 = new Thread(new Runnable() {             @Override             public void run() {                  waitNotifyAllTest.doWait();             }         });         Thread thread2 = new Thread(new Runnable() {             @Override             public void run() {                  waitNotifyAllTest.doNotify();             }         });          thread1.start();          thread2.start();     }  } 5.5 无锁同步 CAS
  除了可以通过锁的方式来实现通信,还可通过无锁的方式来实现,无锁同 CAS(Compare-and-Swap,比较和交换)的实现,需要有 3 个操作数:内存地址 V,旧的预期值 A,即将要更新的目标值 B,当且仅当内存地址 V 的值与预期值 A 相等时,将内存地址 V 的值修改为目标值 B,否则就什么都不做。
  我们通过计算器的案例来演示无锁同步 CAS 的实现方式,非线程安全的计数方式如下: /**  * 非线程安全计数器  */ private void count() {     i++; }
  线程安全的计数方式如下: //基于CAS实现线程安全的计数器方法safeCount public class CountAtomic {      private AtomicInteger atomic = new AtomicInteger(0);      private int i = 0;       /**      * 使用CAS实现线程安全计数器      */     private void safeCount() {         for (;;) {  //获取原子类实例值0(初始值为0)             int i = atomic.get();  //实例值0和预期值0相同,设置实例值为1             boolean sum = atomic.compareAndSet(i, ++i);             if (sum) {  //无限循环,直到sum值为true,设置成功,退出循环,设置失败,不断循环判断,不断重试,类似自旋锁                 break;             }         }     }  /**  * 非线程安全计数器  */ private void count() {     i++; }  //调用main方法,测试两种计数方法     public static void main(String[] args) {          final CountAtomic cas = new CountAtomic();          List ts = new ArrayList();//初始值为600          for (int j = 0; j < 100; j++) {             Thread t = new Thread(new Runnable() {                 @Override                 public void run() {                      for (int i = 0; i < 1000; i++) {                          cas.count();                          cas.safeCount();                      }                 }             });  //将线程加入队列             ts.add(t);         } //启动所有线程         for (Thread t : ts) {             t.start();         }          // 等待所有线程执行完成         for (Thread t : ts) {             try {  //在线程A中调用了线程B的Join()方法,是A线程进入wait状态,直到线程B执行完毕后,才会继续执行线程A //这是让main线程进入等待,直到所有t线程执行完,才继续执行main线程                 t.join();             } catch (InterruptedException e) {                 e.printStackTrace();             }         }          System.out.println("非线程安全累加结果"+cas.i);         System.out.println("线程安全累加结果"+cas.atomic.get());     }  }
  运行结果: 非线程安全累加结果98636 线程安全累加结果100000
  线程安全累加的结果才是正确的,非线程安全会出现少计算值的情况。JDK 1.5 开始,并发包里提供了原子操作的类,AtomicBoolean 用原子方式更新的 boolean 值,AtomicInteger 用原子方式更新 int 值,AtomicLong 用原子方式更新 long 值。 AtomicInteger 和 AtomicLong 还提供了用原子方式将当前值自增 1 或自减 1 的方法,在多线程程序中,诸如 ++i 或 i++ 等运算不具有原子性,是不安全的线程操作之一。 通常我们使用 synchronized 将该操作变成一个原子操作,但 JVM 为此种操作提供了原子操作的同步类 Atomic,使用 AtomicInteger 做自增运算的性能是 ReentantLock 的好几倍。 六、J.U.C 包
  上面我们都是使用底层的方式实现线程间的通信的,但在实际的开发中,我们应该尽量远离底层结构,使用封装好的 API,例如 J.U.C 包(java.util.concurrent,又称并发包)下的工具类 CountDownLath、CyclicBarrier、Semaphore,来实现线程通信,协调线程执行。 6.1 闭锁 CountDownLatch
  CountDownLatch 能够实现线程之间的等待,CountDownLatch 用于某一个线程等待若干个其他线程执行完任务之后,它才开始执行。
  CountDownLatch 类只提供了一个构造器:    public CountDownLatch(int count) {  };  //参数count为计数值
  CountDownLatch 类中常用的 3 个方法: await() 方法:调用 await() 方法的线程会被挂起,它会等待直到 CountDownLatch 的 count 值为 0 才继续执行。   public void await() throws InterruptedException { }; 带时间的 await() 方法,等待一定的时间后 CountDownLatch 的 count 值还没变为 0 的话就会继续执行。   public boolean await(long timeout, TimeUnit unit) throws InterruptedException { }; 将 CountDownLatch 的 count 值减 1。   public void countDown() { }; public class CountDownLatchTest {      public static void main(String[] args) {          final CountDownLatch latch = new CountDownLatch(2);          new Thread() {             public void run() {                 try {                     System.out.println("子线程" + Thread.currentThread().getName() + "正在执行");                      Thread.sleep(3000);                      System.out.println("子线程" + Thread.currentThread().getName() + "执行完毕");                      latch.countDown();                  } catch (InterruptedException e) {                      e.printStackTrace();                 }             }              ;         }.start();          new Thread() {             public void run() {                 try {                      System.out.println("子线程" + Thread.currentThread().getName() + "正在执行");                      Thread.sleep(2000);                      System.out.println("子线程" + Thread.currentThread().getName() + "执行完毕");                      latch.countDown();                  } catch (InterruptedException e) {                      e.printStackTrace();                 }             }              ;         }.start();          try {              System.out.println("main线程等待2个子线程执行完毕");              latch.await();              System.out.println("2个子线程已经执行完毕");              System.out.println("main线程继续执行");          } catch (InterruptedException e) {              e.printStackTrace();         }     }  }
  运行结果: main线程等待2个子线程执行完毕 子线程Thread-1正在执行 子线程Thread-0正在执行 子线程Thread-1执行完毕 子线程Thread-0执行完毕 2个子线程已经执行完毕 main线程继续执行 6.2 循环栅栏 CyclicBarrier
  CyclicBarrier 字面意思循环栅栏,通过它可以让一组线程等待至某个状态之后再全部同时执行。当所有等待线程都被释放以后,CyclicBarrier 可以被重复使用,所以有循环之意。
  相比 CountDownLatch,CyclicBarrier 可以被循环使用,而且如果遇到线程中断等情况时,可以利用 reset() 方法,重置计数器,CyclicBarrier 会比 CountDownLatch 更加灵活。
  CyclicBarrier 提供 2 个构造器:     public CyclicBarrier(int parties, Runnable barrierAction) {      }      public CyclicBarrier(int parties) {      }
  上面的方法中,参数 parties 指让多少个线程或者任务等待至 barrier 状态;参数 barrierAction 为当这些线程都达到 barrier 状态时会执行的内容。
  CyclicBarrier 中最重要的方法 await 方法,它有 2 个重载版本。下面方法用来挂起当前线程,直至所有线程都到达 barrier 状态再同时执行后续任务。     public int await() throws InterruptedException, BrokenBarrierException {       };
  而下面的方法则是让这些线程等待至一定的时间,如果还有线程没有到达 barrier 状态就直接让到达 barrier 的线程执行任务。     public int await(long timeout, TimeUnit unit) throws InterruptedException, BrokenBarrierException, TimeoutException {       };      public class CyclicBarrierTest {         public static void main(String[] args) {             Random random = new Random();             CyclicBarrier cyclicBarrier = new CyclicBarrier(5);         //启动5个线程,对应cyclicBarrier设置的让5个线程等待至barrier状态             for (int i = 0; i < 5; i++) {                 new Thread(new Runnable() {                     @Override                    public void run() {                         int secs = random.nextInt(5);                         System.out.println("线程" + Thread.currentThread().getName() + "正在写入数据");                        try {                             //以睡眠来模拟写入数据操作                             Thread.sleep(secs * 1000);                             System.out.println("线程" + Thread.currentThread().getName() + "写入数据完毕,等待其他线程写入完毕");                             cyclicBarrier.await();                         } catch (InterruptedException e) {                             e.printStackTrace();                         } catch (BrokenBarrierException e) {                             e.printStackTrace();                        }                         System.out.println("所有线程写入完毕,继续处理其他任务");                    }                 }).start();            }        }      }
  运行结果:     线程Thread-0正在写入数据     线程Thread-3正在写入数据     线程Thread-4正在写入数据     线程Thread-1正在写入数据     线程Thread-2正在写入数据     线程Thread-3写入数据完毕,等待其他线程写入完毕     线程Thread-2写入数据完毕,等待其他线程写入完毕     线程Thread-0写入数据完毕,等待其他线程写入完毕     线程Thread-1写入数据完毕,等待其他线程写入完毕     线程Thread-4写入数据完毕,等待其他线程写入完毕     所有线程写入完毕,继续处理其他任务     所有线程写入完毕,继续处理其他任务     所有线程写入完毕,继续处理其他任务     所有线程写入完毕,继续处理其他任务     所有线程写入完毕,继续处理其他任务
  CyclicBarrier 用于一组线程互相等待至某个状态,然后这一组线程再同时执行,CountDownLatch 是不能重用的,而 CyclicBarrier 可以重用。 6.3 信号量 Semaphore
  Semaphore 类是一个计数信号量,它可以设定一个阈值,多个线程竞争获取许可信号,执行完任务后归还,超过阈值后,线程申请许可信号时将会被阻塞。Semaphore 可以用来 构建对象池,资源池,比如数据库连接池。
  假如在服务器上运行着若干个客户端请求的线程。这些线程需要连接到同一数据库,但任一时刻只能获得一定数目的数据库连接。要怎样才能够有效地将这些固定数目的数据库连接分配给大量的线程呢?
  给方法加同步锁,保证同一时刻只能有一个线程去调用此方法,其他所有线程排队等待,但若有 10 个数据库连接,也只有一个能被使用,效率太低。另外一种方法,使用信号量,让信号量许可与数据库可用连接数为相同数量,10 个数据库连接都能被使用,大大提高性能。
  public class SemaphoreDemo {      // 请求总数(客户端请求)     public static int clientTotal = 10;      // 同时并发执行的线程数(服务端连接数)     public static int threadTotal = 2;      public static void main(String[] args) throws Exception {  //      创建缓存线程池         ExecutorService executorService = Executors.newCachedThreadPool();  //      创建许可数和线程数一样,表示最多2个线程同时运行         final Semaphore semaphore = new Semaphore(threadTotal);  //      闭锁countDownLatch的计数设置为10         final CountDownLatch countDownLatch = new CountDownLatch(clientTotal);          for (int i = 0; i < clientTotal; i++) {              final int count = i;              executorService.execute(() -> {                  try {  //                  申请许可                     semaphore.acquire();  //                  睡眠1秒(实际中为具体业务代码)                     resolve(count);  //                  释放许可。使得每隔一秒就有两个线程执行                     semaphore.release();                  } catch (Exception e) {                      log.error("exception", e);                 }  //              for循环每执行一次,计数器就减1                 countDownLatch.countDown();             });         }  //      main线程一直阻塞,直到countDownLatch计数为0         countDownLatch.await();  //       关闭线程池         executorService.shutdown();      }      private static void resolve(int i) throws InterruptedException {          Thread.sleep(1000);      }  }
  上面三个工具类是 J.U.C 包的核心类,J.U.C 包的全景图就比较复杂了:
  J.U.C 包(java.util.concurrent)中的高层类(Lock、同步器、阻塞队列、Executor、并发容器)依赖基础类(AQS、非阻塞数据结构、原子变量类),而基础类是通过 CAS 和 volatile 来实现的。我们尽量使用顶层的类,避免使用基础类 CAS 和 volatile 来协调线程的执行。J.U.C 包其他的内容,在其他的篇章会有相应的讲解。 七、异步线程和线程封闭7.1 异步线程
  Future 是一种异步执行的设计模式,类似 ajax 异步请求,不需要同步等待返回结果,可继续执行代码。使 Runnable(无返回值不支持上报异常)或 Callable(有返回值支持上报异常)均可开启线程执行任务。但是如果需要异步获取线程的返回结果,就需要通过 Future 来实现了。
  Future 是位于 java.util.concurrent 包下的一个接口,Future 接口封装了取消任务,获取任务结果的方法。 public interface Future {  //如果取消任务成功则返回true,如果取消任务失败则返回false。  //参数mayInterruptIfRunning表示是否允许取消正在执行的任务,true为允许    boolean cancel(boolean mayInterruptIfRunning);  //任务是否被取消成功,若是,返回true     boolean isCancelled();  //任务是否已经完成,若是,返回true     boolean isDone();  //阻塞获取任务执行结果,一直去获取,直到任务执行完毕返回     V get() throws InterruptedException, ExecutionException;  //获取执行结果,超时后,未获取到结果,就返回null     V get(long timeout, TimeUnit unit)          throws InterruptedException, ExecutionException, TimeoutException; }
  在 Java 中,一般是通过继承 Thread 类或者实现 Runnable 接口来创建多线程, Runnable 接口不能返回结果,JDK 1.5 之后,Java 提供了 Callable 接口来封装子任务,Callable 接口可以获取返回结果。我们使用线程池提交 Callable 接口任务,将返回 Future 接口添加进 ArrayList 数组,最后遍历 FutureList,实现异步获取返回值。
  public class FutureDesign {      public static void main(String[] args) {          Long start = System.currentTimeMillis();          ExecutorService exs = Executors.newFixedThreadPool(6);         try {              //存储返回结果             List list = new ArrayList();              List> futureList = new ArrayList>();              //提交6个任务,每个任务返回一个Future对象,再加入futureList             for (int i = 0; i < 6; i++) {                  futureList.add(exs.submit(new CallableTask(i + 1)));             }              Long getResultStart = System.currentTimeMillis();              //2.结果归集,用迭代器遍历futureList,高速轮询(模拟并发访问),任务完成就移除             while (futureList.size() > 0) {                  Iterator> iterable = futureList.iterator();                  //如果下一个元素存在                 while (iterable.hasNext()) {                      //获取下一个元素,future对象                     Future future = iterable.next();                      //任务完成后或者被取消后                     if (future.isDone() && !future.isCancelled()) {                          //获取结果                         Integer i = future.get();                          list.add(i);                          //任务完成,可移除任务                         iterable.remove();                      } else {                          Thread.sleep(1);//避免CPU高速运转,休息1毫秒,CPU运行速度是纳秒级别                     }                 }             }              System.out.println("list=" + list);              System.out.println("总耗时=" + (System.currentTimeMillis() - start) + "毫秒" + ",取结果归集耗时=" + (System.currentTimeMillis() - getResultStart) + "毫秒");          } catch (Exception e) {              e.printStackTrace();          } finally {              exs.shutdown();          }     }      //回调方法     static class CallableTask implements Callable {          Integer i;          public CallableTask(Integer i) {             super();             this.i = i;         }          @Override         public Integer call() throws Exception {              Thread.sleep(3000);//任务耗时3秒              System.out.println("task线程:" + Thread.currentThread().getName() + "任务i=" + i + ",完成!");             return i;         }     }  }
  运行结果: task线程:pool-1-thread-5任务i=5,完成! task线程:pool-1-thread-2任务i=2,完成! task线程:pool-1-thread-4任务i=4,完成! task线程:pool-1-thread-6任务i=6,完成! task线程:pool-1-thread-3任务i=3,完成! task线程:pool-1-thread-1任务i=1,完成! list=[2, 4, 5, 6, 1, 3] 总耗时=3013毫秒,取结果归集耗时=3003毫秒
  上面就是异步线程执行的调用过程,实际开发中用得更多的是使用现成的异步框架来实现异步编程,如 RxJava,有兴趣的可以继续去了解,通常异步框架都是结合远程 HTTP 调用 Retrofit 框架来使用的,两者结合起来用,可以避免调用远程接口时,花费过多的时间在等待接口返回上。 7.2 线程封闭
  线程封闭是通过本地线程 ThreadLocal 来实现的,ThreadLocal 是线程局部变量(local vari able),它为每个线程都提供一个变量值的副本,每个线程对该变量副本的修改相互不影响。
  在 JVM 虚拟机中,堆内存用于存储共享的数据(实例对象),也就是主内存。Thread Local .set()、ThreadLocal.get() 方法直接在本地内存(工作内存)中写和读共享变量的副本,而不需要同步数据,不用像 synchronized 那样保证数据可见性,修改主内存数据后还要同步更新到工作内存。
  Myabatis、hibernate 是通过 threadlocal 来存储 session 的,每一个线程都维护着一个 session,对线程独享的资源操作很方便,也避免了线程阻塞。
  ThreadLocal 类位于 Thread 线程类内部,我们分析下它的源码: public class Thread implements Runnable {      ThreadLocal.ThreadLocalMap threadLocals = null;       public T get() {          // 得到当前线程         Thread t = Thread.currentThread();          // 获取当前线程的ThreadLocalMap容器         ThreadLocalMap map = getMap(t);          if (map != null) {              //容器存在,找到当前线程对应的键值对Entry             ThreadLocalMap.Entry e = map.getEntry(this);              if (e != null) {                 @SuppressWarnings("unchecked")                  //获取要返回的值                 T result = (T)e.value;                  return result;             }         }           // 不存在ThreadLocalMap容器,就初始化一个ThreadLocalMap         return setInitialValue();     }      private T setInitialValue() {          // 初始化ThreadLocalMap         T value = initialValue();          Thread t = Thread.currentThread();          ThreadLocalMap map = getMap(t);          if (map != null)          // 实例化ThreadLocalMap之后,将key为当前线程对象,value为null的初始值设置到Map中             map.set(this, value);          else             createMap(t, value);          return value;     }      public void set(T value) {         // 找到当前线程的ThreadLocalMap,设置对应的值,         Thread t = Thread.currentThread();          ThreadLocalMap map = getMap(t);          if (map != null)              map.set(this, value);          else//找不到ThreadLocalMap,创建ThreadLocalMap容器              createMap(t, value);     }      static class ThreadLocalMap {  //ThreadLocalMap容器中存放的就是键值对Entry,Entry的KEY就是ThreadLocal(当前线程),VALUE就是值(共享变量值)。         static class Entry extends WeakReference> {               Object value;              Entry(ThreadLocal<?> k, Object v) {                  super(k);                 value = v;             }         } }
  ThreadLocal 和 Synchonized 都用于解决多线程并发访问的问题,访问多线程共享的资源时,Synchronized 同步机制采用了以时间换空间的方式,提供一份变量让多个线程排队访问,而 ThreadLocal 采用了以空间换时间的方式,提供每个线程一个变量,实现数据隔离。
  ThreadLocal 可用于数据库连接 Connection 对象的隔离,使得每个请求线程都可以复用连接而又相互不影响。 public class ConnectionManager {          private static ThreadLocal connections = new ThreadLocal() {          @Override         protected Connection initialValue() {              Connection conn = null;              try {                  conn = DriverManager.getConnection(                         "jdbc:mysql://localhost:3306/shop", "name","password");              } catch (SQLException e) {                  e.printStackTrace();              }              return conn;         }     };      public static Connection getConnection() {          return connections.get();     }      public static void setConnection(Connection conn) {          connections.set(conn);     }  } 7.3 Java 的引用和内存泄漏
  在  Java 里面,存在强引用、弱引用、软引用、虚引用。我们主要来了解下强引用和弱引用:  A a = new A();   B b = new B();
  上面 a、b 对实例 A、B 都是强引用  C c = new C(b);   b = null;
  而上面这种情况就不一样了,即使 b 被置为 null,但是 c 仍然持有对 C 对象实例的引用,而间接的保持着对 b 的强引用,所以 GC 不会回收分配给 b 的空间,导致 b 无法回收也没有被使用,造成了内存泄漏。这时可以通过 c = null; 来使得 c 被回收,但也可以通过弱引用来达到同样目的: WeakReference c = new WeakReference(b);
  从源码中可以看出 Entry 里的 key 对 ThreadLocal 实例是弱引用:     static class Entry extends WeakReference> {          Object value;         Entry(ThreadLocal<?> k, Object v) {             super(k);             value = v;         }     }
  Entry 里的 key 对 ThreadLocal 实例是弱引用,将 key 值置为 null,堆中的 ThreadLocal 实例是可以被垃圾收集器(GC)回收的。但是 value 却存在一条从 Current Thread 过来的强引用链,只有当当前线程 Current Thread 销毁时,value 才能被回收。在 threadLocal 被设为 null 以及线程结束之前,Entry 的键值对都不会被回收,出现内存泄漏。为了避免泄漏,在 ThreadLocalMap 中的 set/get Entry 方法里,会对 key 为 null 的情况进行判断,如果为 null 的话,就会对 value 置为 null。也可以通过 ThreadLocal 的 remove 方法(类似加锁和解锁,最后 remove 一下,解锁对象的引用)直接清除,释放内存空间。
  总结来说,利用 ThreadLocal 来访问共享数据时,JVM 通过设置 ThreadLocalMap 的 Key 为弱引用,来避免内存泄露,同时通过调用 remove、get、set 方法的时候,回收弱引用(Key 为 null 的 Entry)。当使用 static ThreadLocal 的时候(如上面的 Spring 多数据源),static 变量在类未加载的时候,它就已经加载,当线程结束的时候,static 变量不一定会被回收,比起普通成员变量使用的时候才加载,static 的生命周期变长了,若没有及时回收,容易产生内存泄漏。 八、线程池8.1 线程池的核心参数和作用顺序
  使用线程池,可以重用存在的线程,减少对象创建、消亡的开销,可控制最大并发线程数,避免资源竞争过度,还能实现线程定时执行、单线程执行、固定线程数执行等功能。
  Java 把线程的调用封装成了一个 Executor 接口,Executor 接口中定义了一个 execute 方法,用来提交线程的执行。Executor 接口的子接口是 ExecutorService,负责管理线程的执行。通过 Executors 类的静态方法可以初始化
  ExecutorService 线程池。Executors 类的静态方法可创建不同类型的线程池: newFixedThreadPool 固定线程线程池  newSingleThreadExecutor 单线程线程池  newCachedThreadPool 缓存线程线程池  newScheduledThreadPool 调度线程线程池
  但是,不建议使用 Executors 去创建线程池,而是通过 ThreadPoolExecutor 的方式,明确给出线程池的参数去创建,规避资源耗尽的风险。
  如果使用 Executors 去创建线程池: 对于 Executors.newFixedThreadPool() 和 Executors.newSingleThreadExecutor() 线程池: 队列用的是 LinkedBlockingQueue,默认大小为 Integer.MAX_VALUE ,可能会堆积大量的请求(可以无限的添加任务),从而导致内存溢出。 对于 Executors.newCachedThreadPool() 和 Executors.ScheduledThreadPool 线程池: 允许的创建的最大线程数量为 Integer.MAX_VALUE,即 2147483647,大量线程的创建会导致严重的性能问题(线程上下文切换带来的开销),线程创建占用堆外内存,任务对象占用堆内内存,大量线程执行任务会导致堆外内存或堆内内存任意一个首先内存溢出。
  最佳的实践是通过 ThreadPoolExecutor 手动地去创建线程池,选取合适的队列存储任务,并指定线程池线程大小。通过线程池实现类 ThreadPoolExecutor 可构造出线程池的,构造函数有下面几个重要的参数: public ThreadPoolExecutor(int corePoolSize,      i nt maximumPoolSize,       long keepAliveTime,      TimeUnit unit,       BlockingQueue workQueue,   ThreadFactory threadFactory,     RejectedExecutionHandler handler) {     if (corePoolSize < 0 ||    maximumPoolSize <= 0 ||     maximumPoolSize < corePoolSize ||    keepAliveTime < 0)    throw new IllegalArgumentException();   if (workQueue == null || threadFactory == null || handler == null)    throw new NullPointerException();   this.corePoolSize = corePoolSize;  this.maximumPoolSize = maximumPoolSize;   this.workQueue = workQueue;     this.keepAliveTime = unit.toNanos(keepAliveTime);   this.threadFactory = threadFactory;    this.handler = handler;}
  参数 1:corePoolSize
  线程池核心线程数。
  参数 2:workQueue
  阻塞队列,用于保存执行任务的线程,有 4 种阻塞队列可选: 基于数组的有界阻塞队列,按 FIFO 先进先出任务的 ArrayBlockingQueue(); 基于链表的阻塞队列(可支持有界或无界),按 FIFO 先进先出任务的 LinkedBlockingQueue(); 不存储元素的阻塞队列,一个线程插入元素后会被阻塞,直到被其他线程取出元素才会唤醒的 Synchronous Queue(吞吐量高于 LinkedBlockingQueue,是一个无界阻塞队列,理论上可存储无限个元素); 具有优先级的可以针对任务排序的无界阻塞队列 PriorityBlockingQueue()。
  关于队列的其他内容,会在并发编程解惑之队列这篇里做详细的介绍。
  参数 3:maximunPoolSize
  线程池最大线程数。如果阻塞队列满了(有界的阻塞队列),来了一个新的任务,若线程池当前线程数小于最大线程数,则创建新的线程执行任务,否则交给饱和策略处理。如果是无界队列就不存在这种情况,任务都在无界队列里存储着。
  参数 4:RejectedExecutionHandler
  拒绝策略,当队列满了,而且线程达到了最大线程数后,对新任务采取的处理策略。
  有 4 种策略可选: 丢弃新任务并抛出 Rejected Execution Exception 异常的 AbortPolicy 策略 直接丢弃新任务不抛出异常的 DiscardPolicy 策略 由调用线程处理该任务的 CallerRunsPolicy() 策略 丢弃队列最前面的任务,然后重新尝试执行任务(不断重复该过程)的 DiscardOldestPolicy 策略
  最后,还可以自定义处理策略。
  参数 5:ThreadFactory
  创建线程的工厂。
  参数 6:keeyAliveTime
  线程没有任务执行时最多保持多久时间终止。当线程池中的线程数大于 corePoolSize 时,线程池中所有线程中的某一个线程的空闲时间若达到 keepAliveTime,则会终止,直到线程池中的线程数不超过 corePoolSize。但如果调用了 allowCoreThread TimeOut(boolean value) 方法,线程池中的线程数就算不超过 corePoolSize,keepAlive Time 参数也会起作用,直到线程池中的线程数量变为 0。
  参数 7:TimeUnit
  配合第 6 个参数使用,表示存活时间的时间单位最佳的实践是通过 ThreadPoolExecutor 手动地去创建线程池,选取合适的队列存储任务,并指定线程池线程大小。 public class FixedThreadPoolTest {      public static void main(String[] args) {          ExecutorService es = Executors.newFixedThreadPool(2, Executors.defaultThreadFactory());          final LinkedBlockingDeque que = new LinkedBlockingDeque();          for(int i = 1; i <= 10; i ++) {              //将数字转换成字符串             que.add(i + "");         }          Future result = es.submit(new Callable() {             @Override             public String call() throws Exception {                  while (!que.isEmpty()) {                      System.out.println(que.poll());                 }                  return "运行完毕";             }         });          System.out.println(result.isDone());          // get方法会阻塞,直到拿到返回值         try {              System.out.println(result.get());          } catch (InterruptedException e) {             e.printStackTrace();         } catch (ExecutionException e) {             e.printStackTrace();         }      }  }
  运行结果: false 1 2 3 4 5 6 7 8 9 10 运行完毕 8.2 线程池源码
  线程池创建线程时,会将线程封装成工作线程 Worker,Worker 在执行完任务后,还会不断的去获取队列里的任务来执行。Worker 的加锁解锁机制是继承 AQS 实现的。 private final class Worker extends AbstractQueuedSynchronizer implements Runnable {      private static final long serialVersionUID = 6138294804551838833L;      //被封装的线程,就是Worker自己     final Thread thread;      //Worker要执行的第一个任务     Runnable firstTask;      //记录执行完成的任务数量     volatile long completedTasks;      //Worker类构造器     Worker(Runnable firstTask) {          setState(-1); // 在worker线程没有启动前是-1状态,无法加锁          this.firstTask = firstTask;          this.thread = getThreadFactory().newThread(this);     }      // state为0代表没加锁     // state为1代表加锁了     //通过CAS尝试加锁,将状态从0设置为1     //该方法重写了父类AQS的同名方法     protected boolean tryAcquire(int unused) {          if (compareAndSetState(0, 1)) {              setExclusiveOwnerThread(Thread.currentThread());              return true;         }          return false;     }      //尝试释放锁,直接将state置为0     //该方法重写了父类AQS的同名方法     protected boolean tryRelease(int unused) {          setExclusiveOwnerThread(null);          setState(0);          return true;     }      //需要注意的是:tryAcquire与tryRelease是重写了父类AQS的方法,并且不可以直接调用,它们被下面的方法调用,实现加锁和解除     //加锁      //acquire方法是它父类AQS类的方法,方法里会调用tryAcquire方法加锁     public void lock() {         acquire(1);     }      //尝试加锁     public boolean tryLock() {         return tryAcquire(1);     }      //解锁      //release方法是它父类AQS类的方法,方法里会调用tryRelease方法释放锁     public void unlock() {         release(1);     }      //返回锁的状态     public boolean isLocked() {         return isHeldExclusively();     }  }
  我们来看下 Worker 线程的运行过程: //Worker执行任务时会调用run方法         public void run() {              runWorker(this);          }      final void runWorker(Worker w) {          Thread wt = Thread.currentThread();          Runnable task = w.firstTask;// 得到Worker中的任务task(通过addWorker方法提交的任务)          w.firstTask = null;          w.unlock(); //允许中断          boolean completedAbruptly = true;          try {             while (task != null || (task = getTask()) != null) {                  w.lock();// 拿到了任务,给Worker上锁,表示当前Worker开始执行任务了                 if ((runStateAtLeast(ctl.get(), STOP) ||                      (Thread.interrupted() &&                       runStateAtLeast(ctl.get(), STOP))) &&                     !wt.isInterrupted())  //先判断线程池状态是否允许继续执行任务: //如果是stop、tidying、terminated状态(这种状态是不接受任务,且不执行任务的),并且线程是非中断状态 //又或者是shutingdown、runing状态 ,并且处于中断状态 //这个时候则中断线程                     wt.interrupt();                 try {                      beforeExecute(wt, task);                      Throwable thrown = null;                      try {                          task.run();// 调用task的run方法执行任务,而不是start方法。线程池调用shutdownNow方法可以中断run的运行                     } catch (RuntimeException x) {                         thrown = x; throw x;                     } catch (Error x) {                         thrown = x; throw x;                     } catch (Throwable x) {                         thrown = x; throw new Error(x);                     } finally {                         afterExecute(task, thrown);                     }                 } finally {                      task = null;                     w.completedTasks++;                     w.unlock();// 执行完任务后,解锁Worker,当前Worker线程变成闲置Worker线程                 }             }             completedAbruptly = false;         } finally {             processWorkerExit(w, completedAbruptly);// 回收Worker线程         }     }
  总结来说,如果当前运行的线程数小于 corePoolSize 线程数,则获取全局锁,然后创建新的线程来执行任务如果运行的线程数大于等于 corePoolSize 线程数,则将任务加入阻塞队列 BlockingQueue 如果阻塞队列已满,无法将任务加入 BlockingQueue,则获取全局所,再创建新的线程来执行任务
  如果新创建线程后使得线程数超过了 maximumPoolSize 线程数,则调用 Rejected ExecutionHandler.rejectedExecution() 方法根据对应的拒绝策略处理任务。
  CPU 密集型任务,线程执行任务占用 CPU 时间会比较长,应该配置相对少的线程数,避免过度争抢资源,可配置 N 个 CPU+1 个线程的线程池;但 IO 密集型任务则由于需要等待 IO 操作,线程经常处于等待状态,应该配置相对多的线程如 2*N 个 CPU 个线程,A 线程阻塞后,B 线程能马上执行,线程多竞争激烈,能饱和的执行任务。线程提交 SQL 后等待数据库返回结果时间较长的情况,CPU 空闲会较多,线程数应设置大些,让更多线程争取 CPU 的调度。

聊一聊下跌中的片仔癀腾讯以及阿里大家好啊,今天这篇文章是扫地僧司马懿写的第262篇原创文章。如果您要和我互动,文末留言。本文语音版01hr7月份没有被温柔以待今年的7月份气温异常反常,用骄阳似火来形容毫不为过。经单身的人老得更快?看看研究怎么说不想变老的壹读君敏敏单身一时爽,一直单身一直爽。这句被很多单身者拿来自我调侃的话,眼看着就要被推翻了有研究显示,单身的人衰老速度更快。对此,有网友表示质疑,直言有很多明星未婚但非常四五十岁的中年女性,半身裙穿得对不对,对气质的影响太大了四五十岁的成熟女性们已经步入了不惑之年,这个阶段的成熟女性们已经很少会去追求穿搭上的时尚感和前卫感了,相对于那些新潮的搭配方式而言,成熟女性们在穿搭上面更关注的应该是舒适度与气质感欧冠刺激之夜本菲卡74遭绝杀,黑马32大逆转欧冠附加赛资格赛今日凌晨继续进行,在结束的10场次回合比赛中,一共10支球队杀入到了最终的资格赛中。他们将在下周,争夺5个决赛圈席位。其中来自葡萄牙的豪门本菲卡客场31大胜对手,总ampampquot婆媳不合ampampquot刘诗诗被吴奇隆天价彩礼迎娶,却被婆婆甩锅生不出娃刘诗诗躲了12年刘亦菲到底在害怕什么?或许这是一个赝品的自我修养2005年唐人投资的仙剑火遍全国也捧红了清纯灵儿刘亦菲源源不断的商业价值令老板蔡艺侬迫切想将刘亦菲收入囊中为避免夜长细语,孙怡董子健还真并非和平分开细语,董子健孙怡还真并非和平分开!相信是孙怡无法忍受了!1未婚先孕,生了女孩,不被公婆宠爱。223岁的小女孩,原先挺火,结婚基本上没天然资源了。3在电视节目中他们带小孩不禁聊婆婆。吴磊和向涵之疑似恋爱上热搜吴磊曾说恋爱不会公开当大家还在为董子健和孙怡离婚的事情操碎了心的时候,吴磊曾说恋爱不会公开的热搜就凭空而来。吴磊是内地新生代当红男演员,自入行以来参演了多部影视剧。向涵之是美籍华人,签约徐静蕾演艺公司98英寸大屏电视降至1。3万,康佳精准打击小米TCL创维?还没有到金九银十促销季,但康佳电视已经提前抢在暑期开始引爆大屏电视市场。日前,康佳将一款售价在2万元左右的98英寸大屏电视售价直降7千,以1。3万元的史上超低价开启了对大屏电视市场媒体报道清华博士创立昇科能源,争做能源服务领跑者近日,昇科能源CEO褚政宇受邀接受北京市大学生创业板创头条的专访,从昇科的技术谈到产品矩阵再到商业落地,与记者畅谈个人3年多的创业故事。近年来,新能源汽车起火事件频频发生,不仅把新降价,救不了小鹏被三杀本文来自微信公众号表外表里(IDexcelers),作者张冉冉赫晋一,编辑付晓玲,Reno焦虑烦躁,可能是何小鹏近期的状态写照。这当然和小鹏汽车遇到的糟心事有关。G9发布在即,却被中国旺旺靠什么一直旺?斑马消费陈晓京进入大陆30年,中国旺旺从来没有消失在大众视野,也从没有离开过货架柜台,还时不时来一波你旺我旺大家旺的回忆杀。这家台企为什么会一直旺?出圈中国旺旺可能自己也不会想到,
发现动物新种!武夷山国家公园族谱再添新成员武夷山国家公园生物资源丰富,生物多样性富集,被誉为蛇的王国鸟的天堂昆虫的世界。国家公园成立以来,这里启动了为期三年的生物资源本底调查,日前,武夷山国家公园正式发布,生物资源本底调查武夷山喜讯!五夫镇兴贤村入选2022年中国美丽休闲乡村日前,农业农村部办公厅公布2022年中国美丽休闲乡村名单。全国共255个村入选,其中84个村同时为农家乐特色村。五夫镇兴贤村名列其中。为贯彻落实2022年中央一号文件和农业农村部关回声报分析芬威出售利物浦40亿镑市值无力与国家俱乐部竞争直播吧11月23日讯回声报报道,经过12年的经营,FSG芬威体育集团正在探索出售利物浦俱乐部的可能性,现在他们正在悄悄地寻找来自世界各地的潜在报价。2010年10月,芬威以3亿镑收联想感言司马南抨击联想带来什么联想联想感言司马南抨击联想带来什么联想?!!视频加载中司马南老师直击联想多处要害,包括在股份制改革中贱卖国有资产,导致国有资产流失企业资不抵债,高官却拿着天价薪酬,是穷庙富方丈联想27国内8个观赏火山的旅游景点,都是大自然的鬼斧神工世界上有不少著名的火山和火山旅游景点,比如日本富士山,夏威夷的基拉韦厄火山,印尼的卡瓦伊真火山等等。其实我国也有不少观赏火山的好地方,让我们来看看它们都藏在哪里吧!1。长白山天池长歌手支朴喜欢大胆的性感风格,把蕾丝连衣裙变成了逛街的衣服歌手支朴喜欢大胆的性感风格,所以不介意把蕾丝连衣裙变成逛街的衣服。开胸连衣裙,轻薄的丝绸面料被支普与皮衣相结合,以节省暴露。美女还选择了及膝的靴子,把性感的睡裙变成了逛街的服装。裙揭阳地区的芥蓝薳是潮汕人十分喜欢吃的青菜,你吃过吗?在潮汕地区有这样的一句俗语好鱼马鲛鲳,好菜芥蓝薳,好戏苏六娘。也有人说爱食好鱼白腹鲳,爱食好菜芥蓝薳,爱娶雅亩苏六娘!但不管是哪种说法,芥蓝薳就是潮汕人公认最好吃的青菜。潮语中的芥2022年全国大型采购商广西行贵港站(平南)暨产销对接大会召开11月23日,由自治区商务厅与贵港市共同举办的2022年全国大型采购商广西行贵港站(平南)暨产销对接大会在平南县召开。本次大会以加强市场推广,促进产销对接,推动产业发展,助力乡村振沙漠中竟惊现圆形草场?!原来是它近期,有网友航拍时发现在新疆维吾尔自治区塔克拉玛干沙漠南部的金色沙海中竟然出现了一个个圆形绿色草场它们与一望无际的黄沙相间绿草萋萋,十分壮观塔克拉玛干沙漠南部沙海中的圆形绿色草场来打开路由器的双频合一,让WiFi加速随着移动互联网的发展,路由器作为新时代移动互联网发展的产物,已经进入了寻常百姓家。现在相信每家每户多多少少的都能见到路由器的身影。移动互联网和智能手机的发展,也深刻地影响了人们日常中国科学院院士黄建平全球干旱半干旱区将加速扩张沙漠具有不可忽视的固碳能力封面新闻记者杨芮雯摄影报道11月26日,第四届世界科技与发展论坛气候变化与环境可持续性分论坛在成都天府国际会议中心举行。论坛由成都信息工程大学与成都市科普文化产业协会联合承办,以天