模仿J2EE的session機制的App後端會話信息管理

来源:http://www.cnblogs.com/niuxiaozu/archive/2017/11/23/7886600.html
-Advertisement-
Play Games

為App後端服務模擬Session創建一個tokenMapPool,透明化token管理,方便開發者存取會話信息。封裝了請求響應報文,以便做統一處理,修改了springMvc的參數轉換器,適應報文。 ...


此文章只將思想,不提供具體完整實現(博主太懶,懶得整理),有疑問或想瞭解的可以私信或評論

背景

  在傳統的java web 中小型項目中,一般使用session暫存會話信息,比如登錄者的身份信息等。此機制是借用http的cookie機制實現,但是對於app來說每次請求都保存並共用cookie信息比較麻煩,並且傳統的session對集群並不友好,所以一般app後端服務都使用token來區分用戶登錄信息。

  j2ee的session機制大家都很瞭解,使用非常方便,在傳統java web應用中很好用,但是在互聯網項目中或用得到集群的一些項目就有些問題,比如序列化問題,同步的延時問題等等,所以我們需要一個使用起來類似session的卻能解決得了集群等問題的一個工具。

方案

  我們使用cache機制來解決這個問題,比較流行的redis是個nosql記憶體資料庫,而且帶有cache的失效機制,很適合做會話數據的存儲。而token字元串需要在第一次請求時伺服器返回給客戶端,客戶端以後每次請求都使用這個token標識身份。為了對業務開發透明,我們把app的請求和響應做的報文封裝,只需要對客戶端的http請求工具類做點手腳,對服務端的mvc框架做點手腳就可以了,客戶端的http工具類修改很簡單,主要是服務端的協議封裝。

實現思路

  一、制定請求響應報文協議。

  二、解析協議處理token字元串。

  三、使用redis存儲管理token以及對應的會話信息。

  四、提供保存、獲取會話信息的API。

  我們逐步講解下每一步的實現方案。

一、制定請求響應報文協議。

  既然要封裝報文協議,就需要考慮什麼是公共欄位,什麼是業務欄位,報文的數據結構等。

  請求的公共欄位一般有token、版本、平臺、機型、imei、app來源等,其中token是我們這次的主角。

  響應的公共欄位一般有token、結果狀態(success,fail)、結果碼(code)、結果信息等。

  報文數據結構,我們選用json,原因是json普遍、可視化好、位元組占用低。

請求報文如下,body中存放業務信息,比如登錄的用戶名和密碼等。

{
    "token": "客戶端token",
    /**客戶端構建版本號*/
    "version": 11,
    /**客戶端平臺類型*/
    "platform": "IOS",
    /**客戶端設備型號*/
    "machineModel": "Iphone 6s",
    "imei": "客戶端串號(手機)",
    /**真正的消息體,應為map*/
    "body": {
        "key1": "value1",
        "key2": {
            "key21": "value21"
        },
        "key3": [
            1,
            2
        ]
    }
}

響應的報文

 {
        /**是否成功*/
        "success": false,
        /**每個請求都會返回token,客戶端每次請求都應使用最新的token*/
        "token": "伺服器為當前請求選擇的token",
        /**失敗碼*/
        "failCode": 1,
        /**業務消息或者失敗消息*/
        "msg": "未知原因",
        /**返回的真實業務數據,可為任意可序列化的對象*/
        "body": null
    }
}

