单例模式能保证某个类在程序中只存在唯一一份实例, 而不会创建出多个实例
饿汉模式:类加载的同时, 创建实例
public class Singleton {
private static Singleton instance = new Singleton();//类加载时实例化
private Singleton(){}//构造私有化
public static Singleton getInstance(){
return instance;
}
}
懒汉模式:第一次使用的时候才创建实例
public class Singleton {
private volatile static Singleton instance = null;//volatile保证内存可见性
private Singleton(){}//构造私有化
public static Singleton getInstance(){
if(instance == null) {//预判断 降低锁竞争
synchronized (Singleton.class) {
if(instance == null) {//获取锁后判断 保证数据原子性
instance = new Singleton();
}
}
}
return instance;
}
}
阻塞队列是一种特殊的队列. 也遵守 “先进先出” 的原则
阻塞队列能是一种线程安全的数据结构, 并且具有以下特性:
阻塞队列的一个典型应用场景就是 “生产者消费者模型”. 这是一种非常典型的开发模型
生产者消费者模式:
生产者消费者模式就是通过一个容器来解决生产者和消费者的强耦合问题
生产者和消费者彼此之间不直接通讯,而通过阻塞队列来进行通讯,所以生产者生产完数据之后不用等
待消费者处理,直接扔给阻塞队列,消费者不找生产者要数据,而是直接从阻塞队列里取
标准库中的阻塞队列:
BlockingQueue<String> queue = new LinkedBlockingQueue<>();
// 入队列
queue.put("abc");
// 出队列. 如果没有 put 直接 take, 就会阻塞.
String elem = queue.take();
阻塞队列实现:
public class BlockingQueue {
private int[] items = new int[1000];
private volatile int head = 0;
private volatile int tail = 0;
public void put(int value) throws InterruptedException {
synchronized (this) {
while((tail+1)%items.length == head) {
wait();
}
items[tail++] = value;
tail %= items.length;
notifyAll();
}
}
public int take() throws InterruptedException {
int ret;
synchronized (this) {
while(head == tail) {
wait();
}
ret = items[head++];
head %= items.length;
notifyAll();
}
return ret;
}
}
定时器也是软件开发中的一个重要组件. 类似于一个 “闹钟”. 达到一个设定的时间之后, 就执行某个指定好的代码
标准库中的定时器:
标准库中提供了一个 Timer 类. Timer 类的核心方法为 schedule
schedule 包含两个参数:第一个参数指定即将要执行的任务代码, 第二个参数指定多长时间之后执行 (单位为毫秒)
Timer timer = new Timer();
timer.schedule(new TimerTask() {
@Override
public void run() {
System.out.println("hello");
}
}, 3000);
定时器的构成:
public class Timer {
static class Task implements Comparable<Task> {
private Runnable task;//执行的任务
private long time;//多久执行
public Task(Runnable task, long time) {
this.task = task;
this.time = System.currentTimeMillis() + time;//绝对时间
}
public void run() {
task.run();//调用执行
}
@Override
public int compareTo(Task o) {
return (int)(time - o.time);//绝对时间小先执行
}
}
private PriorityBlockingQueue<Task> queue = new PriorityBlockingQueue<Task>();//储存任务列表
private Object lock = new Object();
class worker extends Thread {//worker线程
@Override
public void run() {
while(true) {
try {
Task task = queue.take();
long curTime = System.currentTimeMillis();//当前时间
if(task.time > curTime) {//不可执行
queue.put(task);
synchronized (lock) {
lock.wait(task.time -curTime);//等待间隔时长
}
} else {
task.run();//可执行任务
}
} catch (InterruptedException e) {
e.printStackTrace();
break;
}
}
}
}
public Timer() {
worker worker = new worker();//创建启动工作线程
worker.start();
}
public void schedule(Runnable comm, long after) {
Task task = new Task(comm, after);
queue.put(task);
synchronized (lock) {
lock.notify();//唤醒等待线程
}
}
}
线程池使用池化技术,将预先创建好批量线程,等任务到达进行获取执行。
线程池最大的好处就是减少每次启动、销毁线程的损耗。
标准库中的线程池:
ExecutorService pool = Executors.newFixedThreadPool(10);
pool.submit(new Runnable() {
@Override
public void run() {
System.out.println("hello");
}
});
Executors 创建线程池:
Executors 本质上是 ThreadPoolExecutor 类的封装,ThreadPoolExecutor 提供了更多的可选参数, 可以进一步细化线程池行为的设定
实现线程池:
public class ThreadPool {
private List<Thread> workers = new ArrayList<>();
private LinkedBlockingQueue<Runnable> queue = new LinkedBlockingQueue<>();
class Worker extends Thread {
@Override
public void run() {
try {
while(!Thread.interrupted()){
Runnable runnable = queue.take();
runnable.run();
}
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
}
public ThreadPool(int threadNums) {
for (int i = 0; i < threadNums; i++) {
Worker worker = new Worker();
worker.start();
workers.add(worker);
}
}
public void submit(Runnable comm) {
try {
queue.put(comm);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
}