主頁 > 後端開發 > 訊息佇列之activeMQ

訊息佇列之activeMQ

2021-01-04 06:08:17 後端開發

1.activeMQ的主要功能

  1. 實作高可用、高伸縮、高性能、易用和安全的企業級面向訊息服務的系統
  2. 異步訊息的消費和處理
  3. 控制訊息的消費順序
  4. 可以和Spring/springBoot整合簡化編碼
  5. 配置集群容錯的MQ集群

2.activeMQ安裝

下載地址:http://activemq.apache.org/components/classic/download/

這里筆者是下載的linux版的:

因為activeMQ底層是使用java撰寫的,所以需要安裝jdk,這個請移步我之前的博客:

https://www.cnblogs.com/pluto-charon/p/11746636.html

安裝activeMq:

# 安裝apache
[root@localhost ~]# yum install ttpd
# 下載的apache-activemq并上傳到linux的home下,解壓
[root@localhost home]# tar -zxvf apache-activemq-5.16.0-bin.tar.gz 
# 進入到bin目錄下
[root@localhost home]# cd /apache-activemq-5.16.0/bin
# 啟動
[root@localhost bin]# ./activemq start
INFO: Loading '/home/apache-activemq-5.16.0//bin/env'
INFO: Using java '/usr/local/java/jdk1.8.0_20//bin/java'
INFO: Starting - inspect logfiles specified in logging.properties and log4j.properties to get details
INFO: pidfile created : '/home/apache-activemq-5.16.0//data/activemq.pid' (pid '7517')

# activemq的默認埠是61616,查看是否啟動的三種方式
# 第一種
[root@localhost bin]# ps -ef |grep activemq
# 第二種
[root@localhost bin]# netstat -ano|grep 61616
tcp6       0      0 :::61616                :::*                    LISTEN      off (0.00/0/0)
# 第三種
[root@localhost bin]# lsof -i:61616
COMMAND  PID USER   FD   TYPE DEVICE SIZE/OFF NODE NAME
java    7517 root  132u  IPv6  39926      0t0  TCP *:61616 (LISTEN)

# 帶日志的啟動方式
[root@localhost bin]# ./activemq start > /home/apache-activemq-5.16.0/myrunmq.log
[root@localhost bin]# cd ..
# 可以看到,啟動日志都已經記錄到日志里了
[root@localhost apache-activemq-5.16.0]# cat myrunmq.log 
INFO: Loading '/home/apache-activemq-5.16.0//bin/env'
INFO: Using java '/usr/local/java/jdk1.8.0_20//bin/java'
INFO: Starting - inspect logfiles specified in logging.properties and log4j.properties to get details
INFO: pidfile created : '/home/apache-activemq-5.16.0//data/activemq.pid' (pid '7787')
# 關閉activemq
[root@localhost bin]# ./activemq stop

前臺訪問的埠是8161,在查看前臺時,要關閉linux和windows的防火墻:

# 關閉linux防火墻
[root@localhost apache-activemq-5.16.0]# systemctl stop firewalld

在訪問之前,需要修改conf目錄下的jetty.xml,將下面的host修改成自己的ip,以及修改用戶名和密碼,

<bean id="jettyPort"  init-method="start">
    <!-- the default port number for the web console -->
    <property name="host" value="https://www.cnblogs.com/pluto-charon/p/127.0.0.1"/>
    <property name="port" value="https://www.cnblogs.com/pluto-charon/p/8161"/>
</bean>

# 用戶名和密碼可修改可不修改,默認為admin/admin
<bean id="securityConstraint" >
    <property name="name" value="https://www.cnblogs.com/pluto-charon/p/BASIC" />
    <property name="roles" value="https://www.cnblogs.com/pluto-charon/p/user,admin" />
    <!-- set authenticate=false to disable login -->
    <property name="authenticate" value="https://www.cnblogs.com/pluto-charon/p/true" />
</bean>	

修改完成之后重啟activemq

[root@localhost bin]# ./activemq restart

查看,地址為192.168.189.150:8161

到這里就說明activemq安裝成功了,

3.JMS

JMS(java message service)是一個用于提供訊息服務的技術規范,他制定了在整個訊息服務提供程序中的所有資料結構和互動流程,當兩個程式使用jms進行通信時,他們并不是直接相連的,而是通過一個共同的訊息收發服務連接起來的,達到解耦的效果,jms為標準訊息協議和訊息服務提供了一組通用的介面,包括創建、發送、讀取訊息等,

1 JMS的優勢:

異步:客戶端不用發送請求,JMS自動將訊息發送給客戶端

