最近做了一个Iot项目,上位机是用Java实现的Web服务,下位机是用C#开发的程序,通过socket的方式连接。其中有一段敏感信息需要加密,由于下位机有多个,且数量数据不具体。由于该方案是服务器接到数据后,用RSA加密后返回去,下位机用C#进行解析。前段时间折腾多次都未成功,今天终于成功,记录一下。
java端加密:
private String generate(String inputData ) { String encryptedText = null; try { ClassPathResource classPathResource = new ClassPathResource("lickey/RsaPrivateKey.pem"); String privateKeyString = FileUtils.readFileToString(classPathResource.getFile(),String.valueOf(StandardCharsets.UTF_8)); PrivateKey privateKey = decodePrivateKeyFromXml(privateKeyString); log.info("私钥从xml转为primary为[{}]",Base64.encode(privateKey.getEncoded())); byte outData[] = RsaHelper.encryptData(inputData.getBytes(),privateKey); if(outData!=null){ encryptedText = Base64.encode(outData); } } catch (Exception e) { e.printStackTrace(); } return encryptedText; };
// 用私钥加密 public static byte[] encryptData(byte[] data, PrivateKey priKey) { try { Cipher cipher = Cipher.getInstance("RSA"); cipher.init(Cipher.ENCRYPT_MODE, priKey); return cipher.doFinal(data); } catch (Exception e) { return null; } }
C#端
/// <summary>
/// 用公钥解密
/// </summary>
/// <param name="data"></param>
/// <param name="xmlPublicKey"></param>
/// <returns></returns>
static string decryptByPublicKey(String data, String xmlPublicKey)
{
RSACryptoServiceProvider publicRsa = new RSACryptoServiceProvider();
publicRsa.FromXmlString(xmlPublicKey);
//转换密钥
AsymmetricKeyParameter keyPair = DotNetUtilities.GetRsaPublicKey(publicRsa);
IBufferedCipher c = CipherUtilities.GetCipher("RSA");// 参数与Java中加密解密的参数一致
c.Init(false, keyPair); //第一个参数为true表示加密,为false表示解密;第二个参数表示密钥
byte[] dataToEncrypt = Convert.FromBase64String(data);
//解密
byte[] outBytes = c.DoFinal(dataToEncrypt);
string outStrin = System.Text.Encoding.Default.GetString(outBytes);
Console.WriteLine(outStrin);
return outStrin;
}