二、解析協議處理token字元串。

  服務端的mvc框架我們選用的是SpringMVC框架,SpringMVC也比較普遍,不做描述。

  暫且不提token的處理,先解決制定報文後怎麼做參數傳遞。

  因為請求信息被做了封裝,所以要讓springmvc框架能正確註入我們在Controller需要的參數,就需要對報文做解析和轉換。

  要對請求信息做解析,我們需要自定義springmvc的參數轉換器,通過實現HandlerMethodArgumentResolver介面可以定義一個參數轉換器

  RequestBodyResolver實現resolveArgument方法,對參數進行註入,以下代碼為示例代碼,切勿拿來直用。

 

        @Override
    public Object resolveArgument(MethodParameter parameter,
            ModelAndViewContainer mavContainer, NativeWebRequest webRequest,
            WebDataBinderFactory binderFactory) throws Exception {
        String requestBodyStr = webRequest.getParameter(requestBodyParamName);//獲取請求報文,可以使用任意方式傳遞報文,只要在這獲取到就可以
        if(StringUtils.isNotBlank(requestBodyStr)){
            String paramName = parameter.getParameterName();//獲取Controller中參數名
            Class<?> paramClass = parameter.getParameterType();//獲取Controller中參數類型
            /* 通過json工具類解析報文 */
            JsonNode jsonNode = objectMapper.readTree(requestBodyStr);
            if(paramClass.equals(ServiceRequest.class)){//ServiceRequest為請求報文對應的VO
                ServiceRequest serviceRequest = objectMapper.readValue(jsonNode.traverse(),ServiceRequest.class);
                return serviceRequest;//返回這個object就是註入到參數中了,一定要對應類型,否則異常不容易捕獲
            }
            if(jsonNode!=null){//從報文中查找Controller中需要的參數
                JsonNode paramJsonNode = jsonNode.findValue(paramName);
                if(paramJsonNode!=null){
                    return objectMapper.readValue(paramJsonNode.traverse(), paramClass);
                }
                
            }
        }
        return null;
    }

 

  將自己定義的參數轉換器配置到SrpingMVC的配置文件中<mvc:argument-resolvers>

<mvc:argument-resolvers>
    <!-- 統一的請求信息處理,從ServiceRequest中取數據 -->
          <bean id="requestBodyResolver" class="com.niuxz.resolver.RequestBodyResolver">
              <property name="objectMapper"><bean class="com.shoujinwang.utils.json.ObjectMapper"></bean></property>
              <!-- 配置請求中ServiceRequest對應的欄位名,預設為requestBody -->
              <property name="requestBodyParamName"><value>requestBody</value></property>
          </bean>
</mvc:argument-resolvers>

  這樣就可以使報文中的參數能被springmvc正確識別了。

  接下來我們要對token做處理了,我們需要添加一個SrpingMVC攔截器將每次請求都攔截下來,這屬於常用功能,不做細節描述

Matcher m1 =Pattern.compile("\"token\":\"(.*?)\"").matcher(requestBodyStr);
   
if(m1.find()){
    token = m1.group(1);
}
tokenMapPool.verifyToken(token);//對token做公共處理,驗證

  這樣就簡單的獲取到了token了,可以做公共處理了。

三、使用redis存儲管理token以及對應的會話信息。

  其實就是寫一個redis的操作工具類,因為使用了spring作為項目主框架,而且我們用到redis的功能並不多,所以直接使用spring提供的CacheManager功能

  配置org.springframework.data.redis.cache.RedisCacheManager

 

<!-- 緩存管理器  全局變數等可以用它存取-->
<bean id="cacheManager" class="org.springframework.data.redis.cache.RedisCacheManager">
    <constructor-arg>
        <ref bean="redisTemplate"/>
    </constructor-arg>
    <property name="usePrefix" value="true" />
    <property name="cachePrefix">
        <bean class="org.springframework.data.redis.cache.DefaultRedisCachePrefix">
            <constructor-arg name="delimiter" value=":@WebServiceInterface"/>
        </bean>
    </property>
    <property name="expires"><!-- 緩存有效期 -->
        <map>
            <entry>
                <key><value>tokenPoolCache</value></key><!-- tokenPool緩存名 -->
                <value>2592000</value><!-- 有效時間 -->
            </entry>
        </map>
    </property>
</bean>

 

四、提供保存、獲取會話信息的API。

  通過以上前戲我們已經把token處理的差不多了,接下來我們要實現token管理工作了

  我們需要讓業務開發方便的保存獲取會話信息,還要使token是透明的。

 

import java.util.HashMap;
import java.util.Map;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.cache.Cache;
import org.springframework.cache.Cache.ValueWrapper;
import org.springframework.cache.CacheManager;

/**
 * 
 *  類           名:    TokenMapPoolBean
 *  描           述:    token以及相關信息調用處理類
 *  修 改 記 錄:    
 *  @version    V1.0
 *  @date    2016年4月22日
 *  @author    NiuXZ
 *
 */
