Java 監控直播流rtsp協議轉rtmp、hls、httpflv協議返回瀏覽器

来源:https://www.cnblogs.com/isyuesen/archive/2022/09/07/16663900.html
-Advertisement-
Play Games

Java 監控直播流rtsp協議轉rtmp、hls、httpflv協議返回瀏覽器 需求背景: 在做之前的項目的時候有一個對接攝像頭實時播放的需求,由於我們攝像頭的購買量不是很多,海康威視不給我們提供流媒體雲伺服器,所以需要我們自己去 一個去滿足我們能在瀏覽器看到監控畫面。項目源代碼在以前公司沒有拷貝 ...


Java 監控直播流rtsp協議轉rtmp、hls、httpflv協議返回瀏覽器

目錄

需求背景:

在做之前的項目的時候有一個對接攝像頭實時播放的需求,由於我們攝像頭的購買量不是很多,海康威視不給我們提供流媒體雲伺服器,所以需要我們自己去 一個去滿足我們能在瀏覽器看到監控畫面。項目源代碼在以前公司沒有拷貝就不能複習,最近又在準備面試,所以寫了這個博客來複盤和擴展一下,由於我現在沒有Liunx,我就用Windows來演示,生產環境還是要使用Liunx,下麵這些操作在Liunx也是一樣的流程,大家自行百度。

一:瞭解音視頻流協議:

媒體流協議對比
協議HttpFlvRTMPHLSDash
全稱FLASH VIDEO over HTTPReal Time Message ProtocolHTTP Living Streaming
傳輸方式HTTP長連接TCP長連接HTTP短連接HTTP短連接
視頻封裝格式FLV

FLV TAG

TS文件

Mp4

3gp

webm

原理

RTMP,使用HTTP協議(80埠)

每個時刻的數據收到後立刻轉發

集合一段時間的數據,生成TS切片文件(三片),並更新m3u8索引

延時

1~3秒

1~3秒

5~20秒(依切片情況)

數據分段連續流連續流切片文件切片文件
Html5播放

可通過HTML5解封包播放

(flv.js)

不支持

可通過HTML5解封包播放

(hls.js)

如果dash文件列表是MP4,

webm文件,可直接播放

其它

需要Flash技術支持,不支持多音頻流、多視頻流,不便於seek(即拖進度條)

跨平臺支持較差,需要Flash技術支持

播放時需要多次請求,對於網路質量要求高

二:方案一 rtsp 轉rtmp

1、下載nginx + nginx-rtmp-module

nginx:下載地址:http://nginx-win.ecsds.eu/download/

nginx-rtmp-module:nginx 的擴展,安裝後支持rtmp協議,下載地址:https://github.com/arut/nginx-rtmp-module

解壓nginx-rtmp-module到nginx根目錄下,並修改其文件夾名為nginx-rtmp-module(原名為nginx-rtmp-module-master)

2、nginx配置文件

到nginx根目錄下的conf目錄下複製一份nginx-win.conf 重命名 nginx-win-rtmp.conf

1662446414092

nginx-win-rtmp.conf:

#user  nobody;
# multiple workers works !
worker_processes  2;

#error_log  logs/error.log;
#error_log  logs/error.log  notice;
#error_log  logs/error.log  info;

#pid        logs/nginx.pid;

events {
    worker_connections  8192;
    # max value 32768, nginx recycling connections+registry optimization = 
    #   this.value * 20 = max concurrent connections currently tested with one worker
    #   C1000K should be possible depending there is enough ram/cpu power
    # multi_accept on;
}

rtmp {
    server {
        listen 1935;
        chunk_size 4000;
		
        application live {
             live on;
             # 播放時進行回調,如果HttpRespone statusCode不等於200會斷開
			 # on_play http://localhost:8081/auth;
        }
		
		application hls {
		     live on; 
		     # 開啟hls切片
             hls on;
             # m3u8地址
			 hls_path html/hls;
			 # 一個切片多少秒
			 hls_fragment 8s;
			 # on_play http://localhost:8081/auth;
			 # on_publish http://localhost:8081/auth;
			 # on_done http://localhost:8081/auth;
        }
    }
}

