<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          四連問:API 接口應(yīng)該如何設(shè)計?如何保證安全?如何簽名?如何防重?

          共 20120字,需瀏覽 41分鐘

           ·

          2022-02-18 20:51

          關(guān)注我們,設(shè)為星標(biāo),每天7:30不見不散,架構(gòu)路上與您共享

          回復(fù)架構(gòu)師獲取資源


          大家好,我是架構(gòu)君,一個會寫代碼吟詩的架構(gòu)師。

          'javajgs.com';

          在實際的業(yè)務(wù)中,難免會跟第三方系統(tǒng)進行數(shù)據(jù)的交互與傳遞,那么如何保證數(shù)據(jù)在傳輸過程中的安全呢(防竊?。??除了https的協(xié)議之外,能不能加上通用的一套算法以及規(guī)范來保證傳輸?shù)陌踩阅兀?br style="max-width: 100%;box-sizing: border-box !important;overflow-wrap: break-word !important;">

          下面我們就來討論下常用的一些API設(shè)計的安全方法,可能不一定是最好的,有更牛逼的實現(xiàn)方式,但是這篇是我自己的經(jīng)驗分享.

          一、token 簡介


          Token:訪問令牌access token, 用于接口中, 用于標(biāo)識接口調(diào)用者的身份、憑證,減少用戶名和密碼的傳輸次數(shù)。一般情況下客戶端(接口調(diào)用方)需要先向服務(wù)器端申請一個接口調(diào)用的賬號,服務(wù)器會給出一個appId和一個key, key用于參數(shù)簽名使用,注意key保存到客戶端,需要做一些安全處理,防止泄露。

          Token的值一般是UUID,服務(wù)端生成Token后需要將token做為key,將一些和token關(guān)聯(lián)的信息作為value保存到緩存服務(wù)器中(redis),當(dāng)一個請求過來后,服務(wù)器就去緩存服務(wù)器中查詢這個Token是否存在,存在則調(diào)用接口,不存在返回接口錯誤,一般通過攔截器或者過濾器來實現(xiàn),Token分為兩種:

          • API Token(接口令牌): 用于訪問不需要用戶登錄的接口,如登錄、注冊、一些基本數(shù)據(jù)的獲取等。獲取接口令牌需要拿appId、timestamp和sign來換,sign=加密(timestamp+key)
          • USER Token(用戶令牌): 用于訪問需要用戶登錄之后的接口,如:獲取我的基本信息、保存、修改、刪除等操作。獲取用戶令牌需要拿用戶名和密碼來換

          關(guān)于Token的時效性:token可以是一次性的、也可以在一段時間范圍內(nèi)是有效的,具體使用哪種看業(yè)務(wù)需要。

          一般情況下接口最好使用https協(xié)議,如果使用http協(xié)議,Token機制只是一種減少被黑的可能性,其實只能防君子不能防小人。

          一般token、timestamp和sign 三個參數(shù)會在接口中會同時作為參數(shù)傳遞,每個參數(shù)都有各自的用途。
          ?

          二、timestamp 簡介


          timestamp: 時間戳,是客戶端調(diào)用接口時對應(yīng)的當(dāng)前時間戳,時間戳用于防止DoS攻擊。

          當(dāng)黑客劫持了請求的url去DoS攻擊,每次調(diào)用接口時接口都會判斷服務(wù)器當(dāng)前系統(tǒng)時間和接口中傳的的timestamp的差值,如果這個差值超過某個設(shè)置的時間(假如5分鐘),那么這個請求將被攔截掉,如果在設(shè)置的超時時間范圍內(nèi),是不能阻止DoS攻擊的。timestamp機制只能減輕DoS攻擊的時間,縮短攻擊時間。如果黑客修改了時間戳的值可通過sign簽名機制來處理。

          DoS
          DoS是Denial of Service的簡稱,即拒絕服務(wù),造成DoS的攻擊行為被稱為DoS攻擊,其目的是使計算機或網(wǎng)絡(luò)無法提供正常的服務(wù)。最常見的DoS攻擊有計算機網(wǎng)絡(luò)帶寬攻擊和連通性攻擊。

          DoS攻擊是指故意的攻擊網(wǎng)絡(luò)協(xié)議實現(xiàn)的缺陷或直接通過野蠻手段殘忍地耗盡被攻擊對象的資源,目的是讓目標(biāo)計算機或網(wǎng)絡(luò)無法提供正常的服務(wù)或資源訪問,使目標(biāo)系統(tǒng)服務(wù)系統(tǒng)停止響應(yīng)甚至崩潰,而在此攻擊中并不包括侵入目標(biāo)服務(wù)器或目標(biāo)網(wǎng)絡(luò)設(shè)備。這些服務(wù)資源包括網(wǎng)絡(luò)帶寬,文件系統(tǒng)空間容量,開放的進程或者允許的連接。這種攻擊會導(dǎo)致資源的匱乏,無論計算機的處理速度多快、內(nèi)存容量多大、網(wǎng)絡(luò)帶寬的速度多快都無法避免這種攻擊帶來的后果。

          • Pingflood: 該攻擊在短時間內(nèi)向目的主機發(fā)送大量ping包,造成網(wǎng)絡(luò)堵塞或主機資源耗盡。

          • Synflood: 該攻擊以多個隨機的源主機地址向目的主機發(fā)送SYN包,而在收到目的主機的SYN ACK后并不回應(yīng),這樣,目的主機就為這些源主機建立了大量的連接隊列,而且由于沒有收到ACK一直維護著這些隊列,造成了資源的大量消耗而不能向正常請求提供服務(wù)。

          • Smurf:該攻擊向一個子網(wǎng)的廣播地址發(fā)一個帶有特定請求(如ICMP回應(yīng)請求)的包,并且將源地址偽裝成想要攻擊的主機地址。子網(wǎng)上所有主機都回應(yīng)廣播包請求而向被攻擊主機發(fā)包,使該主機受到攻擊。

          • Land-based:攻擊者將一個包的源地址和目的地址都設(shè)置為目標(biāo)主機的地址,然后將該包通過IP欺騙的方式發(fā)送給被攻擊主機,這種包可以造成被攻擊主機因試圖與自己建立連接而陷入死循環(huán),從而很大程度地降低了系統(tǒng)性能。

          • Ping of Death:根據(jù)TCP/IP的規(guī)范,一個包的長度最大為65536字節(jié)。盡管一個包的長度不能超過65536字節(jié),但是一個包分成的多個片段的疊加卻能做到。當(dāng)一個主機收到了長度大于65536字節(jié)的包時,就是受到了Ping of Death攻擊,該攻擊會造成主機的宕機。

          • Teardrop:IP數(shù)據(jù)包在網(wǎng)絡(luò)傳遞時,數(shù)據(jù)包可以分成更小的片段。攻擊者可以通過發(fā)送兩段(或者更多)數(shù)據(jù)包來實現(xiàn)TearDrop攻擊。第一個包的偏移量為0,長度為N,第二個包的偏移量小于N。為了合并這些數(shù)據(jù)段,TCP/IP堆棧會分配超乎尋常的巨大資源,從而造成系統(tǒng)資源的缺乏甚至機器的重新啟動。

          • PingSweep:使用ICMP Echo輪詢多個主機。


          三、sign 簡介


          nonce:隨機值,是客戶端隨機生成的值,作為參數(shù)傳遞過來,隨機值的目的是增加sign簽名的多變性。隨機值一般是數(shù)字和字母的組合,6位長度,隨機值的組成和長度沒有固定規(guī)則。關(guān)注Java架構(gòu)師社區(qū)不掉隊哦。


          sign: 一般用于參數(shù)簽名,防止參數(shù)被非法篡改,最常見的是修改金額等重要敏感參數(shù), sign的值一般是將所有非空參數(shù)按照升續(xù)排序然后+token+key+timestamp+nonce(隨機數(shù))拼接在一起,然后使用某種加密算法進行加密,作為接口中的一個參數(shù)sign來傳遞,也可以將sign放到請求頭中。

          接口在網(wǎng)絡(luò)傳輸過程中如果被黑客挾持,并修改其中的參數(shù)值,然后再繼續(xù)調(diào)用接口,雖然參數(shù)的值被修改了,但是因為黑客不知道sign是如何計算出來的,不知道sign都有哪些值構(gòu)成,不知道以怎樣的順序拼接在一起的,最重要的是不知道簽名字符串中的key是什么,所以黑客可以篡改參數(shù)的值,但沒法修改sign的值,當(dāng)服務(wù)器調(diào)用接口前會按照sign的規(guī)則重新計算出sign的值然后和接口傳遞的sign參數(shù)的值做比較,如果相等表示參數(shù)值沒有被篡改,如果不等,表示參數(shù)被非法篡改了,就不執(zhí)行接口了。


          四、防止重復(fù)提交


          對于一些重要的操作需要防止客戶端重復(fù)提交的(如非冪等性重要操作),具體辦法是當(dāng)請求第一次提交時將sign作為key保存到redis,并設(shè)置超時時間,超時時間和Timestamp中設(shè)置的差值相同。


          當(dāng)同一個請求第二次訪問時會先檢測redis是否存在該sign,如果存在則證明重復(fù)提交了,接口就不再繼續(xù)調(diào)用了。如果sign在緩存服務(wù)器中因過期時間到了,而被刪除了,此時當(dāng)這個url再次請求服務(wù)器時,因token的過期時間和sign的過期時間一直,sign過期也意味著token過期,那樣同樣的url再訪問服務(wù)器會因token錯誤會被攔截掉,這就是為什么sign和token的過期時間要保持一致的原因。拒絕重復(fù)調(diào)用機制確保URL被別人截獲了也無法使用(如抓取數(shù)據(jù))。

          對于哪些接口需要防止重復(fù)提交可以自定義個注解來標(biāo)記。

          注意:所有的安全措施都用上的話有時候難免太過復(fù)雜,在實際項目中需要根據(jù)自身情況作出裁剪,比如可以只使用簽名機制就可以保證信息不會被篡改,或者定向提供服務(wù)的時候只用Token機制就可以了。如何裁剪,全看項目實際情況和對接口安全性的要求。


          五、使用流程


          1.接口調(diào)用方(客戶端)向接口提供方(服務(wù)器)申請接口調(diào)用賬號,申請成功后,接口提供方會給接口調(diào)用方一個appId和一個key參數(shù)


          2.客戶端攜帶參數(shù)appId、timestamp、sign去調(diào)用服務(wù)器端的API token,其中sign=加密(appId + timestamp + key)

          3.客戶端拿著api_token 去訪問不需要登錄就能訪問的接口

          4.當(dāng)訪問用戶需要登錄的接口時,客戶端跳轉(zhuǎn)到登錄頁面,通過用戶名和密碼調(diào)用登錄接口,登錄接口會返回一個usertoken, 客戶端拿著usertoken 去訪問需要登錄才能訪問的接口

          sign的作用是防止參數(shù)被篡改,客戶端調(diào)用服務(wù)端時需要傳遞sign參數(shù),服務(wù)器響應(yīng)客戶端時也可以返回一個sign用于客戶度校驗返回的值是否被非法篡改了??蛻舳藗鞯膕ign和服務(wù)器端響應(yīng)的sign算法可能會不同。

          ?


          六、示例代碼


          1. dependency


          <dependency>    <groupId>org.springframework.bootgroupId>    <artifactId>spring-boot-starter-data-redisartifactId>dependency><dependency>    <groupId>redis.clientsgroupId>    <artifactId>jedisartifactId>    <version>2.9.0version>dependency>

          <dependency> <groupId>org.springframework.bootgroupId> <artifactId>spring-boot-starter-webartifactId>dependency>

          ?

          2. RedisConfiguration


          @Configurationpublic class RedisConfiguration {@Beanpublic JedisConnectionFactory jedisConnectionFactory(){return new JedisConnectionFactory();    }

          /** * 支持存儲對象 * @return */@Beanpublic RedisTemplate<String, String> redisTemplate(){ RedisTemplate<String, String> redisTemplate = new StringRedisTemplate(); redisTemplate.setConnectionFactory(jedisConnectionFactory()); Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class); ObjectMapper objectMapper = new ObjectMapper(); objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY); objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);

          jackson2JsonRedisSerializer.setObjectMapper(objectMapper); redisTemplate.setValueSerializer(jackson2JsonRedisSerializer); redisTemplate.afterPropertiesSet();

          return redisTemplate; }}

          ?

          3. TokenController


          @Slf4j@RestController@RequestMapping("/api/token")public class TokenController {

          @Autowired private RedisTemplate redisTemplate;

          /** * API Token * * @param sign * @return */ @PostMapping("/api_token") public ApiResponse apiToken(String appId, @RequestHeader("timestamp") String timestamp, @RequestHeader("sign") String sign) { Assert.isTrue(!StringUtils.isEmpty(appId) && !StringUtils.isEmpty(timestamp) && !StringUtils.isEmpty(sign), "參數(shù)錯誤");

          long reqeustInterval = System.currentTimeMillis() - Long.valueOf(timestamp); Assert.isTrue(reqeustInterval < 5 * 60 * 1000, "請求過期,請重新請求");

          // 1. 根據(jù)appId查詢數(shù)據(jù)庫獲取appSecret AppInfo appInfo = new AppInfo("1", "12345678954556");

          // 2. 校驗簽名 String signString = timestamp + appId + appInfo.getKey(); String signature = MD5Util.encode(signString); log.info(signature); Assert.isTrue(signature.equals(sign), "簽名錯誤");

          // 3. 如果正確生成一個token保存到redis中,如果錯誤返回錯誤信息 AccessToken accessToken = this.saveToken(0, appInfo, null);

          return ApiResponse.success(accessToken); }



          @NotRepeatSubmit(5000) @PostMapping("user_token") public ApiResponse userToken(String username, String password) { // 根據(jù)用戶名查詢密碼, 并比較密碼(密碼可以RSA加密一下) UserInfo userInfo = new UserInfo(username, "81255cb0dca1a5f304328a70ac85dcbd", "111111"); String pwd = password + userInfo.getSalt(); String passwordMD5 = MD5Util.encode(pwd); Assert.isTrue(passwordMD5.equals(userInfo.getPassword()), "密碼錯誤");

          // 2. 保存Token AppInfo appInfo = new AppInfo("1", "12345678954556"); AccessToken accessToken = this.saveToken(1, appInfo, userInfo); userInfo.setAccessToken(accessToken); return ApiResponse.success(userInfo); }

          private AccessToken saveToken(int tokenType, AppInfo appInfo, UserInfo userInfo) { String token = UUID.randomUUID().toString();

          // token有效期為2小時 Calendar calendar = Calendar.getInstance(); calendar.setTime(new Date()); calendar.add(Calendar.SECOND, 7200); Date expireTime = calendar.getTime();

          // 4. 保存token ValueOperations operations = redisTemplate.opsForValue(); TokenInfo tokenInfo = new TokenInfo(); tokenInfo.setTokenType(tokenType); tokenInfo.setAppInfo(appInfo);

          if (tokenType == 1) { tokenInfo.setUserInfo(userInfo); }

          operations.set(token, tokenInfo, 7200, TimeUnit.SECONDS);

          AccessToken accessToken = new AccessToken(token, expireTime);

          return accessToken; }

          public static void main(String[] args) { long timestamp = System.currentTimeMillis(); System.out.println(timestamp); String signString = timestamp + "1" + "12345678954556"; String sign = MD5Util.encode(signString); System.out.println(sign);

          System.out.println("-------------------"); signString = "password=123456&username=1&12345678954556" + "ff03e64b-427b-45a7-b78b-47d9e8597d3b1529815393153sdfsdfsfs" + timestamp + "A1scr6"; sign = MD5Util.encode(signString); System.out.println(sign); }}

          ?

          4. WebMvcConfiguration


          @Configurationpublic class WebMvcConfiguration extends WebMvcConfigurationSupport {

          private static final String[] excludePathPatterns = {"/api/token/api_token"};

          @Autowired private TokenInterceptor tokenInterceptor;

          @Override public void addInterceptors(InterceptorRegistry registry) { super.addInterceptors(registry); registry.addInterceptor(tokenInterceptor) .addPathPatterns("/api/**") .excludePathPatterns(excludePathPatterns); }}5. TokenInterceptor@Componentpublic class TokenInterceptor extends HandlerInterceptorAdapter {

          @Autowired private RedisTemplate redisTemplate;

          /** * * @param request * @param response * @param handler 訪問的目標(biāo)方法 * @return * @throws Exception */ @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { String token = request.getHeader("token"); String timestamp = request.getHeader("timestamp"); // 隨機字符串 String nonce = request.getHeader("nonce"); String sign = request.getHeader("sign"); Assert.isTrue(!StringUtils.isEmpty(token) && !StringUtils.isEmpty(timestamp) && !StringUtils.isEmpty(sign), "參數(shù)錯誤");

          // 獲取超時時間 NotRepeatSubmit notRepeatSubmit = ApiUtil.getNotRepeatSubmit(handler); long expireTime = notRepeatSubmit == null ? 5 * 60 * 1000 : notRepeatSubmit.value();

          // 2. 請求時間間隔 long reqeustInterval = System.currentTimeMillis() - Long.valueOf(timestamp); Assert.isTrue(reqeustInterval < expireTime, "請求超時,請重新請求");

          // 3. 校驗Token是否存在 ValueOperations tokenRedis = redisTemplate.opsForValue(); TokenInfo tokenInfo = tokenRedis.get(token); Assert.notNull(tokenInfo, "token錯誤");

          // 4. 校驗簽名(將所有的參數(shù)加進來,防止別人篡改參數(shù)) 所有參數(shù)看參數(shù)名升續(xù)排序拼接成url // 請求參數(shù) + token + timestamp + nonce String signString = ApiUtil.concatSignString(request) + tokenInfo.getAppInfo().getKey() + token + timestamp + nonce; String signature = MD5Util.encode(signString); boolean flag = signature.equals(sign); Assert.isTrue(flag, "簽名錯誤");

          // 5. 拒絕重復(fù)調(diào)用(第一次訪問時存儲,過期時間和請求超時時間保持一致), 只有標(biāo)注不允許重復(fù)提交注解的才會校驗 if (notRepeatSubmit != null) { ValueOperations signRedis = redisTemplate.opsForValue(); boolean exists = redisTemplate.hasKey(sign); Assert.isTrue(!exists, "請勿重復(fù)提交"); signRedis.set(sign, 0, expireTime, TimeUnit.MILLISECONDS); }

          return super.preHandle(request, response, handler); }}

          ?

          5. MD5Util ----MD5工具類,加密生成數(shù)字簽名


          public class MD5Util {

          private static final String hexDigits[] = { "0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "a", "b", "c", "d", "e", "f" };

          private static String byteArrayToHexString(byte b[]) { StringBuffer resultSb = new StringBuffer(); for (int i = 0; i < b.length; i++) resultSb.append(byteToHexString(b[i]));
          return resultSb.toString(); }

          private static String byteToHexString(byte b) { int n = b; if (n < 0) n += 256; int d1 = n / 16; int d2 = n % 16; return hexDigits[d1] + hexDigits[d2]; }

          public static String encode(String origin) { return encode(origin, "UTF-8"); } public static String encode(String origin, String charsetname) { String resultString = null; try { resultString = new String(origin); MessageDigest md = MessageDigest.getInstance("MD5"); if (charsetname == null || "".equals(charsetname)) resultString = byteArrayToHexString(md.digest(resultString .getBytes())); else resultString = byteArrayToHexString(md.digest(resultString .getBytes(charsetname))); } catch (Exception exception) { } return resultString; }}


          6. @NotRepeatSubmit ??-----自定義注解,防止重復(fù)提交。


          /** * 禁止重復(fù)提交 */@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)public @interface NotRepeatSubmit {    /** 過期時間,單位毫秒 **/    long value() default 5000;}

          ?

          7. AccessToken


          @Data@AllArgsConstructorpublic class AccessToken {    /** token */    private String token;

          /** 失效時間 */ private Date expireTime;}


          8. AppInfo


          @Data@NoArgsConstructor@AllArgsConstructorpublic class AppInfo {    /** App id */    private String appId;    /** API 秘鑰 */    private String key;}


          9. TokenInfo


          @Datapublic class TokenInfo {    /** token類型: api:0 、user:1 */    private Integer tokenType;

          /** App 信息 */ private AppInfo appInfo;

          /** 用戶其他數(shù)據(jù) */ private UserInfo userInfo;}


          10. UserInfo


          @Datapublic class UserInfo {    /** 用戶名 */    private String username;    /** 手機號 */    private String mobile;    /** 郵箱 */    private String email;    /** 密碼 */    private String password;    /** 鹽 */    private String salt;

          private AccessToken accessToken;

          public UserInfo(String username, String password, String salt) { this.username = username; this.password = password; this.salt = salt; }}?

          ?

          11. ApiCodeEnum


          /** * 錯誤碼code可以使用純數(shù)字,使用不同區(qū)間標(biāo)識一類錯誤,也可以使用純字符,也可以使用前綴+編號 * * 錯誤碼:ERR + 編號 * * 可以使用日志級別的前綴作為錯誤類型區(qū)分 Info(I) Error(E) Warning(W) * * 或者以業(yè)務(wù)模塊 + 錯誤號 * * TODO 錯誤碼設(shè)計 * * Alipay 用了兩個code,兩個msg(https://docs.open.alipay.com/api_1/alipay.trade.pay) */public enum ApiCodeEnum {    SUCCESS("10000", "success"),    UNKNOW_ERROR("ERR0001","未知錯誤"),    PARAMETER_ERROR("ERR0002","參數(shù)錯誤"),    TOKEN_EXPIRE("ERR0003","認(rèn)證過期"),    REQUEST_TIMEOUT("ERR0004","請求超時"),    SIGN_ERROR("ERR0005","簽名錯誤"),    REPEAT_SUBMIT("ERR0006","請不要頻繁操作"),    ;

          /** 代碼 */ private String code;

          /** 結(jié)果 */ private String msg;

          ApiCodeEnum(String code, String msg) { this.code = code; this.msg = msg; }

          public String getCode() { return code; }

          public String getMsg() { return msg; }}

          ?

          12. ApiResult


          @Data@NoArgsConstructor@AllArgsConstructorpublic class ApiResult {

          /** 代碼 */private String code;

          /** 結(jié)果 */private String msg;}

          ?

          13. ApiUtil ?-------這個參考支付寶加密的算法寫的.我直接Copy過來了。


          public class ApiUtil {

          /** * 按參數(shù)名升續(xù)拼接參數(shù) * @param request * @return */ public static String concatSignString(HttpServletRequest request) {Map paramterMap = new HashMap<>(); request.getParameterMap().forEach((key, value) -> paramterMap.put(key, value[0]));// 按照key升續(xù)排序,然后拼接參數(shù)Set keySet = paramterMap.keySet();String[] keyArray = keySet.toArray(new String[keySet.size()]); Arrays.sort(keyArray); StringBuilder sb = new StringBuilder();for (String k : keyArray) {// 或略掉的字段if (k.equals("sign")) {continue; }if (paramterMap.get(k).trim().length() > 0) {// 參數(shù)值為空,則不參與簽名 sb.append(k).append("=").append(paramterMap.get(k).trim()).append("&"); } }

          return sb.toString(); }

          public static String concatSignString(Map map) {Map paramterMap = new HashMap<>(); map.forEach((key, value) -> paramterMap.put(key, value));// 按照key升續(xù)排序,然后拼接參數(shù)Set keySet = paramterMap.keySet();String[] keyArray = keySet.toArray(new String[keySet.size()]); Arrays.sort(keyArray); StringBuilder sb = new StringBuilder();for (String k : keyArray) {if (paramterMap.get(k).trim().length() > 0) {// 參數(shù)值為空,則不參與簽名 sb.append(k).append("=").append(paramterMap.get(k).trim()).append("&"); } }return sb.toString(); }

          /** * 獲取方法上的@NotRepeatSubmit注解 * @param handler * @return */ public static NotRepeatSubmit getNotRepeatSubmit(Object handler) {if (handler instanceof HandlerMethod) { HandlerMethod handlerMethod = (HandlerMethod) handler; Method method = handlerMethod.getMethod(); NotRepeatSubmit annotation = method.getAnnotation(NotRepeatSubmit.class);

          return annotation; }

          return null; }}


          14. ApiResponse


          @Data@Slf4jpublic class ApiResponse {/** 結(jié)果 */private ApiResult result;

          /** 數(shù)據(jù) */private T data;

          /** 簽名 */private String sign;



          public static ApiResponse success(T data) {return response(ApiCodeEnum.SUCCESS.getCode(), ApiCodeEnum.SUCCESS.getMsg(), data); }

          public static ApiResponse error(String code, String msg) {return response(code, msg, null); }

          public static ApiResponse response(String code, String msg, T data) { ApiResult result = new ApiResult(code, msg); ApiResponse response = new ApiResponse(); response.setResult(result); response.setData(data);

          String sign = signData(data); response.setSign(sign);

          return response; }

          private static String signData(T data) {// TODO 查詢keyString key = "12345678954556"; Map responseMap = null;try { responseMap = getFields(data); } catch (IllegalAccessException e) {return null; }String urlComponent = ApiUtil.concatSignString(responseMap);String signature = urlComponent + "key=" + key;String sign = MD5Util.encode(signature);

          return sign; }

          /** * @param data 反射的對象,獲取對象的字段名和值 * @throws IllegalArgumentException * @throws IllegalAccessException */public static Map getFields(Object data) throws IllegalAccessException, IllegalArgumentException {if (data == null) return null; Map map = new HashMap<>(); Field[] fields = data.getClass().getDeclaredFields();for (int i = 0; i < fields.length; i++) { Field field = fields[i]; field.setAccessible(true);

          String name = field.getName();Object value = field.get(data);if (field.get(data) != null) { map.put(name, value.toString()); } }

          return map; }}

          ?


          七、ThreadLocal


          ThreadLocal是線程內(nèi)的全局上下文。就是在單個線程中,方法之間共享的內(nèi)存,每個方法都可以從該上下文中獲取值和修改值。


          實際案例:

          在調(diào)用api時都會傳一個token參數(shù),通常會寫一個攔截器來校驗token是否合法,我們可以通過token找到對應(yīng)的用戶信息(User),如果token合法,然后將用戶信息存儲到ThreadLocal中,這樣無論是在controller、service、dao的哪一層都能訪問到該用戶的信息。作用類似于Web中的request作用域。

          傳統(tǒng)方式我們要在方法中訪問某個變量,可以通過傳參的形式往方法中傳參,如果多個方法都要使用那么每個方法都要傳參;如果使用ThreadLocal所有方法就不需要傳該參數(shù)了,每個方法都可以通過ThreadLocal來訪問該值。

          • ThreadLocalUtil.set("key", value); 保存值
          • T value = ThreadLocalUtil.get("key"); 獲取值

          ThreadLocalUtil


          public class ThreadLocalUtil {    private static final ThreadLocal> threadLocal = new ThreadLocal() {        @Override        protected Map initialValue() {return new HashMap<>(4);        }    };



          public static Map getThreadLocal(){return threadLocal.get(); }

          public static T get(String key) {Map map = (Map)threadLocal.get();return (T)map.get(key); }

          public static T get(String key,T defaultValue) {Map map = (Map)threadLocal.get();return (T)map.get(key) == null ? defaultValue : (T)map.get(key); }

          public static void set(String key, Object value) {Map map = (Map)threadLocal.get(); map.put(key, value); }

          public static void set(Map keyValueMap) {Map map = (Map)threadLocal.get(); map.putAll(keyValueMap); }

          public static void remove() { threadLocal.remove(); }

          public static Map fetchVarsByPrefix(String prefix) {Map vars = new HashMap<>();if( prefix == null ){return vars; }Map map = (Map)threadLocal.get();Set set = map.entrySet();

          for( Map.Entry entry : set){Object key = entry.getKey();if( key instanceof String ){if( ((String) key).startsWith(prefix) ){ vars.put((String)key,(T)entry.getValue()); } } }return vars; }

          public static T remove(String key) {Map map = (Map)threadLocal.get();return (T)map.remove(key); }

          public static void clear(String prefix) {if( prefix == null ){return; }Map map = (Map)threadLocal.get();Set set = map.entrySet(); List removeKeys = new ArrayList<>();

          for( Map.Entry entry : set ){Object key = entry.getKey();if( key instanceof String ){if( ((String) key).startsWith(prefix) ){ removeKeys.add((String)key); } } }for( String key : removeKeys ){ map.remove(key); } }}

          ?

          總結(jié):這個是目前第三方數(shù)據(jù)接口交互過程中常用的一些參數(shù)與使用示例,希望對大家有點幫助。

          當(dāng)然如果為了保證更加的安全,可以加上RSA,RSA2,AES等等加密方式,保證了數(shù)據(jù)的更加的安全,但是唯一的缺點是加密與解密比較耗費CPU的資源。

          文章來源:https://www.cnblogs.com/jurendage/p/12653865.html



          到此文章就結(jié)束了。如果今天的文章對你在進階架構(gòu)師的路上有新的啟發(fā)和進步,歡迎轉(zhuǎn)發(fā)給更多人。歡迎加入架構(gòu)師社區(qū)技術(shù)交流群,眾多大咖帶你進階架構(gòu)師,在后臺回復(fù)“加群”即可入群。














          這些年小編給你分享過的干貨

          Kubernetes的前世今生

          你們公司的架構(gòu)師是什么樣的?

          《Docker與CI持續(xù)集成/CD持續(xù)部署》

          《還有40天,Java 11就要橫空出世了》

          《JDK 10 的 109 項新特性》

          《學(xué)習(xí)微服務(wù)的十大理由》

          《進大廠必須掌握的50個微服務(wù)面試問題》


          轉(zhuǎn)發(fā)在看就是最大的支持??

          瀏覽 44
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  国产免费观看高清完整版亚洲日本 | 日日干av| 麻豆AV无码精品一区二区色欲 | 天天日天天舔天天爽天天操 | 99热这里只有精 |