public class TokenMapPoolBean {
    
    
    private static final Log log = LogFactory.getLog(TokenMapPoolBean.class);
    
    /** 當前請求對應的token*/
    private ThreadLocal<String> currentToken;
    
    private CacheManager cacheManager;
    
    private String cacheName;
    
    private TokenGenerator tokenGenerator;
    
    public TokenMapPoolBean(CacheManager cacheManager, String cacheName, TokenGenerator tokenGenerator) {
        this.cacheManager = cacheManager;
        this.cacheName = cacheName;
        this.tokenGenerator = tokenGenerator;
        currentToken = new ThreadLocal<String>();
    }
    
    /**
     * 如果token合法就返回token,不合法就創建一個新的token並返回,
     * 將token放入ThreadLocal中 並初始化一個tokenMap
     * @param token
     * @return token
     */
    public String verifyToken(String token) {
        //        log.info("校驗Token:\""+token+"\"");
        String verifyedToken = null;
        if (tokenGenerator.checkTokenFormat(token)) {
            //            log.info("校驗Token成功:\""+token+"\"");
            verifyedToken = token;
        }
        else {
            verifyedToken = newToken();
        }
        currentToken.set(verifyedToken);
        Cache cache = cacheManager.getCache(cacheName);
        if (cache == null) {
            throw new RuntimeException("獲取不到存放token的緩存池,chacheName:" + cacheName);
        }
        ValueWrapper value = cache.get(verifyedToken);
        //token對應的值為空,就創建一個新的tokenMap放入緩存中
        if (value == null || value.get() == null) {
            verifyedToken = newToken();
            currentToken.set(verifyedToken);
            Map<String, Object> tokenMap = new HashMap<String, Object>();
            cache.put(verifyedToken, tokenMap);
        }
        return verifyedToken;
    }
    
    /**
     * 生成新的token
     * @return token
     */
    private String newToken() {
        Cache cache = cacheManager.getCache(cacheName);
        if (cache == null) {
            throw new RuntimeException("獲取不到存放token的緩存池,chacheName:" + cacheName);
        }
        String newToken = null;
        int count = 0;
        do {
            count++;
            newToken = tokenGenerator.generatorToken();
        }
        while (cache.get(newToken) != null);
        //        log.info("創建Token成功:\""+newToken+"\" 嘗試生成:"+count+"次");
        return newToken;
    }
    
    /**
     * 獲取當前請求的tokenMap中對應key的對象
     * @param key
     * @return 當前請求的tokenMap中對應key的屬性,模擬session
     */
    public Object getAttribute(String key) {
        Cache cache = cacheManager.getCache(cacheName);
        if (cache == null) {
            throw new RuntimeException("獲取不到存放token的緩存池,chacheName:" + cacheName);
        }
        ValueWrapper tokenMapWrapper = cache.get(currentToken.get());
        Map<String, Object> tokenMap = null;
        if (tokenMapWrapper != null) {
            tokenMap = (Map<String, Object>) tokenMapWrapper.get();
        }
        if (tokenMap == null) {
            verifyToken(currentToken.get());
            tokenMapWrapper = cache.get(currentToken.get());
            tokenMap = (Map<String, Object>) tokenMapWrapper.get();
        }
        return tokenMap.get(key);
    }
    
    /**
     * 設置到當前請求的tokenMap中,模擬session<br>
     * TODO:此種方式設置attribute有問題:<br>
     * 1、可能在同一token併發的情況下執行cache.put(currentToken.get(),tokenMap);時,<br>
     *     tokenMap可能不是最新,會導致丟失數據。<br>
     * 2、每次都put整個tokenMap,數據量太大,需要優化<br>
     * @param key value
     */
    public void setAttribute(String key, Object value) {
        Cache cache = cacheManager.getCache(cacheName);
        if (cache == null) {
            throw new RuntimeException("獲取不到存放token的緩存池,chacheName:" + cacheName);
        }
        ValueWrapper tokenMapWrapper = cache.get(currentToken.get());
        Map<String, Object> tokenMap = null;
        if (tokenMapWrapper != null) {
            tokenMap = (Map<String, Object>) tokenMapWrapper.get();
        }
        if (tokenMap == null) {
            verifyToken(currentToken.get());
            tokenMapWrapper = cache.get(currentToken.get());
            tokenMap = (Map<String, Object>) tokenMapWrapper.get();
        }
        log.info("TokenMap.put(key=" + key + ",value=" + value + ")");
        tokenMap.put(key, value);
        cache.put(currentToken.get(), tokenMap);
    }
    
