欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Redis實(shí)現(xiàn)延遲隊(duì)列的全流程詳解

 更新時(shí)間:2023年03月14日 11:20:52   作者:吹老師個(gè)人app編程教學(xué)  
Redisson是Redis服務(wù)器上的分布式可伸縮Java數(shù)據(jù)結(jié)構(gòu),這篇文中主要為大家介紹了Redisson實(shí)現(xiàn)的優(yōu)雅的延遲隊(duì)列的方法,需要的可以參考一下

1、前言

1.1、什么是延遲隊(duì)列

延時(shí)隊(duì)列相比于普通隊(duì)列最大的區(qū)別就體現(xiàn)在其延時(shí)的屬性上,普通隊(duì)列的元素是先進(jìn)先出,按入隊(duì)順序進(jìn)行處理,而延時(shí)隊(duì)列中的元素在入隊(duì)時(shí)會(huì)指定一個(gè)延遲時(shí)間,表示其希望能夠在經(jīng)過(guò)該指定時(shí)間后處理。從某種意義上來(lái)講,延遲隊(duì)列的結(jié)構(gòu)并不像一個(gè)隊(duì)列,而更像是一種以時(shí)間為權(quán)重的有序堆結(jié)構(gòu)。

1.2、應(yīng)用場(chǎng)景

我們?cè)谝恍I(yè)務(wù)場(chǎng)景中,經(jīng)常會(huì)遇到一些需要經(jīng)歷一段時(shí)間后,或者到達(dá)某個(gè)時(shí)間節(jié)點(diǎn)才會(huì)執(zhí)行的功能。就比如以下這些場(chǎng)景:

新建一個(gè)訂單,在規(guī)定時(shí)間內(nèi)未支付需要自動(dòng)取消 外賣或者打車在預(yù)計(jì)時(shí)間到達(dá)的前十分鐘提醒騎手或者司機(jī)即將超時(shí) 快遞收貨后在規(guī)定時(shí)間內(nèi)用戶沒(méi)有確認(rèn)收貨會(huì)自動(dòng)確認(rèn)收貨 預(yù)定的會(huì)議在會(huì)議開始前十分鐘會(huì)去提醒你盡快加入會(huì)議 每日周報(bào)在截止半小時(shí)前會(huì)提醒你盡快提交

1.3、為什么要使用延遲隊(duì)列

對(duì)于一些數(shù)據(jù)量小并且對(duì)數(shù)據(jù)的時(shí)效性不怎么要求的項(xiàng)目來(lái)說(shuō),最簡(jiǎn)單有效的方法就是寫一個(gè)定時(shí)任務(wù)去掃描數(shù)據(jù)庫(kù)以達(dá)到業(yè)務(wù)的實(shí)現(xiàn)。當(dāng)然,如果在數(shù)據(jù)達(dá)到數(shù)百萬(wàn)或者千萬(wàn)級(jí)別的時(shí)候,如果去定時(shí)掃描數(shù)據(jù)庫(kù),容易挨揍哈。想信大家也有所了解,當(dāng)數(shù)據(jù)達(dá)到這種地步的時(shí)候,還去定時(shí)掃表會(huì)非常低效,甚至對(duì)于那些定時(shí)間隔比較小的情景來(lái)說(shuō),這一遍還沒(méi)掃完下一遍就要開始了。這時(shí)候如果用延遲隊(duì)列的話或許會(huì)很有效。

實(shí)現(xiàn)延遲隊(duì)列的幾種途徑

  • Quartz 定時(shí)任務(wù)
  • DelayQueue 延遲隊(duì)列
  • Redis sorted set Redis
  • 過(guò)期鍵監(jiān)聽回調(diào)
  • RabbitMQ死信隊(duì)列
  • RabbitMQ基于插件實(shí)現(xiàn)延遲隊(duì)列
  • wheel時(shí)間輪算法

2、Redis sorted set