http {
    #include      /nginx/conf/naxsi_core.rules;
    include       mime.types;
    default_type  application/octet-stream;

    #log_format  main  '$remote_addr:$remote_port - $remote_user [$time_local] "$request" '
    #                  '$status $body_bytes_sent "$http_referer" '
    #                  '"$http_user_agent" "$http_x_forwarded_for"';

    #access_log  logs/access.log  main;

#     # loadbalancing PHP
#     upstream myLoadBalancer {
#         server 127.0.0.1:9001 weight=1 fail_timeout=5;
#         server 127.0.0.1:9002 weight=1 fail_timeout=5;
#         server 127.0.0.1:9003 weight=1 fail_timeout=5;
#         server 127.0.0.1:9004 weight=1 fail_timeout=5;
#         server 127.0.0.1:9005 weight=1 fail_timeout=5;
#         server 127.0.0.1:9006 weight=1 fail_timeout=5;
#         server 127.0.0.1:9007 weight=1 fail_timeout=5;
#         server 127.0.0.1:9008 weight=1 fail_timeout=5;
#         server 127.0.0.1:9009 weight=1 fail_timeout=5;
#         server 127.0.0.1:9010 weight=1 fail_timeout=5;
#         least_conn;
#     }

    sendfile        off;
    #tcp_nopush     on;

    server_names_hash_bucket_size 128;

## Start: Timeouts ##
    client_body_timeout   10;
    client_header_timeout 10;
    keepalive_timeout     30;
    send_timeout          10;
    keepalive_requests    10;
## End: Timeouts ##

    #gzip  on;

    server {
        listen       5080;
        server_name  localhost;


        location /stat {
            rtmp_stat all;
            rtmp_stat_stylesheet stat.xsl;
        }
        location /stat.xsl {
            root nginx-rtmp-module/;
        }
        location /control {
            rtmp_control all;
        }
		
		location /hls {
            # Serve HLS fragments
            types {
                application/vnd.apple.mpegurl m3u8;
                video/mp2t ts;
            }
            expires -1;
            add_header Access-Control-Allow-Origin *;
        }

        #charset koi8-r;
        #access_log  logs/host.access.log  main;

        ## Caching Static Files, put before first location
        #location ~* \.(jpg|jpeg|png|gif|ico|css|js)$ {
        #    expires 14d;
        #    add_header Vary Accept-Encoding;
        #}

# For Naxsi remove the single # line for learn mode, or the ## lines for full WAF mode
        location / {
            #include    /nginx/conf/mysite.rules; # see also http block naxsi include line
            ##SecRulesEnabled;
         ##DeniedUrl "/RequestDenied";
         ##CheckRule "$SQL >= 8" BLOCK;
         ##CheckRule "$RFI >= 8" BLOCK;
         ##CheckRule "$TRAVERSAL >= 4" BLOCK;
         ##CheckRule "$XSS >= 8" BLOCK;
            root   html;
            index  index.html index.htm;
        }

# For Naxsi remove the ## lines for full WAF mode, redirect location block used by naxsi
        ##location /RequestDenied {
        ##    return 412;
        ##}

## Lua examples !
#         location /robots.txt {
#           rewrite_by_lua '
#             if ngx.var.http_host ~= "localhost" then
#               return ngx.exec("/robots_disallow.txt");
#             end
#           ';
#         }

        #error_page  404              /404.html;

        # redirect server error pages to the static page /50x.html
        #
        error_page   500 502 503 504  /50x.html;
        location = /50x.html {
            root   html;
        }

        # proxy the PHP scripts to Apache listening on 127.0.0.1:80
        #
        #location ~ \.php$ {
        #    proxy_pass   http://127.0.0.1;
        #}

        # pass the PHP scripts to FastCGI server listening on 127.0.0.1:9000
        #
        #location ~ \.php$ {
        #    root           html;
        #    fastcgi_pass   127.0.0.1:9000; # single backend process
        #    fastcgi_pass   myLoadBalancer; # or multiple, see example above
        #    fastcgi_index  index.php;
        #    fastcgi_param  SCRIPT_FILENAME  $document_root$fastcgi_script_name;
        #    include        fastcgi_params;
        #}

        # deny access to .htaccess files, if Apache's document root
        # concurs with nginx's one
        #
        #location ~ /\.ht {
        #    deny  all;
        #}
    }

    # another virtual host using mix of IP-, name-, and port-based configuration
    #
    #server {
    #    listen       8000;
    #    listen       somename:8080;
    #    server_name  somename  alias  another.alias;

    #    location / {
    #        root   html;
    #        index  index.html index.htm;
    #    }
    #}

    # HTTPS server
    #
    #server {
    #    listen       443 ssl spdy;
    #    server_name  localhost;

    #    ssl                  on;
    #    ssl_certificate      cert.pem;
    #    ssl_certificate_key  cert.key;
    #    ssl_session_timeout  5m;
    #    ssl_prefer_server_ciphers On;
    #    ssl_protocols TLSv1 TLSv1.1 TLSv1.2;
    #    ssl_ciphers ECDH+AESGCM:ECDH+AES256:ECDH+AES128:ECDH+3DES:RSA+AESGCM:RSA+AES:RSA+3DES:!aNULL:!eNULL:!MD5:!DSS:!EXP:!ADH:!LOW:!MEDIUM;

    #    location / {
    #        root   html;
    #        index  index.html index.htm;
    #    }
    #}

}