    /** 
     * 獲取當前線程綁定的用戶token
     * @return token
     */
    public String getToken() {
        if (currentToken.get() == null) {
            //初始化一次token
            verifyToken(null);
        }
        return currentToken.get();
    }
    
    /**
     * 刪除token以及tokenMap
     * @param token
     */
    public void removeTokenMap(String token) {
        if (token == null) {
            return;
        }
        Cache cache = cacheManager.getCache(cacheName);
        if (cache == null) {
            throw new RuntimeException("獲取不到存放token的緩存池,chacheName:" + cacheName);
        }
        log.info("刪除Token:token=" + token);
        cache.evict(token);
    }
    
    public CacheManager getCacheManager() {
        return cacheManager;
    }
    
    public void setCacheManager(CacheManager cacheManager) {
        this.cacheManager = cacheManager;
    }
    
    public String getCacheName() {
        return cacheName;
    }
    
    public void setCacheName(String cacheName) {
        this.cacheName = cacheName;
    }
    
    public TokenGenerator getTokenGenerator() {
        return tokenGenerator;
    }
    
    public void setTokenGenerator(TokenGenerator tokenGenerator) {
        this.tokenGenerator = tokenGenerator;
    }
    
    public void clear() {
        currentToken.remove();
    }
    
}

 

  這裡用到了ThreadLocal變數是因為servlet容器一個請求對應一個線程,在一個請求的生命周期內都是處於同一個線程中,而同時又有多個線程共用token管理器,所以需要這個線程本地變數來保存token字元串。

註意事項:1、verifyToken方法的調用,一定要在每次請求最開始調用。並且在請求結束後調用clear做清除,以免下次有未知異常導致verifyToken未被執行,卻在返回時從ThreadLocal里取出token返回。(這個bug困擾我好幾天,公司n個開發檢查代碼也沒找到,最後我經過測試發現是在發生404的時候沒有進入攔截器,所以就沒有調用verifyToken方法,導致返回的異常信息中的token為上一次請求的token,導致詭異的串號問題。嗯,記我一大鍋)。

  2、客戶端一定要在封裝http工具的時候把每次token保存下來,並用於下一次請求。公司ios開發請的外包,但是外包沒按要求做,在未登錄時,不保存token,每次傳遞的都是null,導致每次請求都會創建一個token,伺服器創建了大量的無用token。

使用

  使用方式也很簡單,以下是封裝的登錄管理器,可以參考一下token管理器對於登陸管理器的應用

 

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.cache.Cache;
import org.springframework.cache.Cache.ValueWrapper;
import org.springframework.cache.CacheManager;

import com.niuxz.base.Constants;

/**
 * 
 *  類           名:    LoginManager
 *  描           述:    登錄管理器
 *  修 改 記 錄:    
 *  @version    V1.0
 *  @date    2016年7月19日
 *  @author    NiuXZ
 *
 */
public class LoginManager {
    
    
    private static final Log log = LogFactory.getLog(LoginManager.class);
    
    private CacheManager cacheManager;
    
    private String cacheName;
    
    private TokenMapPoolBean tokenMapPool;
    
    public LoginManager(CacheManager cacheManager, String cacheName, TokenMapPoolBean tokenMapPool) {
        this.cacheManager = cacheManager;
        this.cacheName = cacheName;
        this.tokenMapPool = tokenMapPool;
    }
    public void login(String userId) {
        log.info("用戶登錄:userId=" + userId);
        Cache cache = cacheManager.getCache(cacheName);
        ValueWrapper valueWrapper = cache.get(userId);
        String token = (String) (valueWrapper == null ? null : valueWrapper.get());
        tokenMapPool.removeTokenMap(token);//退出之前登錄記錄
        tokenMapPool.setAttribute(Constants.LOGGED_USER_ID, userId);
        cache.put(userId, tokenMapPool.getToken());
    }
    