可靠:JMS保證訊息只傳遞一次

2.JMS的四大組件:

  • JMS provider:實作了jms介面和規范的訊息中間件

  • JMS producer:訊息生產者,創建和發送JMS訊息的客戶端應用

  • JMS consumer:訊息消費者,接受和處理JMS訊息的客戶端應用

  • JMS message:由訊息頭、訊息屬性、訊息體組成

    訊息頭(在send方法之前,通過setXXX()設定):

    JMSDestination:訊息發送的目的地,主要是指Queue(點對點傳送模型)和Topic(發布訂閱模型)

    JMSDeliverMode:訊息是否持久

    JMSExpiration:設定訊息過期時間

    JMSPriority:訊息優先級,0-4被稱為普通訊息,5-9是加急訊息,默認為4

    JMSMessageID:唯一識別每個訊息的標識,由MQ產者或者自己設定

    訊息屬性:除訊息頭以外的值,如識別,去重,重點標注等方法,如textMessage.setStringProperty("c1","VIP");

    訊息體:

    TextMessage:普通字串

    MapMessage:map型別,其中key為String型別,而值為java的基本型別

    BytesMessage:二進制陣列訊息

    StreamMessage:java資料流訊息,用個標準流來順序填充和讀取

    ObjectMessage:物件訊息,包含一個可序列化的java物件

3.JMS的傳送模型:

  • 點對點訊息傳送模型:應用程式由訊息佇列、發送者、接收者組成,每個訊息發送給一個特殊的訊息佇列,該佇列保存了所有發送給它的訊息,處理消費掉的和已過期的訊息

    點對點訊息傳送的特性:

    1.每個訊息只有一個接收者

    2.訊息發送者和接收者沒有時間依賴性

    3.當訊息發送者發送訊息時,無論接收者程式在不在運行,都能發送訊息

    4.當接收者收到訊息時,會發送確認收到通知

  • 發布訂閱訊息傳遞模型:發布者發布一個訊息,該訊息通過topic傳遞給所有訂閱的客戶端,發布者和訂閱者彼此不知道對方,是匿名的且可以動態發布和訊息訂閱,

    發布訂閱訊息傳遞的特性:

    1.一個訊息可以傳遞給多個訂閱者

    2.發布者和訂閱者有時間依賴性

    3.為了緩和嚴格的時間相關性,JMS允許訂閱者創建一個可持久化的訂閱

4.生產者代碼實作

1.引入jar包

<dependency>
    <groupId>org.apache.activemq</groupId>
    <artifactId>activemq-all</artifactId>
    <version>5.16.0</version>
</dependency>

2.生產者代碼

package activemq;

import org.apache.activemq.ActiveMQConnectionFactory;

import javax.jms.Connection;
import javax.jms.JMSException;
import javax.jms.MessageProducer;
import javax.jms.Queue;
import javax.jms.Session;
import javax.jms.TextMessage;

/**
 * @className: Jmsproducer
 * @description: activemq生產者
 * @author: charon
 * @create: 2020-12-27 22:36
 */
public class JmsProducer {
    
    /** 宣告activemq的地址 */
    private static final String ACTIVEMQ_URL = "tcp://192.168.189.150:61616";

    /** 佇列名 */
    private static final String QUEUE_NAME = "queue01";

    /**
     * @param args 引數
     */
    public static void main(String[] args) throws JMSException {
        // 創建連接工廠
        ActiveMQConnectionFactory activeMQConnectionFactory = new ActiveMQConnectionFactory(ACTIVEMQ_URL);
        // 獲得連接
        Connection conn = activeMQConnectionFactory.createConnection();
        conn.start();
        // 創建會話
        Session session = conn.createSession(false,Session.AUTO_ACKNOWLEDGE);
        // 創建佇列
        Queue queue = session.createQueue(QUEUE_NAME);
        // 創建訊息的生產者
        MessageProducer messageProducer = session.createProducer(queue);
        // 創建訊息
        for (int i = 0; i < 5; i++) {
            // 訊息體
            TextMessage textMessage = session.createTextMessage("textMessage:第【 "+i+" 】條訊息");
            // 訊息頭
            // textMessage.setJMSDeliveryMode(DeliveryMode.NON_PERSISTENT));
            // 訊息屬性
            // textMessage.setStringProperty("c1","VIP");
            messageProducer.send(textMessage);
        }
        // 關閉資源
        messageProducer.close();
        session.close();
        conn.close();
    }
}