3、cmd 到nginx根目錄啟動nginx

nginx.exe -c conf\nginx-win-rtmp.conf

測試:瀏覽器輸入 http://localhost:5080/stat,看到

1662447230276

代表安裝成功

4、下載ffmpeg安裝

ffmpeg:一個處理音視頻強大的庫,我們需要用它來轉協議,下載地址:https://www.gyan.dev/ffmpeg/builds/ ,這裡可以下載essential和full版本,essential就是簡版,只包含ffmpeg.exe、ffplay.exe、
ffprobe.exe, 而full版本就包含了動態庫和相關頭文件,方便我們在開發中調用。1662447683259

5、配置ffmpeg環境變數

將ffmpeg解壓後裡面的bin路徑複製到Path裡面去1662447882146

6、測試ffmpeg

cmd ffmpeg -version 命令看到代表成功
1662448084603

7、下載VLC播放器

下載地址:https://www.videolan.org/vlc/
1662448603611

8、查攝像頭的rtsp協議格式

我這裡截圖是海康威視的

1662449507796

1662449600452

現在沒有測試的流,我找了個點播的rtsp

rtsp://wowzaec2demo.streamlock.net/vod/mp4:BigBuckBunny_115k.mp4,用這個代替是一樣的

9、執行ffmpeg命令

ffmpeg強大,命令也是複雜,我們cmd 執行

ffmpeg -re -rtsp_transport tcp -stimeout 20000000 -i "rtsp://wowzaec2demo.streamlock.net/vod/mp4:BigBuckBunny_115k.mp4" -buffer_size 1024000 -max_delay 500000 -codec:v libx264 -r 25 -rtbufsize 10M -s 1280x720 -map:v 0 -an -f flv rtmp://127.0.0.1:1935/live/test

rtsp://wowzaec2demo.streamlock.net/vod/mp4:BigBuckBunny_115k.mp4,是輸入源頭
rtmp://127.0.0.1:1935/live/test 是輸出地址

如果沒有報錯的話,到現在rtsp就已經轉換好了

ffmpeg命令學習:https://www.jianshu.com/p/df3216a52e59https://blog.csdn.net/fuhanghang/article/details/123565920

10、測試rtmp是否轉換成功

我們打開VLC,媒體->打開網路串流->輸入 rtmp://127.0.0.1:1935/live/test-> 播放

1662450258097

11、測試是否成功

等待幾秒鐘看到有視頻播放就是成功了

1662450826524

12、為什麼放棄了用rtmp

rtmp的優點是延遲低,效率高,但是在瀏覽器需要安裝flash才能放,也就老版本的瀏覽器在用,rtmp可能會在別的地方支持,所以還是把他方式方法貼出來了。

三:方案二 rtsp轉hls

1、nginx配置:

在前面已經貼出來了,其中這幾個是針對hls的

1662453005569

1662453069930

2、執行ffmepg命令

ffmpeg -i "rtsp://wowzaec2demo.streamlock.net/vod/mp4:BigBuckBunny_115k.mp4" -vcodec libx264 -acodec aac -f flv rtmp://127.0.0.1:1935/hls/test

3、查看nginx根目錄 -> hls -> test.m3u8 是否生成

生成了代表一切正常

1662453591008

4、m3u8在網頁上播放

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <title>前端播放m3u8格式視頻</title>
    <!--https://www.bootcdn.cn/video.js/-->
    <link href="https://cdn.bootcss.com/video.js/7.6.5/alt/video-js-cdn.min.css" rel="stylesheet">
    <script src="https://cdn.bootcss.com/video.js/6.6.2/video.js"></script>
    <!--https://www.bootcdn.cn/videojs-contrib-hls/-->
    <script src="https://cdn.bootcss.com/videojs-contrib-hls/5.15.0/videojs-contrib-hls.min.js"></script>
</head>
<body>
    <video id="myVideo" class="video-js vjs-default-skin vjs-big-play-centered" controls preload="auto" width="1080" height="708" data-setup='{}'>    
        <source id="source" src="http://127.0.0.1:5080/hls/test.m3u8"  type="application/x-mpegURL">
    </video>