    public void logoutCurrent(String phoneTel) {
        String curUserId = getCurrentUserId();
        log.info("用戶退出:userId=" + curUserId);
        tokenMapPool.removeTokenMap(tokenMapPool.getToken());//退出登錄
        if (curUserId != null) {
            Cache cache = cacheManager.getCache(cacheName);
            cache.evict(curUserId);
            cache.evict(phoneTel);
        }
    }
    
    /**
     * 獲取當前用戶的id
     * @return
     */
    public String getCurrentUserId() {
        return (String) tokenMapPool.getAttribute(Constants.LOGGED_USER_ID);
    }
    
    public CacheManager getCacheManager() {
        return cacheManager;
    }
    
    public String getCacheName() {
        return cacheName;
    }
    
    public TokenMapPoolBean getTokenMapPool() {
        return tokenMapPool;
    }
    
    public void setCacheManager(CacheManager cacheManager) {
        this.cacheManager = cacheManager;
    }
    
    public void setCacheName(String cacheName) {
        this.cacheName = cacheName;
    }
    
    public void setTokenMapPool(TokenMapPoolBean tokenMapPool) {
        this.tokenMapPool = tokenMapPool;
    }
    
}

 

  下麵是一段常見的發送簡訊驗證碼介面,有的應用也是用session存儲驗證碼,我不建議用這種方式,存session弊端相當大。大家看看就好,不是我寫的

public void sendValiCodeByPhoneNum(String phoneNum, String hintMsg, String logSuffix) {
        validatePhoneTimeSpace();
        // 獲取6位隨機數
        String code = CodeUtil.getValidateCode();
        log.info(code + "------->" + phoneNum);
        // 調用簡訊驗證碼下發介面
        RetStatus retStatus = msgSendUtils.sendSms(code + hintMsg, phoneNum);
        if (!retStatus.getIsOk()) {
            log.info(retStatus.toString());
            throw new ThrowsToDataException(ServiceResponseCode.FAIL_INVALID_PARAMS, "手機驗證碼獲取失敗,請稍後再試");
        }
        // 重置session
        tokenMapPool.setAttribute(Constants.VALIDATE_PHONE, phoneNum);
        tokenMapPool.setAttribute(Constants.VALIDATE_PHONE_CODE, code.toString());
        tokenMapPool.setAttribute(Constants.SEND_CODE_WRONGNU, 0);
        tokenMapPool.setAttribute(Constants.SEND_CODE_TIME, new Date().getTime());
        log.info(logSuffix + phoneNum + "簡訊驗證碼:" + code);
    }

  

處理響應

  有的同學會問了 那麼響應的報文封裝呢?

 

@RequestMapping("record")
@ResponseBody
public ServiceResponse record(String message){
    String userId = loginManager.getCurrentUserId(); 
    messageBoardService.recordMessage(userId, message);
    return ServiceResponseBuilder.buildSuccess(null);
}

 

  其中ServiceResponse是封裝的響應報文VO,我們直接使用springmvc的@ResponseBody註解就好了。關鍵在於這個builder。

  

import org.apache.commons.lang3.StringUtils;

import com.niuxz.base.pojo.ServiceResponse;
import com.niuxz.utils.spring.SpringContextUtil;
import com.niuxz.web.server.token.TokenMapPoolBean;

/**
 * 
 * 類 名: ServiceResponseBuilder
 * 
 * @version V1.0
 * @date 2016年4月25日
 * @author NiuXZ
 *
 */
public class ServiceResponseBuilder {

    /**
     * 構建一個成功的響應信息
     * 
     * @param body
     * @return 一個操作成功的 ServiceResponse
     */
    public static ServiceResponse buildSuccess(Object body) {
        return new ServiceResponse(
                ((TokenMapPoolBean) SpringContextUtil.getBean("tokenMapPool"))
                        .getToken(),
                "操作成功", body);
    }