運行代碼在瀏覽器上查看,可以看到queue01里面有5條訊息:

  • Number Of Pending Messages:等待消費的訊息 這個是當前未出佇列的數量,可以理解為總接收數-總出佇列數
  • Number Of Consumers:消費者的數量
  • Messages Enqueued:進入佇列的訊息 進入佇列的總數量,包括出佇列的, 這個數量只增不減
  • Messages Dequeued:出了佇列的訊息 可以理解為是消費這消費掉的數量

5.消費者代碼實作

package activemq;

import org.apache.activemq.ActiveMQConnectionFactory;

import javax.jms.Connection;
import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.MessageConsumer;
import javax.jms.MessageListener;
import javax.jms.Queue;
import javax.jms.Session;
import javax.jms.TextMessage;
import java.io.IOException;

/**
 * @className: JmsConsumer
 * @description: activeMq的消費者
 * @author: charon
 * @create: 2020-12-28 08:10
 */
public class JmsConsumer {
    /** 宣告activemq的地址 */
    private static final String ACTIVEMQ_URL = "tcp://192.168.189.150:61616";

    /** 佇列名 */
    private static final String QUEUE_NAME = "queue01";

    public static void main(String[] args) throws JMSException, IOException {
        // 創建連接工廠
        ActiveMQConnectionFactory activeMQConnectionFactory = new ActiveMQConnectionFactory(ACTIVEMQ_URL);
        // 獲得連接
        Connection conn = activeMQConnectionFactory.createConnection();
        conn.start();
        // 創建會話
        Session session = conn.createSession(false,Session.AUTO_ACKNOWLEDGE);
        // 創建佇列
        Queue queue = session.createQueue(QUEUE_NAME);
        // 創建訊息的生產者
        MessageConsumer messageConsumer = session.createConsumer(queue);
        // 同步方式,生產環境并不適用,這種方式將阻塞知道獲得并回傳第一條訊息
//        while (true){
//            TextMessage textMessage  =(TextMessage) messageConsumer.receive();
//            if(null!=textMessage){
//                System.out.println("---消費者收到訊息:"+textMessage.getText());
//            }else{
//                break;
//            }
//        }

        // 異步方式,創建監聽,在又訊息到達時,呼叫listener的onMessage方法,
        messageConsumer.setMessageListener(new MessageListener() {
            @Override
            public void onMessage(Message message) {
                if(message != null && message instanceof TextMessage){
                    TextMessage textMessage = (TextMessage) message;
                    System.out.println("--消費者接受到訊息:"+textMessage);
                }
            }
        });
        
        System.in.read();
        // 關閉資源
        messageConsumer.close();
        session.close();
        conn.close();
    }
}

運行消費者的代碼,應該我上面生產者的代碼運行了兩次,所以訊息有10條,

6.activeMQ集群搭建

在這里,筆者使用的基于Zookeeper+levelDb搭建的activeMq集群,為了避免單點故障,使用一主兩從的架構,使用Zookeeper集群注冊所有的ActiveMQ Broker但只有其中一個Broker可以提供服務,它被視為master,也就是說如果master因為故障而不能提供服務,Zookeeper會從SLave中選舉出一個Broker充當master,

我這邊的zookeeper集群已經搭建好了,150和151是follower,152是leader,

# 每臺服務器上安裝activeMq,同時在集群環境下,activemq的jetty.xml檔案重的host要改成0.0.0.0
# 修改activeMq.xml,注釋掉kahadb這個配置,actviemq默認的是kahadb,并且添加leveldb
[root@localhost conf]# vi activemq.xml
<!--        <persistenceAdapter>
            <kahaDB directory="${activemq.data}/kahadb"/>
        </persistenceAdapter> -->
<persistenceAdapter>
   <replicatedLevelDB
      directory="${activemq.data}/leveldb"
      replicas="3"
      <!--實體間的通信地址-->
      bind="tcp://0.0.0.0:62222"
      <!--zookeeper的地址-->
      zkAddress="192.168.189.150:2181,192.168.189.151:2181,192.168.189.152:2181"
      <!--修改為每個服務器的節點的ip-->
      hostname="192.168.189.152"
      sync="local_disk"
      zkPath="/activemq/leveldb-stores"/>
</persistenceAdapter>
# 啟動三個節點的activemq
[root@localhost bin]# ./activemq restart