</body>
<script>    
    // videojs 簡單使用  
    var myVideo = videojs('myVideo',{
        bigPlayButton : true, 
        textTrackDisplay : false, 
        posterImage: false,
        errorDisplay : false,
    })
    myVideo.play() // 視頻播放
    myVideo.pause() // 視頻暫停
</script>
</html>

source標簽的src屬性:http://你的nginx ip:nginx http埠/hls/test.m3u8

1662453869453

rtsp轉HLS成功!

5、認識一下m3u8格式

m3u8文件裡面存儲了一個索引,以文本格式打開是這樣的

#EXTM3U
#EXT-X-VERSION:3
#EXT-X-MEDIA-SEQUENCE:56
#EXT-X-TARGETDURATION:13
#EXTINF:10.381,
test-56.ts
#EXTINF:10.422,
test-57.ts
#EXTINF:13.453,
test-58.ts

m3u8文件它不是視頻源,源頭是ts尾碼文件
1662454031515

6、為什麼放棄了用HLS

轉HLS協議及網頁載入過程:

ffmepg收到rtsp的流時候,會等一個切片的時間,一個切片時間到了,切片ts會放到伺服器中,同時m3u8文件中加一個索引,對應著新進入的切片。網頁在載入m3u8的時候,就是讀取m3u8中的的索引去載入ts文件,所以在不斷的請求ts,對ts進行解析,不斷的和TCP握手,這就是為什麼HLS延遲高和對網速的要求高的原因,我們監控肯定是要延遲低的,HLS相容性好,適合點播。

四:方案三rtsp 轉httpflv(採用)

1、安裝nginx-flv-module

這個插件需要編譯,教程:https://blog.csdn.net/KayChanGEEK/article/details/105095844

我這裡已經編譯好了,直接下載啟動:

https://gitee.com/isyuesen/nginx-flv-file

2、nginx配置

看我git裡面的https://gitee.com/isyuesen/nginx-flv-file/blob/master/conf/nginx.conf,和預設的config差別主要是添加了這幾個

rtmp {  
    server {  
        listen 1935;
        # 流復用的最大塊大小
        chunk_size 4000;  
        application liveapp { 
            live on;
            # 推流開始
			on_publish http://localhost:8081/auth;
			# 推流關閉
			on_publish_done http://localhost:8081/auth;
			# 客戶端開始播放
			on_play http://localhost:8081/auth;
			# 客戶端結束播放
			on_play_done http://localhost:8081/auth;
        }  
    }  
}
location /live {
    flv_live on;
    chunked_transfer_encoding on;
    add_header 'Access-Control-Allow-Credentials' 'true'; #add additional HTTP header
    add_header 'Access-Control-Allow-Origin' '*'; #add additional HTTP header
    add_header Access-Control-Allow-Headers X-Requested-With;
    add_header Access-Control-Allow-Methods GET,POST,OPTIONS;
    add_header 'Cache-Control' 'no-cache';
}

3、做java許可權認證

nginx rtmp配置中有配置on_publish鉤子介面 http://localhost:8081/auth,這個回調HttpResponse stausCode如果不等於200會拒絕I/O,更多回調鉤子看:https://github.com/arut/nginx-rtmp-module/wiki/Directives#on_connect

@PostMapping("/auth")
    public void getVideo(String token, HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
        if (token.equals("tokenValue")) {
            httpServletResponse.setStatus(200);
        } else {
            // 拒絕服務
            httpServletResponse.setStatus(500);
        }
    }

4、執行ffmepg命令:

ffmpeg -re  -rtsp_transport tcp -i "rtsp://wowzaec2demo.streamlock.net/vod/mp4:BigBuckBunny_115k.mp4" -f flv -vcodec h264 -vprofile baseline -acodec aac -ar 44100 -strict -2 -ac 1 -f flv -s 640*360 -q 10 "rtmp://127.0.0.1:1935/liveapp/test"

4.1 採用java代碼去執行ffmepg命令

依賴 javaCV

<dependency>
    <groupId>org.bytedeco</groupId>
    <artifactId>javacv-platform</artifactId>
    <version>1.5.2</version>