    /**
     * 構建一個成功的響應信息
     * 
     * @param body
     * @return 一個操作成功的 ServiceResponse
     */
    public static ServiceResponse buildSuccess(String token, Object body) {
        return new ServiceResponse(token, "操作成功", body);
    }

    /**
     * 構建一個失敗的響應信息
     * 
     * @param failCode
     *            msg
     * @return 一個操作失敗的 ServiceResponse
     */
    public static ServiceResponse buildFail(int failCode, String msg) {
        return buildFail(failCode, msg, null);
    }

    /**
     * 構建一個失敗的響應信息
     * 
     * @param failCode
     *            msg body
     * @return 一個操作失敗的 ServiceResponse
     */
    public static ServiceResponse buildFail(int failCode, String msg,
            Object body) {
        return new ServiceResponse(
                ((TokenMapPoolBean) SpringContextUtil.getBean("tokenMapPool"))
                        .getToken(),
                failCode, StringUtils.isNotBlank(msg) ? msg : "操作失敗", body);
    }
}

  由於使用的是靜態工具類的形式,不能通過spring註入tokenMapPool(token管理器)對象,則通過spring提供的api獲取。然後構建響應信息的時候直接調用tokenMapPool的getToken()方法,此方法會返回當前線程綁定的token字元串。再次強調在請求結束後一定要手動調用clear(我通過全局攔截器調用)。

 


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

-Advertisement-
Play Games
更多相關文章
  • 天龍八部 八步操作資料庫 七賤下天山 七步操作資料庫 (將判斷錯誤省略) 六脈神劍 六步操作資料庫(將判斷錯誤省略,將選擇資料庫添加到第一步) 1.連接資料庫 mysqli_connect(); 參數1:資料庫主機地址 參數2:資料庫用戶名 參數3:資料庫密碼 參數4:[可選參數] 要操作的資料庫名 ...
  • 需要單一入口文件,可以使用autoload來載入 response方法用來返回json數據包 JSON_UNESCAPED_UNICODE:以字面編碼多位元組 Unicode 字元(預設是編碼成 \uXXXX)。自 PHP 5.4.0 起生效。 Auth類用來檢測傳值是否正確,這裡用了兩個檢測參數的方 ...
  • Oracle資料庫應用 一:.Oracle資料庫應用知識 二:表空間和用戶許可權管理 表空間是數據邏輯結構的一個重要組件,表空間可以存放各種應用對象,如表,索引。而每個表空間由一個或者多個數據文件組成 2.表空間的分類可以分成三類: 永久性表空間 一般保存表,視圖,過程和索引等的數據。SYSTEM,S ...
  • 1. Struts2的核心配置(詳解) 本章內容目錄: 配置struts.xml文件 struts.xml文件 常量配置 包配置 包含配置 Action配置 實現Action控制類 配置Action 使用通配符 Action訪問Servlet API 通過ActionContext類訪問 通過特定接 ...
  • 章節:why的使用 用法: why 概念|辭彙(比概念更一般的形式的keyword)|短語|句子 用法1: why 概念|why keyword(【比概念更一般的形式的keyword】) “why 概念”其實可以看作是“why引入 概念(即,等同於“why創造、why提出、why使用、why發明 ” ...
  • 搖搖車這個行業在中國至少已經存在了7,8年以上,這期間也越來越多的投放商加入到這個隊伍裡面,說明這個行業本身是剛性需求,不要小看這一塊錢現金流,如果投放的數量達到一定程度,每天的現金收入是非常可觀的。這麼來算(粗略的算),投放100輛車出去,每輛車每天消費15次也就是說每天賺15塊錢,每天總收入有1 ...
  • 定時任務,是指定一個未來的時間範圍執行一定任務的功能。在當前WEB應用中,多數應用都具備任務調度功能,針對不同的語音,不同的操作系統, 都有其自己的語法及解決方案,windows操作系統把它叫做任務計劃,linux中cron服務都提供了這個功能,在我們開發業務系統中很多時候會涉及到這個功能。本場ch ...
  • 我是一名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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...