【Java安全】關於Java中常用加密/解密方法的實現

来源:http://www.cnblogs.com/tancky/archive/2017/02/17/6409823.html
-Advertisement-
Play Games

安全問題已經成為一個越來越重要的問題,在Java中如何對重要數據進行加密解密是本文的主要內容。 一、常用的加密/解密演算法 1.Base64 嚴格來說Base64並不是一種加密/解密演算法,而是一種編碼方式。Base64不生成密鑰,通過Base64編碼後的密文就可以直接“翻譯”為明文,但是可以通過向明文 ...


安全問題已經成為一個越來越重要的問題,在Java中如何對重要數據進行加密解密是本文的主要內容。

一、常用的加密/解密演算法

1.Base64

  嚴格來說Base64並不是一種加密/解密演算法,而是一種編碼方式。Base64不生成密鑰,通過Base64編碼後的密文就可以直接“翻譯”為明文,但是可以通過向明文中添加混淆字元來達到加密的效果。

2.DES

  DES是一種基於56位密鑰的對稱演算法,1976年被美國聯邦政府的國家標準局確定為聯邦資料處理標準(FIPS),隨後在國際上廣泛流傳開來。現在DES已經不是一種安全的加密演算法,已被公開破解,現在DES已經被高級加密標準(AES)所代替。

3.3DES

  3DES是DES的一種派生演算法,主要提升了DES的一些實用所需的安全性。

4.AES

  AES是現在對稱加密演算法中最流行的演算法之一。

 