</dependency>
public class App {
    public static void main( String[] args ) throws IOException, InterruptedException {
        String name = "test";
        // rtsp地址
        String rtspDir = "rtsp://wowzaec2demo.streamlock.net/vod/mp4:BigBuckBunny_115k.mp4";
        // rtmp地址
        String rtmpDir = "rtmp://192.168.0.140:1935/liveapp/" + name + "?token=tokenValue";

        String ffmpeg = Loader.load(org.bytedeco.ffmpeg.ffmpeg.class);
        ProcessBuilder pb = new ProcessBuilder(ffmpeg,
                "-re",
                "-rtsp_transport",
                "tcp",
                "-i",
                rtspDir,
                "-f",
                "flv",
                "-vcodec",
                "h264",
                "-vprofile",
                "baseline",
                "-acodec",
                "aac",
                "-ar",
                "44100",
                "-strict",
                "-2",
                "-ac",
                "1",
                "-f",
                "flv",
                "-s",
                "640*360",
                "-q",
                "10",
                rtmpDir
        );
        pb.inheritIO().start().waitFor();
    }
}

5、測試http-flv鏈接

如果你跟著我做的,那鏈接就是 http://127.0.0.1:18080/live?port=1935&app=liveapp&stream=test&token=tokenValue,在VLC播放器中點擊媒體 -> 打開網路串流 -> 輸入http://127.0.0.1:18080/live?port=1935&app=liveapp&stream=test&token=tokenValue -> 播放

有視頻證明你離成功就差最後一步

6、前端使用flv.js播放:

<!doctype html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta name="viewport"
        content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
  <meta http-equiv="X-UA-Compatible" content="ie=edge">
  <title>播放http-flv</title>
</head>
<body>
<video id="videoElement"></video>
<script src="https://cdn.bootcdn.net/ajax/libs/flv.js/1.6.2/flv.min.js"></script>
<script>
  if (flvjs.isSupported()) {
      const videoElement = document.getElementById('videoElement');
      const flvPlayer = flvjs.createPlayer({
        type: 'flv',
        url: 'http://127.0.0.1:18080/live?port=1935&app=liveapp&stream=test&token=tokenValue'
      });
      flvPlayer.attachMediaElement(videoElement);
      flvPlayer.load();
      flvPlayer.play();
	}
</script>
</body>
</html>

7、大功告成

1662482995076


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • vue簡介 1.vue是什麼 一套用於構建用戶界面的漸進式javascript框架 構建用戶界面:拿到的數據轉換為用戶可以看到的數據 漸進式:vue可以自底向上逐層用 從一個輕量小巧的庫逐漸遞進到使用各式各樣的vue插件 2.vue開發者(老二次元了,哈哈) 2013 | 受到Angular的啟發, ...
  • JavaScript 鉤子 點擊打開視頻講解更加詳細 可以在 attribute 中聲明 JavaScript 鉤子 完整案例: <template> <div id="app"> <div id="example-3"> <button @click="show = !show"> Toggle ...
  • 自定義一個類似jquer庫==> Hq庫的具體步驟: // 1.通過$_$('div')方法就可以直接返回一個由Hq構造函數實例化出來的一個對象 // 2.在通過Hq構造函數,獲取你以選擇器為參數的所有節點 // 3.Hq在調用原型上addEles方法將所有的節點,賦值給對應的this的每一項(實例 ...
  • 使用three.js(webgl)搭建智慧樓宇、3D定位、三維室內定位、設備檢測、數字孿生、物聯網3D、物業3D監控、物業基礎設施可視化運維、3d建築,3d消防,消防演習模擬,3d庫房,webGL,threejs,3d機房,bim管理系統 ...
  • js實現將excel表格copy到頁面 點擊打開視頻講解更加詳細 其實最核心的技術,還是copy的是我們粘貼板上的數據 就像平常怎麼粘貼複製其他的數據一樣,只是我們在excel粘貼的是一個表格數據 這時我們首先也時獲取我們粘貼板上的數據,如何對粘貼板上的數據進行處理,處理成 我們想要的表格形式。 完 ...
  • 蒼穹之邊,浩瀚之摯,眰恦之美; 悟心悟性,善始善終,惟善惟道! —— 朝槿《朝槿兮年說》 寫在開頭 對於Java領域中的鎖,其實從接觸Java至今,我相信每一位Java Developer都會有這樣的一個感覺?不論是Java對鎖的實現還是應用,真的是一種“群英薈萃”,而且每一種鎖都有點各有各的驢,各 ...
  • 正反向查詢進階操作 '''正反向查詢進階操作''' # 1.查詢主鍵為1的書籍對應的出版社名稱及書名 res = models.Publish.objects.filter(book__pk=1).values('name', 'book__title') print(res) # <QuerySe ...
  • 單例模式,是java設計模式中最簡單的設計模式,是屬於創建類型模式。單例模式就是只能有一個實例,即一個類有且僅有一個實例,並且自行實例化向整個系統提供。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...