在Redis中,zet作為有序集合,可以利用其有序的特性,將任務(wù)添加到zset中,將任務(wù)的到期時(shí)間作為score,利用zset的默認(rèn)有序特性,獲取score值最小的元素(也就是最近到期的任務(wù)),判斷系統(tǒng)時(shí)間與該任務(wù)的到期時(shí)間大小,如果達(dá)到到期時(shí)間,就執(zhí)行業(yè)務(wù),并刪除該到期任務(wù),繼續(xù)判斷下一個(gè)元素,如果沒(méi)有到期,就sleep一段時(shí)間(比如1秒),如果集合為空,也sleep一段時(shí)間。

通過(guò)zadd命令向隊(duì)列delayqueue中添加元素,并設(shè)置score值表示元素過(guò)期的時(shí)間;向delayqueue添加三個(gè)order1、order2、order3,分別是10秒、20秒、30秒后過(guò)期。

zadd delayqueue 3 order3

消費(fèi)端輪詢隊(duì)列delayqueue,將元素排序后取最小時(shí)間與當(dāng)前時(shí)間比對(duì),如小于當(dāng)前時(shí)間代表已經(jīng)過(guò)期移除key。

/**
 * 消費(fèi)消息
 */
public void pollOrderQueue() {
    while (true) {
        Set<Tuple> set = jedis.zrangeWithScores(DELAY_QUEUE, 0, 0);
        String value = ((Tuple) set.toArray()[0]).getElement();
        int score = (int) ((Tuple) set.toArray()[0]).getScore();
        Calendar cal = Calendar.getInstance();
        int nowSecond = (int) (cal.getTimeInMillis() / 1000);
        if (nowSecond >= score) {
            jedis.zrem(DELAY_QUEUE, value);
            System.out.println(sdf.format(new Date()) + " removed key:" + value);
        }
        if (jedis.zcard(DELAY_QUEUE) <= 0) {
            System.out.println(sdf.format(new Date()) + " zset empty ");
            return;
        }
        Thread.sleep(1000);
    }
} 

我們看到執(zhí)行結(jié)果符合預(yù)期:

2020-05-07 13:24:09 add finished.
2020-05-07 13:24:19 removed key:order1
2020-05-07 13:24:29 removed key:order2
2020-05-07 13:24:39 removed key:order3
2020-05-07 13:24:39 zset empty 

3、Redis 過(guò)期鍵監(jiān)聽回調(diào)

Redis的key過(guò)期回調(diào)事件,也能達(dá)到延遲隊(duì)列的效果,簡(jiǎn)單來(lái)說(shuō)我們開啟監(jiān)聽key是否過(guò)期的事件,一旦key過(guò)期會(huì)觸發(fā)一個(gè)callback事件。

修改redis.conf文件開啟notify-keyspace-events Ex。 notify-keyspace-events Ex

Redis監(jiān)聽配置,注入Bean RedisMessageListenerContainer。

其次,配置redis監(jiān)聽器 最后,編寫redis key過(guò)期監(jiān)聽回調(diào)方法

@Configuration
public class RedisListenerConfig {
@Bean
RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory) {
        RedisMessageListenerContainer container = new RedisMessageListenerContainer();
        container.setConnectionFactory(connectionFactory);
         return container;
    }
} 

編寫Redis過(guò)期回調(diào)監(jiān)聽方法,必須繼承KeyExpirationEventMessageListener ,有點(diǎn)類似于MQ的消息監(jiān)聽。

@Component
public class RedisKeyExpirationListener extends KeyExpirationEventMessageListener {
public RedisKeyExpirationListener(RedisMessageListenerContainer listenerContainer) {
    super(listenerContainer);
}
@Override
public void onMessage(Message message, byte[] pattern) {
      String expiredKey = message.toString();
      System.out.println("監(jiān)聽到key:" + expiredKey + "已過(guò)期");
    }
} 

到這代碼就編寫完成,非常的簡(jiǎn)單,接下來(lái)測(cè)試一下效果,在redis-cli客戶端添加一個(gè)key并給定3s的過(guò)期時(shí)間。

set xiaofu 123 ex 3

在控制臺(tái)成功監(jiān)聽到了這個(gè)過(guò)期的key。

監(jiān)聽到過(guò)期的key為:xiaofu

4、Quartz定時(shí)任務(wù)