# 查看 連接zookeeper客戶端
[root@localhost bin]# zkCli.sh
[zk: localhost(CONNECTED) 1] ls /activemq/leveldb-stores
[00000000022, 00000000020, 00000000021]
# 訪問
[zk: 192.168.189.150(CONNECTED) 3] get /activemq/leveldb-stores/00000000020
{"id":"localhost","container":null,"address":"tcp://192.168.189.150:62222","position":-1,"weight":1,"elected":"0000000020"}
[zk: 192.168.189.150(CONNECTED) 4] get /activemq/leveldb-stores/00000000021
{"id":"localhost","container":null,"address":null,"position":-1,"weight":1,"elected":null}
[zk: 192.168.189.150(CONNECTED) 5] get /activemq/leveldb-stores/00000000022
{"id":"localhost","container":null,"address":null,"position":-1,"weight":1,"elected":null}

從上面可以看到,只有00000000020這個幾點的elected里面有值,表明它被選舉為master節點了,

在瀏覽器上依次訪問:192.168.189.150:8161 , 192.168.189.151:8161,192.168.189.152:8161

只有192.168.189.150:8161可以訪問成功,因為只有master節點可以對外提供訪問,所以只有一個節點能訪問到,那么它就是master節點,

第二種查看的方式:

查看activemq的日志,最后一行,可以看到,MasterLevelDBStore即為master節點,SlaveLevelDBStore即為slave節點,

第三種查看的方式為使用zookeeper的可視化工具,

由于activeMq集群是基于zookeeper集群實作的,所以要注意一下三點:

  1. activeMQ的客戶端只能訪問master的Broker,其它處于Slave的Broker不能訪問,所以客戶端連接的Broker應該使用failover協議
  2. 當一個activeMQ節點掛掉或者一個Zookeeper節點掛掉,activeMQ服務正常運轉,但是如果僅剩一個activeMQ節點,由于不能選舉Master,所以activeMQ不能正常運行;(一個就不成集群了)
  3. 同理,如果Zookeeper僅剩一個節點是活動的,不管activeMQ是都存活或者說不管activeMQ個節點是否存活,activeMQ不能正常提供服務,必須依賴于Zookeeper集群服務,

7.集群代碼實作

集群的代碼和上面單機的代碼大致是一直的,就只需要修改一個activemq的地址,

 /** 宣告集群中activemq的地址,使用failover協議,隨機 */
    private static final String ACTIVEMQ_URL = "failover:(tcp://192.168.189.150:61616,tcp://192.168.189.151:61616,tcp://192.168.189.152:61616)?Randomize=false";

8.activemq的高級特性

1.訊息發送方式

默認情況下,非持久化的訊息是異步發送的,持久化的訊息是同步發送的,但是在開啟事務的情況下,訊息都是異步發送的,效率會有2個數量級的提升,所以在發送持久化訊息時,請開啟事務模式,

2.儲存機制

在通常情況下,非持久化的訊息時存盤在記憶體中的,持久化訊息時存盤在檔案中的,他們的最大限制在組態檔中的節點配置的,但是在非持久化訊息堆積到一定程度(記憶體告急)時,actviemq會將記憶體中的非持久化訊息寫入臨時檔案中,以騰出記憶體,但是它和持久化訊息的區別在于,重啟后持久化訊息會從檔案中恢復,非持久化訊息的臨時檔案會洗掉,

所以盡量不要用非持久化檔案,如果非要用的化,可以將臨時檔案的限制調大,同時,非持久化的訊息要及時處理,不要堆積,或者啟動事務,啟動事務后,commit()會等待服務器的訊息回傳,也不會導致訊息丟失了,

3.死信佇列

一條訊息在被重發多次后(默認是6次),將會被ActiveMQ移入死信佇列;說白了就是例外訊息的歸并處理的集合,主要是處理失敗的訊息,可以在activeMQ.DLQ這個佇列中查看,

4.重復訊息,冪等性呼叫

在網路延遲的情況洗啊,可能會造成MQ重試,可能會造成重復消費,如果訊息是做資料庫的插入操作,給這個訊息做一個唯一主鍵,那么就算出現重復消費的情況,因為唯一主鍵,會造成主鍵沖突,避免資料庫出現臟資料,如果是第三方消費,可以在每條資料里面加一個全域唯一的id,如果訊息消費了,就將訊息存在redis中,在消費訊息之前將id到redis中查詢一下,判斷是否消費過,如果沒有消費過,就處理,如果消費過了,就不處理了,

參考網址:

https://blog.csdn.net/weixin_34122548/article/details/91929810?utm_medium=distribute.pc_relevant.none-task-blog-baidujs_title-2&spm=1001.2101.3001.4242

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/244118.html

標籤:Java

上一篇:后端程式員必備:SQL高性能優化方案!50條優化,建議馬上收藏!

下一篇:【Java面向物件】繼承的使用,方法重寫(覆寫),super關鍵字

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more