二、實現所需的一些庫

  為了實現上述的演算法,我們可以實用JDK自帶的實現,也可以使用一些開源的第三方庫,例如Bouncy Castle(https://www.bouncycastle.org/)和comnons codec(https://commons.apache.org/proper/commons-codec/)。

 

三、具體實現

  1.Base64

 1 package com.tancky.security;
 2 
 3 
 4 import java.io.IOException;
 5 
 6 import sun.misc.BASE64Decoder;
 7 import sun.misc.BASE64Encoder;
 8 
 9 public class Base64Demo {
10     
11     
12     private static String src = "TestBase64";
13     
14 
15     public static void main(String[] args) {
16         Base64Demo.jdkBase64();
17         Base64Demo.commonsCodecBase64 ();
18         Base64Demo.bouncyCastleBase64 ();
19     }
20     
21     //使用JDK的base64實現,
22     public static void jdkBase64 (){
23         BASE64Encoder encoder = new BASE64Encoder();
24         String encode = encoder.encode(Base64Demo.src.getBytes());
25         System.out.println("encode:  " + encode);
26         
27         BASE64Decoder decoder = new BASE64Decoder();
28         try {
29             String decode = new String ( decoder.decodeBuffer(encode));
30             System.out.println("decode:  " + decode);
31         } catch (IOException e) {
32             e.printStackTrace();
33         }    
34     }
35     
36     
37     //使用apache的commonsCodec實現
38     public static void commonsCodecBase64 (){
39         byte[] encodeBytes = org.apache.commons.codec.binary.Base64.encodeBase64(Base64Demo.src.getBytes());
40         String encode = new String (encodeBytes);
41         System.out.println("encode:  " + encode);
42         
43         byte[] decodeBytes = org.apache.commons.codec.binary.Base64.decodeBase64(encode);
44         String decode = new String(decodeBytes);
45         System.out.println("decode:  " + decode);
46         
47     }
48     
49     //使用bouncyCastlede實現
50     public static void bouncyCastleBase64 () {
51         byte[] encodeBytes = org.bouncycastle.util.encoders.Base64.encode(Base64Demo.src.getBytes()) ;
52         String encode = new String (encodeBytes);
53         System.out.println("encode:  " + encode);
54 
55         byte[] decodeBytes = org.bouncycastle.util.encoders.Base64.decode(encode);
56         String decode = new String(decodeBytes);
57         System.out.println("decode:  " + decode);
58         
59     }
60 
61 }

 

  2.DES

  

  1 package com.tancky.security;
  2 
  3 import java.security.InvalidKeyException;
  4 import java.security.Key;
  5 import java.security.NoSuchAlgorithmException;
  6 import java.security.NoSuchProviderException;
  7 import java.security.Security;
  8 import java.security.spec.InvalidKeySpecException;
  9 
 10 import javax.crypto.BadPaddingException;
 11 import javax.crypto.Cipher;
 12 import javax.crypto.IllegalBlockSizeException;
 13 import javax.crypto.KeyGenerator;
 14 import javax.crypto.NoSuchPaddingException;
 15 import javax.crypto.SecretKey;
 16 import javax.crypto.SecretKeyFactory;
 17 import javax.crypto.spec.DESKeySpec;
 18 
 19 import org.bouncycastle.jce.provider.BouncyCastleProvider;
 20 import org.bouncycastle.util.encoders.Hex;
 21 
 22 public class DESDemo {
 23     
 24     private static String src = "TestDES";
 25     
 26     
 27     public static void jdkDES () {
 28         
 29         try {
 30             //生成密鑰Key
 31             KeyGenerator keyGenerator = KeyGenerator.getInstance("DES");
 32             keyGenerator.init(56);
 33             SecretKey secretKey = keyGenerator.generateKey();
 34             byte[] bytesKey = secretKey.getEncoded();
 35         
 36             
 37             //KEY轉換
 38             DESKeySpec deSedeKeySpec = new DESKeySpec(bytesKey);
 39             SecretKeyFactory factory = SecretKeyFactory.getInstance("DES");
 40             Key convertSecretKey = factory.generateSecret(deSedeKeySpec);
 41             
 42             //加密
 43             Cipher cipher = Cipher.getInstance("DES/ECB/PKCS5Padding");
 44             cipher.init(Cipher.ENCRYPT_MODE, convertSecretKey);
 45             byte[] encodeResult = cipher.doFinal(DESDemo.src.getBytes());
 46             System.out.println("DESEncode :" + Hex.toHexString(encodeResult));
 47             
 48             
 49             //解密
 50             cipher.init(Cipher.DECRYPT_MODE,convertSecretKey);
 51             byte[] DecodeResult = cipher.doFinal(encodeResult);
 52             System.out.println("DESDncode :" + new String (DecodeResult));
 53             
 54             
 55             
 56         } catch (NoSuchAlgorithmException e) {
 57             e.printStackTrace();
 58         } catch (InvalidKeyException e) {
 59             // TODO 自動生成的 catch 塊
 60             e.printStackTrace();
 61         } catch (InvalidKeySpecException e) {
 62             // TODO 自動生成的 catch 塊
 63             e.printStackTrace();
 64         } catch (NoSuchPaddingException e) {
 65             // TODO 自動生成的 catch 塊
 66             e.printStackTrace();
 67         } catch (IllegalBlockSizeException e) {
 68             // TODO 自動生成的 catch 塊
 69             e.printStackTrace();
 70         } catch (BadPaddingException e) {
 71             // TODO 自動生成的 catch 塊
 72             e.printStackTrace();
 73         }
 74     
 75     }
 76     
 77     
 78     
 79     public static void bcDES (){
 80         try {
 81             
 82             
 83             //使用BouncyCastle 的DES加密
 84             Security.addProvider(new BouncyCastleProvider());
 85             
 86             
 87             //生成密鑰Key
 88             KeyGenerator keyGenerator = KeyGenerator.getInstance("DES","BC");
 89             keyGenerator.init(56);
 90             SecretKey secretKey = keyGenerator.generateKey();
 91             byte[] bytesKey = secretKey.getEncoded();
 92         
 93             
 94             //KEY轉換
 95             DESKeySpec deSedeKeySpec = new DESKeySpec(bytesKey);
 96             SecretKeyFactory factory = SecretKeyFactory.getInstance("DES");
 97             Key convertSecretKey = factory.generateSecret(deSedeKeySpec);
 98             
 99             //加密
100             Cipher cipher = Cipher.getInstance("DES/ECB/PKCS5Padding");
101             cipher.init(Cipher.ENCRYPT_MODE, convertSecretKey);
102             byte[] encodeResult = cipher.doFinal(DESDemo.src.getBytes());
103             System.out.println("DESEncode :" + Hex.toHexString(encodeResult));
104             
105             
106             //解密
107             cipher.init(Cipher.DECRYPT_MODE,convertSecretKey);
108             byte[] DecodeResult = cipher.doFinal(encodeResult);
109             System.out.println("DESDncode :" + new String (DecodeResult));
110             
111             
112             
113         } catch (NoSuchAlgorithmException e) {
114             e.printStackTrace();
115         } catch (InvalidKeyException e) {
116             // TODO 自動生成的 catch 塊
117             e.printStackTrace();
118         } catch (InvalidKeySpecException e) {
119             // TODO 自動生成的 catch 塊
120             e.printStackTrace();
121         } catch (NoSuchPaddingException e) {
122             // TODO 自動生成的 catch 塊
123             e.printStackTrace();
124         } catch (IllegalBlockSizeException e) {
125             // TODO 自動生成的 catch 塊
126             e.printStackTrace();
127         } catch (BadPaddingException e) {
128             // TODO 自動生成的 catch 塊
129             e.printStackTrace();
130         } catch (NoSuchProviderException e) {
131             // TODO 自動生成的 catch 塊
132             e.printStackTrace();
133         }
134     }
135     
136     
137     public static void main(String[] args) {
138         DESDemo.jdkDES ();
139         DESDemo.bcDES();
140     }
141 
142 }

  

  3.3DES

  

  1 package com.tancky.security;
  2 
  3 import java.security.InvalidKeyException;
  4 import java.security.Key;
  5 import java.security.NoSuchAlgorithmException;
  6 import java.security.NoSuchProviderException;
  7 import java.security.Security;
  8 import java.security.spec.InvalidKeySpecException;
  9 
 10 import javax.crypto.BadPaddingException;
 11 import javax.crypto.Cipher;
 12 import javax.crypto.IllegalBlockSizeException;
 13 import javax.crypto.KeyGenerator;
 14 import javax.crypto.NoSuchPaddingException;
 15 import javax.crypto.SecretKey;
 16 import javax.crypto.SecretKeyFactory;
 17 
 18 import javax.crypto.spec.DESedeKeySpec;
 19 
 20 import org.bouncycastle.jce.provider.BouncyCastleProvider;
 21 import org.bouncycastle.util.encoders.Hex;
 22 
 23 public class TripleDESDemo {
 24     
 25     private static String src = "TestTripleDES";
 26     
 27     public static void jdkTripleDES () {
 28         
 29         try {
 30             //生成密鑰Key
 31             KeyGenerator keyGenerator = KeyGenerator.getInstance("DESede");
 32             keyGenerator.init(168);
 33             SecretKey secretKey = keyGenerator.generateKey();
 34             byte[] bytesKey = secretKey.getEncoded();
 35         
 36             
 37             //KEY轉換
 38             DESedeKeySpec deSedeKeySpec = new DESedeKeySpec(bytesKey);
 39             SecretKeyFactory factory = SecretKeyFactory.getInstance("DESede");
 40             Key convertSecretKey = factory.generateSecret(deSedeKeySpec);
 41             
 42             //加密
 43             Cipher cipher = Cipher.getInstance("DESede/ECB/PKCS5Padding");
 44             cipher.init(Cipher.ENCRYPT_MODE, convertSecretKey);
 45             byte[] encodeResult = cipher.doFinal(TripleDESDemo.src.getBytes());
 46             System.out.println("TripleDESEncode :" + Hex.toHexString(encodeResult));
 47             
 48             
 49             //解密
 50             cipher.init(Cipher.DECRYPT_MODE,convertSecretKey);
 51             byte[] DecodeResult = cipher.doFinal(encodeResult);
 52             System.out.println("TripleDESDncode :" + new String (DecodeResult));
 53             
 54             
 55             
 56         } catch (NoSuchAlgorithmException e) {
 57             e.printStackTrace();
 58         } catch (InvalidKeyException e) {
 59             // TODO 自動生成的 catch 塊
 60             e.printStackTrace();
 61         } catch (InvalidKeySpecException e) {
 62             // TODO 自動生成的 catch 塊
 63             e.printStackTrace();
 64         } catch (NoSuchPaddingException e) {
 65             // TODO 自動生成的 catch 塊
 66             e.printStackTrace();
 67         } catch (IllegalBlockSizeException e) {
 68             // TODO 自動生成的 catch 塊
 69             e.printStackTrace();
 70         } catch (BadPaddingException e) {
 71             // TODO 自動生成的 catch 塊
 72             e.printStackTrace();
 73         }
 74     
 75     }
 76 
 77     
 78     
 79     
 80 public static void bcTripleDES () {
 81         
 82         try {
 83             
 84             Security.addProvider(new BouncyCastleProvider());
 85             //生成密鑰Key
 86             KeyGenerator keyGenerator = KeyGenerator.getInstance("DESede","BC");
 87             keyGenerator.getProvider();
 88             keyGenerator.init(168);
 89             SecretKey secretKey = keyGenerator.generateKey();
 90             byte[] bytesKey = secretKey.getEncoded();
 91         
 92             
 93             //KEY轉換
 94             DESedeKeySpec deSedeKeySpec = new DESedeKeySpec(bytesKey);
 95             SecretKeyFactory factory = SecretKeyFactory.getInstance("DESede");
 96             Key convertSecretKey = factory.generateSecret(deSedeKeySpec);
 97             
 98             //加密
 99             Cipher cipher = Cipher.getInstance("DESede/ECB/PKCS5Padding");
100             cipher.init(Cipher.ENCRYPT_MODE, convertSecretKey);
101             byte[] encodeResult = cipher.doFinal(TripleDESDemo.src.getBytes());
102             System.out.println("TripleDESEncode :" + Hex.toHexString(encodeResult));
103             
104             
105             //解密
106             cipher.init(Cipher.DECRYPT_MODE,convertSecretKey);
107             byte[] DecodeResult = cipher.doFinal(encodeResult);
108             System.out.println("TripleDESDncode :" + new String (DecodeResult));
109             
110             
111             
112         } catch (NoSuchAlgorithmException e) {
113             e.printStackTrace();
114         } catch (InvalidKeyException e) {
115             // TODO 自動生成的 catch 塊
116             e.printStackTrace();
117         } catch (InvalidKeySpecException e) {
118             // TODO 自動生成的 catch 塊
119             e.printStackTrace();
120         } catch (NoSuchPaddingException e) {
121             // TODO 自動生成的 catch 塊
122             e.printStackTrace();
123         } catch (IllegalBlockSizeException e) {
124             // TODO 自動生成的 catch 塊
125             e.printStackTrace();
126         } catch (BadPaddingException e) {
127             // TODO 自動生成的 catch 塊
128             e.printStackTrace();
129         } catch (NoSuchProviderException e) {
130             // TODO 自動生成的 catch 塊
131             e.printStackTrace();
132         }
133     
134     }
135     
136     
137     
138     public static void main(String[] args) {
139         jdkTripleDES ();
140         bcTripleDES ();
141 
142     }
143 
144 }

 

   4.AES

   

  1 package com.tancky.security;
  2 
  3 import java.security.InvalidKeyException;
  4 import java.security.Key;
  5 import java.security.NoSuchAlgorithmException;
  6 import java.security.NoSuchProviderException;
  7 import java.security.SecureRandom;
  8 import java.security.Security;
  9 
 10 import javax.crypto.BadPaddingException;
 11 import javax.crypto.Cipher;
 12 import javax.crypto.IllegalBlockSizeException;
 13 import javax.crypto.KeyGenerator;
 14 import javax.crypto.NoSuchPaddingException;
 15 import javax.crypto.SecretKey;
 16 import javax.crypto.spec.SecretKeySpec;
 17 
 18 import org.bouncycastle.jce.provider.BouncyCastleProvider;
 19 import org.bouncycastle.util.encoders.Hex;
 20 
 21 public class AESDemo {
 22     
 23     private static String src = "TestAES";
 24     
 25     public static void jdkAES (){
 26         try {
 27             
 28             
 29             //生成Key
 30             KeyGenerator keyGenerator = KeyGenerator.getInstance("AES");
 31             keyGenerator.init(128);  
 32             //keyGenerator.init(128, new SecureRandom("seedseedseed".getBytes()));  
 33             //使用上面這種初始化方法可以特定種子來生成密鑰,這樣加密後的密文是唯一固定的。
 34             SecretKey secretKey = keyGenerator.generateKey();
 35             byte[] keyBytes = secretKey.getEncoded();
 36             
 37             //Key轉換
 38             Key key = new SecretKeySpec(keyBytes, "AES");
 39             
 40             //加密
 41             Cipher cipher = Cipher.getInstance("AES/ECB/PKCS5Padding");
 42             cipher.init(Cipher.ENCRYPT_MODE, key);
 43             byte[] encodeResult = cipher.doFinal(AESDemo.src.getBytes());
 44             System.out.println("AESencode : " + Hex.toHexString(encodeResult) );
 45             
 46             //解密
 47             cipher.init(Cipher.DECRYPT_MODE, key);
 48             byte[] decodeResult = cipher.doFinal(encodeResult);
 49             System.out.println("AESdecode : " + new String (decodeResult));
 50             
 51             
 52         
 53         
 54         } catch (NoSuchAlgorithmException e) {
 55             // TODO 自動生成的 catch 塊
 56             e.printStackTrace();
 57         } catch (NoSuchPaddingException e) {
 58             // TODO 自動生成的 catch 塊
 59             e.printStackTrace();
 60         } catch (InvalidKeyException e) {
 61             // TODO 自動生成的 catch 塊
 62             e.printStackTrace();
 63         } catch (IllegalBlockSizeException e) {
 64             // TODO 自動生成的 catch 塊
 65             e.printStackTrace();
 66         } catch (BadPaddingException e) {
 67             // TODO 自動生成的 catch 塊
 68             e.printStackTrace();
 69         }
 70     
 71     }
 72     
 73     
 74     public static void bcAES (){
 75         try {
 76             
 77             //使用BouncyCastle 的DES加密
 78             Security.addProvider(new BouncyCastleProvider());
 79             
 80             //生成Key
 81             KeyGenerator keyGenerator = KeyGenerator.getInstance("AES","BC");
 82             keyGenerator.getProvider();
 83             keyGenerator.init(128);  
 84             SecretKey secretKey = keyGenerator.generateKey();
 85             byte[] keyBytes = secretKey.getEncoded();
 86             
 87             //Key轉換
 88             Key key = new SecretKeySpec(keyBytes, "AES");
 89             
 90             //加密
 91             Cipher cipher = Cipher.getInstance("AES/ECB/PKCS5Padding");
 92             cipher.init(Cipher.ENCRYPT_MODE, key);
 93             byte[] encodeResult = cipher.doFinal(AESDemo.src.getBytes());
 94             System.out.println("AESencode : " + Hex.toHexString(encodeResult) );
 95             
 96             //解密
 97             cipher.init(Cipher.DECRYPT_MODE, key);
 98             byte[] decodeResult = cipher.doFinal(encodeResult);
 99             System.out.println("AESdecode : " + new String (decodeResult));
100             
101             
102         
103         
104         } catch (NoSuchAlgorithmException e) {
105             // TODO 自動生成的 catch 塊
106             e.printStackTrace();
107         } catch (NoSuchPaddingException e) {
108             // TODO 自動生成的 catch 塊
109             e.printStackTrace();
110         } catch (InvalidKeyException e) {
111             // TODO 自動生成的 catch 塊
112             e.printStackTrace();
113         } catch (IllegalBlockSizeException e) {
114             // TODO 自動生成的 catch 塊
115             e.printStackTrace();
116         } catch (BadPaddingException e) {
117             // TODO 自動生成的 catch 塊
118             e.printStackTrace();
119         } catch (NoSuchProviderException e) {
120             // TODO 自動生成的 catch 塊
121             e.printStackTrace();
122         }
123     
124     }
125     
126 
127     public static void main(String[] args) {
128         jdkAES();
129         bcAES();
130 
131     }
132 
133 }

 


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

-Advertisement-
Play Games
更多相關文章
  • 1.什麼是內部類? 定義在類內部的類,稱之為內部類 2.為什麼要使用內部類? 1),增強封裝,把內部類隱藏在外部類中,不允許其他類來訪問內部類 2),內部類能提高代碼的可讀性和可維護性 3.內部類的分類 對於內部類的分類,可以對比於成員變數的分類. 我們可以根據不同的修飾符或者定義的不同位置把成員變 ...
  • 近期做了一個關於百度營銷推廣的一個管理系統(SEM閃投),主要包括閃投欄位到資料庫欄位的關係映射,多庫多表數據的查詢封裝組合到最後生成XML文件,最終SFTP到另一個資源伺服器上,然後百度要能夠訪問這個XML並解析它,你在百度上搜索某個關鍵詞的時候,最上面的幾條帶廣告字樣的,那麼就達到效果了(關鍵是... ...
  • 本文為作者原創,轉載請註明出處,謝謝. 本文適用於mybatis框架初學者,可以通過這個小例子,初識mybatis的簡單易用. 1.創建工程,導入jar包 創建一個java工程或者web工程都可以,然後導入mybatis的jar包和依賴包還有資料庫的jar包,本人使用Oracle10g資料庫 myb ...
  • 在使用“PHPWAMP自動任務”時,不少學生遇到如下問題: “phpwamp綠色集成環境重啟動電腦(伺服器)後,不會自動啟動網站服務” (如果是其他環境或是自己搭建時遇到此問題,也是可以用此法解決) 此文章內容符合: 為什麼網站服務由手動變成自動後還是無法重啟? 為什麼我把服務設置成自動後,開機又變 ...
  • 本文地址 分享提綱: 1.概述 2.安裝 3.編寫第一個測試用例 4.PHPUnit高級 5.參考 分享提綱: 1.概述 2.安裝 3.編寫第一個測試用例 4.PHPUnit高級 5.參考 1.概述 1)【測試框架】 它是一款輕量級的PHP測試框架,是一個xUnit的體繫結構的單元測試框架。複雜的項 ...
  • 本試題共40道選擇題,10道判斷題,考試時間1個半小時 一:選擇題(單項選擇,每題2分): 1. LAMP具體結構不包含下麵哪種(A ) A:Windows系統 B:Apache伺服器 C:MySQL資料庫 D:PHP語言 2. 以下哪個SQL語句是正確的(D) A:insert into user ...
  • 1. LAMP具體結構不包含下麵哪種(A ) A:Windows系統 如果是這個就是WMP B:Apache伺服器 C:MySQL資料庫 D:PHP語言 2. 以下哪個SQL語句是正確的(D ) A:insert into users 少了一個values (‘p001’,’張三’,’男’); B: ...
  • 轉載請標明出處: "http://www.cnblogs.com/why168888/p/6407980.html" 本文出自: "【Edwin博客園】" Python迭代 1. 什麼是迭代 註意: 集合是指包含一組元素的數據結構,我們已經介紹的包括: 1. 有序集合:list,tuple,str和 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...