Quartz一款非常經(jīng)典任務(wù)調(diào)度框架,在Redis、RabbitMQ還未廣泛應(yīng)用時(shí),超時(shí)未支付取消訂單功能都是由定時(shí)任務(wù)實(shí)現(xiàn)的。

導(dǎo)入Quartz依賴

<dependency>
   <groupId>org.springframework.boot</groupId>
   <artifactId>spring-boot-starter-quartz</artifactId>
</dependency>
>在啟動(dòng)類中使用@EnableScheduling注解開啟定時(shí)任務(wù)功能。
```java
@SpringBootApplication
@EnableScheduling
public class DelayQueueApplication {
    public static void main(String[] args) {
        SpringApplication.run(DelayQueueApplication.class, args);
    }
}

編寫定時(shí)任務(wù)

@Slf4j
@Component
public class QuartzDemo {
    /**
     * 每隔五秒開啟一次任務(wù)
     */
    @Scheduled(cron = "0/5 * * * * ? ")
    public void process(){
        log.info("--------------定時(shí)任務(wù)測(cè)試--------------");
    }
}

5、DelayQueue 延遲隊(duì)列

JDK中提供了一組實(shí)現(xiàn)延遲隊(duì)列的API,位于Java.util.concurrent包下DelayQueue。

DelayQueue是一個(gè)BlockingQueue(無(wú)界阻塞)隊(duì)列,它本質(zhì)就是封裝了一個(gè)PriorityQueue(優(yōu)先隊(duì)列),PriorityQueue內(nèi)部使用完全二叉堆(不知道的自行了解哈)來(lái)實(shí)現(xiàn)隊(duì)列元素排序,我們?cè)谙駾elayQueue隊(duì)列中添加元素時(shí),會(huì)給元素一個(gè)Delay(延遲時(shí)間)作為排序條件,隊(duì)列中最小的元素會(huì)優(yōu)先放在隊(duì)首。隊(duì)列中的元素只有到了Delay時(shí)間才允許從隊(duì)列中取出。隊(duì)列中可以放基本數(shù)據(jù)類型或自定義實(shí)體類,在存放基本數(shù)據(jù)類型時(shí),優(yōu)先隊(duì)列中元素默認(rèn)升序排列,自定義實(shí)體類就需要我們根據(jù)類屬性值比較計(jì)算了。 先簡(jiǎn)單實(shí)現(xiàn)一下看看效果,添加三個(gè)order入隊(duì)DelayQueue,分別設(shè)置訂單在當(dāng)前時(shí)間的5秒、10秒、15秒后取消。

要實(shí)現(xiàn)DelayQueue延時(shí)隊(duì)列,隊(duì)中元素要implements Delayed 接口,這哥接口里只有一個(gè)getDelay方法,用于設(shè)置延期時(shí)間。Order類中compareTo方法負(fù)責(zé)對(duì)隊(duì)列中的元素進(jìn)行排序。

public class Order implements Delayed {
/**
 * 延遲時(shí)間
 */
@JsonFormat(locale = "zh", timezone = "GMT+8", pattern = "yyyy-MM-dd HH:mm:ss")
private long time;
String name;
public Order(String name, long time, TimeUnit unit) {
    this.name = name;
    this.time = System.currentTimeMillis() + (time > 0 ? unit.toMillis(time) : 0);
}
@Override
public long getDelay(TimeUnit unit) {
    return time - System.currentTimeMillis();
}
@Override
public int compareTo(Delayed o) {
    Order Order = (Order) o;
    long diff = this.time - Order.time;
    if (diff <= 0) {
        return -1;
    } else {
        return 1;
    }
}
} 

DelayQueue的put方法是線程安全的,因?yàn)閜ut方法內(nèi)部使用了ReentrantLock鎖進(jìn)行線程同步。DelayQueue還提供了兩種出隊(duì)的方法poll()和take() , poll()為非阻塞獲取,沒(méi)有到期的元素直接返回null;take()阻塞方式獲取,沒(méi)有到期的元素線程將會(huì)等待。

public class DelayQueueDemo {
public static void main(String[] args) throws InterruptedException {
    Order Order1 = new Order("Order1", 5, TimeUnit.SECONDS);
    Order Order2 = new Order("Order2", 10, TimeUnit.SECONDS);
    Order Order3 = new Order("Order3", 15, TimeUnit.SECONDS);
    DelayQueue<Order> delayQueue = new DelayQueue<>();
    delayQueue.put(Order1);
    delayQueue.put(Order2);
    delayQueue.put(Order3);
    System.out.println("訂單延遲隊(duì)列開始時(shí)間:" + LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
    while (delayQueue.size() != 0) {
        /**
         * 取隊(duì)列頭部元素是否過(guò)期
         */
        Order task = delayQueue.poll();
        if (task != null) {
            System.out.format("訂單:{%s}被取消, 取消時(shí)間:{%s}\n", task.name,  
            LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
        }
        Thread.sleep(1000);
    }
}
} 

上邊只是簡(jiǎn)單的實(shí)現(xiàn)入隊(duì)與出隊(duì)的操作,實(shí)際開發(fā)中會(huì)有專門的線程,負(fù)責(zé)消息的入隊(duì)與消費(fèi)。

執(zhí)行后看到結(jié)果如下,Order1、Order2、Order3 分別在 5秒、10秒、15秒后被執(zhí)行,至此就用DelayQueue實(shí)現(xiàn)了延時(shí)隊(duì)列。

訂單延遲隊(duì)列開始時(shí)間:2020-05-06 14:59:09

訂單:{Order1}被取消, 取消時(shí)間:{2020-05-06 14:59:14}

訂單:{Order2}被取消, 取消時(shí)間:{2020-05-06 14:59:19}

訂單:{Order3}被取消, 取消時(shí)間:{2020-05-06 14:59:24}

6、RabbitMQ 延時(shí)隊(duì)列

利用 RabbitMQ 做延時(shí)隊(duì)列是比較常見(jiàn)的一種方式,而實(shí)際上RabbitMQ 自身并沒(méi)有直接支持提供延遲隊(duì)列功能,而是通過(guò) RabbitMQ 消息隊(duì)列的 TTL和 DXL這兩個(gè)屬性間接實(shí)現(xiàn)的。

先來(lái)認(rèn)識(shí)一下 TTL和 DXL兩個(gè)概念:

Time To Live(TTL) :

TTL 顧名思義:指的是消息的存活時(shí)間,RabbitMQ可以通過(guò)x-message-tt參數(shù)來(lái)設(shè)置指定Queue(隊(duì)列)和 Message(消息)上消息的存活時(shí)間,它的值是一個(gè)非負(fù)整數(shù),單位為微秒。

RabbitMQ 可以從兩種維度設(shè)置消息過(guò)期時(shí)間,分別是隊(duì)列和消息本身

設(shè)置隊(duì)列過(guò)期時(shí)間,那么隊(duì)列中所有消息都具有相同的過(guò)期時(shí)間。 設(shè)置消息過(guò)期時(shí)間,對(duì)隊(duì)列中的某一條消息設(shè)置過(guò)期時(shí)間,每條消息TTL都可以不同。 如果同時(shí)設(shè)置隊(duì)列和隊(duì)列中消息的TTL,則TTL值以兩者中較小的值為準(zhǔn)。而隊(duì)列中的消息存在隊(duì)列中的時(shí)間,一旦超過(guò)TTL過(guò)期時(shí)間則成為Dead Letter(死信)。

Dead Letter Exchanges(DLX)

DLX即死信交換機(jī),綁定在死信交換機(jī)上的即死信隊(duì)列。RabbitMQ的 Queue(隊(duì)列)可以配置兩個(gè)參數(shù)x-dead-letter-exchange 和 x-dead-letter-routing-key(可選),一旦隊(duì)列內(nèi)出現(xiàn)了Dead Letter(死信),則按照這兩個(gè)參數(shù)可以將消息重新路由到另一個(gè)Exchange(交換機(jī)),讓消息重新被消費(fèi)。

x-dead-letter-exchange:隊(duì)列中出現(xiàn)Dead Letter后將Dead Letter重新路由轉(zhuǎn)發(fā)到指定 exchange(交換機(jī))。

x-dead-letter-routing-key:指定routing-key發(fā)送,一般為要指定轉(zhuǎn)發(fā)的隊(duì)列。

隊(duì)列出現(xiàn)Dead Letter的情況有:

消息或者隊(duì)列的TTL過(guò)期

隊(duì)列達(dá)到最大長(zhǎng)度

消息被消費(fèi)端拒絕(basic.reject or basic.nack)

下邊結(jié)合一張圖看看如何實(shí)現(xiàn)超30分鐘未支付關(guān)單功能,我們將訂單消息A0001發(fā)送到延遲隊(duì)列order.delay.queue,并設(shè)置x-message-tt消息存活時(shí)間為30分鐘,當(dāng)?shù)竭_(dá)30分鐘后訂單消息A0001成為了Dead Letter(死信),延遲隊(duì)列檢測(cè)到有死信,通過(guò)配置x-dead-letter-exchange,將死信重新轉(zhuǎn)發(fā)到能正常消費(fèi)的關(guān)單隊(duì)列,直接監(jiān)聽關(guān)單隊(duì)列處理關(guān)單邏輯即可。

發(fā)送消息時(shí)指定消息延遲的時(shí)間

public void send(String delayTimes) {
    amqpTemplate.convertAndSend("order.pay.exchange", "order.pay.queue","大家好我是延遲數(shù)據(jù)", message -> {
      // 設(shè)置延遲毫秒值
      message.getMessageProperties().setExpiration(String.valueOf(delayTimes));
      return message;
    });
  }
}

設(shè)置延遲隊(duì)列出現(xiàn)死信后的轉(zhuǎn)發(fā)規(guī)則

/**
   * 延時(shí)隊(duì)列
   */
  @Bean(name = "order.delay.queue")
  public Queue getMessageQueue() {
    return QueueBuilder
        .durable(RabbitConstant.DEAD_LETTER_QUEUE)
        // 配置到期后轉(zhuǎn)發(fā)的交換
        .withArgument("x-dead-letter-exchange", "order.close.exchange")
        // 配置到期后轉(zhuǎn)發(fā)的路由鍵
        .withArgument("x-dead-letter-routing-key", "order.close.queue")
        .build();
  }

7、時(shí)間輪

前邊幾種延時(shí)隊(duì)列的實(shí)現(xiàn)方法相對(duì)簡(jiǎn)單,比較容易理解,時(shí)間輪算法就稍微有點(diǎn)抽象了。kafka、netty都有基于時(shí)間輪算法實(shí)現(xiàn)延時(shí)隊(duì)列,下邊主要實(shí)踐Netty的延時(shí)隊(duì)列講一下時(shí)間輪是什么原理。

先來(lái)看一張時(shí)間輪的原理圖,解讀一下時(shí)間輪的幾個(gè)基本概念

wheel :時(shí)間輪,圖中的圓盤可以看作是鐘表的刻度。比如一圈round 長(zhǎng)度為24秒,刻度數(shù)為 8,那么每一個(gè)刻度表示 3秒。那么時(shí)間精度就是 3秒。時(shí)間長(zhǎng)度 / 刻度數(shù)值越大,精度越大。

當(dāng)添加一個(gè)定時(shí)、延時(shí)任務(wù)A,假如會(huì)延遲25秒后才會(huì)執(zhí)行,可時(shí)間輪一圈round 的長(zhǎng)度才24秒,那么此時(shí)會(huì)根據(jù)時(shí)間輪長(zhǎng)度和刻度得到一個(gè)圈數(shù) round和對(duì)應(yīng)的指針位置 index,也是就任務(wù)A會(huì)繞一圈指向0格子上,此時(shí)時(shí)間輪會(huì)記錄該任務(wù)的round和 index信息。當(dāng)round=0,index=0 ,指針指向0格子 任務(wù)A并不會(huì)執(zhí)行,因?yàn)?round=0不滿足要求。

所以每一個(gè)格子代表的是一些時(shí)間,比如1秒和25秒 都會(huì)指向0格子上,而任務(wù)則放在每個(gè)格子對(duì)應(yīng)的鏈表中,這點(diǎn)和HashMap的數(shù)據(jù)有些類似。

Netty構(gòu)建延時(shí)隊(duì)列主要用HashedWheelTimer,HashedWheelTimer底層數(shù)據(jù)結(jié)構(gòu)依然是使用DelayedQueue,只是采用時(shí)間輪的算法來(lái)實(shí)現(xiàn)。

下面我們用Netty 簡(jiǎn)單實(shí)現(xiàn)延時(shí)隊(duì)列,HashedWheelTimer構(gòu)造函數(shù)比較多,解釋一下各參數(shù)的含義。

ThreadFactory :表示用于生成工作線程,一般采用線程池;

tickDuration和unit:每格的時(shí)間間隔,默認(rèn)100ms;

ticksPerWheel:一圈下來(lái)有幾格,默認(rèn)512,而如果傳入數(shù)值的不是2的N次方,則會(huì)調(diào)整為大于等于該參數(shù)的一個(gè)2的N次方數(shù)值,有利于優(yōu)化hash值的計(jì)算。

public HashedWheelTimer(ThreadFactory threadFactory, long tickDuration, TimeUnit unit, int ticksPerWheel) {
    this(threadFactory, tickDuration, unit, ticksPerWheel, true);
  }

TimerTask:一個(gè)定時(shí)任務(wù)的實(shí)現(xiàn)接口,其中run方法包裝了定時(shí)任務(wù)的邏輯。

Timeout:一個(gè)定時(shí)任務(wù)提交到Timer之后返回的句柄,通過(guò)這個(gè)句柄外部可以取消這個(gè)定時(shí)任務(wù),并對(duì)定時(shí)任務(wù)的狀態(tài)進(jìn)行一些基本的判斷。 Timer:是HashedWheelTimer實(shí)現(xiàn)的父接口,僅定義了如何提交定時(shí)任務(wù)和如何停止整個(gè)定時(shí)機(jī)制。

public class NettyDelayQueue {
  public static void main(String[] args) {
    final Timer timer = new HashedWheelTimer(Executors.defaultThreadFactory(), 5, TimeUnit.SECONDS, 2);
    //定時(shí)任務(wù)
    TimerTask task1 = new TimerTask() {
      public void run(Timeout timeout) throws Exception {
        System.out.println("order1 5s 后執(zhí)行 ");
        timer.newTimeout(this, 5, TimeUnit.SECONDS);//結(jié)束時(shí)候再次注冊(cè)
      }
    };
    timer.newTimeout(task1, 5, TimeUnit.SECONDS);
    TimerTask task2 = new TimerTask() {
      public void run(Timeout timeout) throws Exception {
        System.out.println("order2 10s 后執(zhí)行");
        timer.newTimeout(this, 10, TimeUnit.SECONDS);//結(jié)束時(shí)候再注冊(cè)
      }
    };
    timer.newTimeout(task2, 10, TimeUnit.SECONDS);
    //延遲任務(wù)
    timer.newTimeout(new TimerTask() {
      public void run(Timeout timeout) throws Exception {
        System.out.println("order3 15s 后執(zhí)行一次");
      }
    }, 15, TimeUnit.SECONDS);
  }
}

從執(zhí)行的結(jié)果看,order3、order3延時(shí)任務(wù)只執(zhí)行了一次,而order2、order1為定時(shí)任務(wù),按照不同的周期重復(fù)執(zhí)行。

order1 5s 后執(zhí)行
order2 10s 后執(zhí)行
order3 15s 后執(zhí)行一次
order1 5s 后執(zhí)行
order2 10s 后執(zhí)行

到此這篇關(guān)于Redis實(shí)現(xiàn)延遲隊(duì)列的全流程詳解的文章就介紹到這了,更多相關(guān)Redis延遲隊(duì)列內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Maven配置文件settings.xml的實(shí)現(xiàn)

    Maven配置文件settings.xml的實(shí)現(xiàn)

    Maven是一個(gè)用于構(gòu)建和管理Java項(xiàng)目的強(qiáng)大工具,它依賴于設(shè)置文件來(lái)配置和管理其行為,其中最重要的之一便是settings.xml文件,本文主要介紹了Maven配置文件settings.xml的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-01-01
  • Response.AddHeader案例講解

    Response.AddHeader案例講解

    這篇文章主要介紹了Response.AddHeader案例講解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-08-08
  • Java異常(Exception)處理以及常見(jiàn)異常總結(jié)

    Java異常(Exception)處理以及常見(jiàn)異??偨Y(jié)

    在《Java編程思想》中這樣定義異常,阻止當(dāng)前方法或作用域繼續(xù)執(zhí)行的問(wèn)題,雖然java中有異常處理機(jī)制,但是要明確一點(diǎn),決不應(yīng)該用"正常"的態(tài)度來(lái)看待異常,這篇文章主要給大家介紹了關(guān)于Java異常(Exception)處理以及常見(jiàn)異常的相關(guān)資料,需要的朋友可以參考下
    2021-10-10
  • ShardingSphere jdbc集成多數(shù)據(jù)源的實(shí)現(xiàn)步驟

    ShardingSphere jdbc集成多數(shù)據(jù)源的實(shí)現(xiàn)步驟

    本文主要介紹了ShardingSphere jdbc集成多數(shù)據(jù)源的實(shí)現(xiàn)步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-10-10
  • SpringBoot詳細(xì)列舉常用注解的說(shuō)明

    SpringBoot詳細(xì)列舉常用注解的說(shuō)明

    在開發(fā)SpringBoot程序的過(guò)程中,有可能與其他業(yè)務(wù)系統(tǒng)進(jìn)行對(duì)接開發(fā),獲取封裝公共的API接口等等,下面這篇文章主要給大家介紹了關(guān)于SpringBoot常見(jiàn)的注解的相關(guān)資料,需要的朋友可以參考下
    2022-06-06
  • 關(guān)于SpringMVC對(duì)Restful風(fēng)格的支持詳解

    關(guān)于SpringMVC對(duì)Restful風(fēng)格的支持詳解

    Restful就是一個(gè)資源定位及資源操作的風(fēng)格,不是標(biāo)準(zhǔn)也不是協(xié)議,只是一種風(fēng)格,是對(duì)http協(xié)議的詮釋,下面這篇文章主要給大家介紹了關(guān)于SpringMVC對(duì)Restful風(fēng)格支持的相關(guān)資料,需要的朋友可以參考下
    2022-01-01
  • 如何修改覆蓋spring boot默認(rèn)日志策略logback詳解

    如何修改覆蓋spring boot默認(rèn)日志策略logback詳解

    這篇文章主要給大家介紹了關(guān)于如何修改覆蓋spring boot默認(rèn)日志策略logback的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2018-10-10
  • 簡(jiǎn)單介紹一下什么是microservice微服務(wù)

    簡(jiǎn)單介紹一下什么是microservice微服務(wù)

    這篇文章主要介紹了一下什么是microservice微服務(wù)微服務(wù)的定義,微服務(wù)到底是什么意思?什么樣的架構(gòu)可以叫做微服務(wù)?這篇文章可以給你答案
    2023-03-03
  • Java日期時(shí)間與正則表達(dá)式超詳細(xì)整理(適合新手入門)

    Java日期時(shí)間與正則表達(dá)式超詳細(xì)整理(適合新手入門)

    如果使用得當(dāng),正則表達(dá)式是匹配各種模式的強(qiáng)大工具,下面這篇文章主要給大家介紹了關(guān)于Java日期時(shí)間與正則表達(dá)式超詳細(xì)整理的相關(guān)資料,本文非常適合新手入門,需要的朋友可以參考下
    2023-04-04
  • 踩坑Debug啟動(dòng)失敗,無(wú)報(bào)錯(cuò)信息問(wèn)題

    踩坑Debug啟動(dòng)失敗,無(wú)報(bào)錯(cuò)信息問(wèn)題

    在進(jìn)行項(xiàng)目debug時(shí)遇到了無(wú)法啟動(dòng)的問(wèn)題,項(xiàng)目一直處于正在啟動(dòng)狀態(tài),但未出現(xiàn)任何報(bào)錯(cuò)信息,分析原因可能是存在不合法的斷點(diǎn)位置,即斷點(diǎn)未打在方法內(nèi)部,解決方法是檢查所有斷點(diǎn)信息,并移除非法斷點(diǎn),之后項(xiàng)目能夠正常啟動(dòng)
    2023-02-02

最新評(píng)論