Spring boot + LayIM + t-io 實現(xiàn)文件上傳、 監(jiān)聽用戶狀態(tài)的實例代碼
前言
今天的主要內(nèi)容是:LayIM消息中圖片,文件的上傳對接、用戶狀態(tài)的監(jiān)聽、群在線人數(shù)的監(jiān)聽。下面我將挨個介紹。
圖片上傳
關(guān)于Spring boot中的文件上傳的博客很多,我也是摘抄了部分代碼。上傳部分簡單介紹,主要介紹在開發(fā)過程中遇到的問題。首先我們看一下LayIM的相應(yīng)的接口:
layim.config({ //上傳圖片接口 ,uploadImage: {url: '/upload/file'} //上傳文件接口 ,uploadFile: {url: '/upload/file'} //其他代碼 })
是的,沒錯我們只要寫兩個接口就能實現(xiàn)LayIM中發(fā)圖片,發(fā)文件的功能了,我這里省點事,由于LayIM已經(jīng)判斷了文件類型,所以我這里只用了一個上傳接口。返回的格式是醬紫的:
{"code":0,"msg":"success","data":{"src":"/upload/d8740baa-cf7e-497c-a085-5c6ed9633f35.gif","name":"8.gif"}}
上傳代碼就很簡單了,獲取文件后綴,生成guid的名稱,保存到相應(yīng)文件夾下。最后返回LayIM想要的響應(yīng)數(shù)據(jù)。
剛開始想的是直接傳到 /resources/static/upload/文件夾下,后來發(fā)現(xiàn)上傳成功之后訪問路徑會出現(xiàn)404的情況,原因就是:即使文件夾下有那個文件,但是target相應(yīng)的文件夾下沒有,所以重新編譯運行之后才可以,那這樣肯定不行。于是我就找解決方案,后來看到這么一種處理方式,就是把每個文件請求也映射到Controller中的一個路徑上,然后使用 ResourceLoader 去找響應(yīng)的文件。代碼如下:
/** * 上傳文件的路徑配置 * */ @Value("${layim.upload.dir}") private String fileDirPath; private final ResourceLoader resourceLoader; @Autowired public UploadController(ResourceLoader resourceLoader){ this.resourceLoader = resourceLoader; }
獲取文件代碼,那么在訪問那個文件的時候,指定文件名,Resource就會根據(jù)路徑返回相應(yīng)的資源。
/** * 訪問文件路徑,使用resourceLoader獲取文件 * */ @GetMapping(value = "/{filename:.+}") @ResponseBody public ResponseEntity<?> getFile(@PathVariable String filename) { try { String location = "file:"+ Paths.get(fileDirPath, filename).toString(); Resource resource = resourceLoader.getResource(location); return ResponseEntity.ok(resource); } catch (Exception e) { return ResponseEntity.notFound().build(); } }
看一下效果:(為了給大家做演示截圖的時候遇到上傳文件大小限制的問題,配置改一下就可以了。)
http: multipart: max-file-size: 30MB max-request-size: 30MB
上文中的代碼 value 配置為:filename:.+,這樣我們訪問路徑如果遇到 比如 /abc.jpg 那么就會匹配上 getFile這個方法。然后由ResourceLoader幫我們找文件。不過為了多了解一點,我跟蹤了一下源代碼。先看一下DefaultResourceLoader中的getResource方法。
@Override public Resource getResource(String location) { Assert.notNull(location, "Location must not be null"); for (ProtocolResolver protocolResolver : this.protocolResolvers) { Resource resource = protocolResolver.resolve(location, this); if (resource != null) { return resource; } } //我們是使用file:開頭的 if (location.startsWith("/")) { return getResourceByPath(location); } //這里判斷是不是走 classpath: else if (location.startsWith(CLASSPATH_URL_PREFIX)) { return new ClassPathResource(location.substring(CLASSPATH_URL_PREFIX.length()), getClassLoader()); } else { try { //最后將文件地址轉(zhuǎn)化成了url // Try to parse the location as a URL... URL url = new URL(location); return new UrlResource(url); } catch (MalformedURLException ex) { // No URL -> resolve as resource path. return getResourceByPath(location); } } }
后來在我調(diào)試的過程中發(fā)現(xiàn),比如我訪問的路徑是 /abc.gif,那么他后臺就會轉(zhuǎn)換成這個地址:file:///G:/JavaProject/SpringBootLayIM/upload/abc.gif ,所以同理,在本機(jī)上直接將這個地址打開也是能夠訪問的。上文中的地址(file:///..)就是 layim.upload.dir 的值。同樣,我把文件夾改為resources/static/upload/下同樣適用,因為,通過ResourceLoader,他將文件定位到了服務(wù)器的物理地址。
target下甚至連uplaod文件夾都沒有,但是文件還是能夠訪問到
文件上傳
和圖片用的是同一個接口,差別在于上傳大文件需要改一下配置,上文中已經(jīng)寫到,這里不在詳述。
用戶在線狀態(tài)
在LayIM的狀態(tài)監(jiān)聽中,有這么一項:
//監(jiān)聽天窗口的切換 layim.on('chatChange', function(res){ console.log(res.data); });
其實在讀取列表的時候就應(yīng)該要加載用戶在線離線狀態(tài)。不過這里只演示,當(dāng)窗口切換時候,查一下對面的狀態(tài)。所以,在觸發(fā)chatChange事件后,我們向服務(wù)器發(fā)送一條請求。
var t = res.data.type=='friend'; socket.send({ mtype:t? socket.mtype.checkIsOnline:socket.mtype.checkOnlineCount, id:res.data.id });
上面有兩個情況,第一種單聊的時候,我需要知道對方的狀態(tài)。第二種,群聊的時候我想知道有多少人在線。所以稍微做了下區(qū)分。
新加兩個消息處理類:
其實判斷是否在線的代碼如下:
ChannelContext checkChannelContext = Aio.getChannelContextByUserid(channelContext.getGroupContext(),body.getId()); //檢查是否在線 boolean isOnline = checkChannelContext != null && !checkChannelContext.isClosed();
然后封裝消息返回給服務(wù)器。消息流程在 單聊群聊的實現(xiàn) 中已經(jīng)介紹過,這里不在贅述。
判斷群里有多少人的方法如下:
SetWithLock<ChannelContext> channelLocks = Aio.getChannelContextsByGroup(channelContext.getGroupContext(),body.getId()); int onlineCount = channelLocks.getObj().size();
看一下效果,數(shù)據(jù)量有點小,就兩個用戶,不過作為演示還是夠用噠。
另外的賬號登錄后:
總結(jié)
本篇內(nèi)容不是很多,就是一個文件的上傳和tio中的個別api的簡單應(yīng)用,不過令我興奮的是在我調(diào)試代碼的時候,發(fā)現(xiàn)了很多可玩的東西,比如這玩意:
當(dāng)然,作者已經(jīng)在框架介紹中介紹過了,可以監(jiān)聽每個客戶端的發(fā)送消息情況,作為統(tǒng)計之類的。所以,下一步可以拿這些數(shù)據(jù)搞一些事情。
最后,由于剩下的內(nèi)容就是一些簡單的消息歷史記錄,消息的保存,好友申請等增刪改查的東西,后續(xù)不在過多介紹。項目正式寫完之后在發(fā)一篇。接下來的準(zhǔn)備玩玩t-io里面的數(shù)據(jù)~~(等等,是不是跑偏了,本來是學(xué)習(xí)springboot的。。。)
代碼地址: https://github.com/fanpan26/SpringBootLayIM
總結(jié)
以上所述是小編給大家介紹的Spring boot + LayIM + t-io 實現(xiàn)文件上傳、 監(jiān)聽用戶狀態(tài)的實例代碼,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
使用MyBatis-Plus實現(xiàn)聯(lián)表查詢分頁的示例代碼
本文主要講述了如何在SpringBoot項目中使用MyBatis-Plus的分頁插件,通過這個示例,可以學(xué)會如何利用MyBatis-Plus進(jìn)行高效的分頁查詢,感興趣的可以了解一下2024-10-10Java?C++題解leetcode672燈泡開關(guān)示例
這篇文章主要為大家介紹了Java?C++題解leetcode672燈泡開關(guān)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09SpringBoot實現(xiàn)多個子域共享cookie的示例
本文主要介紹了SpringBoot實現(xiàn)多個子域共享cookie的示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-04-04Java開發(fā)環(huán)境配置教程(win7 64bit)
這篇文章主要為大家詳細(xì)介紹了win7 64bit下Java開發(fā)環(huán)境的配置教程,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-08-08基于SpringBoot和Vue3的博客平臺文章詳情與評論功能實現(xiàn)
在前面的教程中,我們已經(jīng)實現(xiàn)了基于Spring Boot和Vue3的發(fā)布、編輯、刪除文章功能以及文章列表與分頁功能。本教程將引導(dǎo)您實現(xiàn)博客平臺的文章詳情與評論功能,需要的朋友可以參考一下2023-04-04