
- 创建项目根目录和主要子模块 - 添加基本的 Activity 和布局文件 - 实现简单的导航和电话拨打功能 - 添加相机和图像处理相关代码 - 创建网络请求和数据加密工具类 - 设置 AndroidManifest 文件和权限
347 lines
14 KiB
Java
347 lines
14 KiB
Java
package com.za.net;
|
||
|
||
import android.util.Base64;
|
||
import android.util.Log;
|
||
|
||
import java.io.ByteArrayOutputStream;
|
||
import java.security.Key;
|
||
import java.security.KeyFactory;
|
||
import java.security.KeyPair;
|
||
import java.security.KeyPairGenerator;
|
||
import java.security.PrivateKey;
|
||
import java.security.PublicKey;
|
||
import java.security.Signature;
|
||
import java.security.interfaces.RSAPrivateKey;
|
||
import java.security.interfaces.RSAPublicKey;
|
||
import java.security.spec.PKCS8EncodedKeySpec;
|
||
import java.security.spec.X509EncodedKeySpec;
|
||
import java.util.HashMap;
|
||
import java.util.Map;
|
||
|
||
import javax.crypto.Cipher;
|
||
|
||
public class RSAUtils {
|
||
|
||
/**
|
||
* 加密算法RSA
|
||
*/
|
||
private static final String KEY_ALGORITHM = "RSA";
|
||
|
||
/**
|
||
* 签名算法
|
||
*/
|
||
// public static final String SIGNATURE_ALGORITHM = "MD5withRSA";
|
||
private static final String SIGNATURE_ALGORITHM = "SHA256withRSA";
|
||
|
||
/**
|
||
* 获取公钥的key
|
||
*/
|
||
private static final String PUBLIC_KEY = "RSAPublicKey";
|
||
|
||
|
||
/**
|
||
* 获取私钥的key
|
||
*/
|
||
private static final String PRIVATE_KEY = "RSAPrivateKey";
|
||
|
||
|
||
/**
|
||
* RSA最大加密明文大小
|
||
*/
|
||
private static final int MAX_ENCRYPT_BLOCK = 117;
|
||
|
||
|
||
/**
|
||
* RSA最大解密密文大小
|
||
*/
|
||
private static final int MAX_DECRYPT_BLOCK = 256;
|
||
|
||
|
||
public static void main(String[] args) {
|
||
// try {
|
||
// genKeyPair(2048);
|
||
// } catch (Exception e) {
|
||
// e.printStackTrace();
|
||
// }
|
||
/*RSA 2048*/
|
||
// String publicKey = "MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAichGTEP0QFswnvn+ZAQrgGHM8VeDZLJuezGhgxh4d9SyRUfnIW/zefT71rwS4bZUs1MPxJwavOyxABJOHLuckdHXknCsGEWz78gsA6D0+O+9dl1gCZR29nnN/NlzmNbSjFnzvsTJYBlS88qSr35RXFE+6DM7uPsS8Fm2I+65FteJ8p2yMvpSg72QkIX8xvI1F1uwXrciIB+4u7uTozxIplMOo4a6uhAm3W+Kjpz3ni2btjGqHRbqb3ebSZyl+nFfnjQaBe3XyVxAWDSanjgFj/wbqbeug9FBs+nQFVPIZR9z0aE5Ndi5o3eSkV7HFmWpkxaiPZ0BLRK3XHMaBtuSpwIDAQAB";
|
||
// String privateKey = "MIIEvwIBADANBgkqhkiG9w0BAQEFAASCBKkwggSlAgEAAoIBAQCJyEZMQ/RAWzCe+f5kBCuAYczxV4Nksm57MaGDGHh31LJFR+chb/N59PvWvBLhtlSzUw/EnBq87LEAEk4cu5yR0deScKwYRbPvyCwDoPT47712XWAJlHb2ec382XOY1tKMWfO+xMlgGVLzypKvflFcUT7oMzu4+xLwWbYj7rkW14nynbIy+lKDvZCQhfzG8jUXW7BetyIgH7i7u5OjPEimUw6jhrq6ECbdb4qOnPeeLZu2MaodFupvd5tJnKX6cV+eNBoF7dfJXEBYNJqeOAWP/Bupt66D0UGz6dAVU8hlH3PRoTk12Lmjd5KRXscWZamTFqI9nQEtErdccxoG25KnAgMBAAECggEBAIPz1b88ZTMtIgdejA7lH3Q4Nbn8gc1yRPSet3uBd/3rKT/IeMZBHQBzaqxgOgUIRV3n8nXsun6sf2b+IOjLlErimH2agnZMauL85YokH/g4QU6WZl9GXBf41xmMd3SsZ8AadaEBfYoXNqZcHtcLNogfFwvx5QRnD+A3SoRnH8OLBeVvOEe4AqHLT2xEZ9TeCf3fJe0Rf0fUIbw7I5ioiRZV/ir0L1VM7+1k2JODUkdC2Luj5Tl3nl1Eg6EmkYCmGE1bip1NAatsfjPBLMF7XdPNjLboiffjgKVBOjb7Y9vL18BCoLtWeTT2GkMpi5Sr94T1te1Ox77dF4BP33Xn7eECgYEA1TNUrAQsh14NbbkwFtUHXS8/YXt81p9wbSpFBymIawF2Lkk0913TB4CHSun45LhYXjdZZxK/TgqC5EIq5v2RA0jY3cSxoqVe6RZKB04E8wszeJHiEJPdu2vFnpZh9iAyhswiM5FmuKZKoWsVc2SZrBXAI02smSn3lXYok1VBS3sCgYEApXEZS6gjUu4o7ZL53Ur1HDfi/nxpkxqrPh+D1HVYjzjT+4vTeZwtLXt2VCInPWNXH+f11mzhxIrLkI0jMcSCah81DuU8aFXnqvPuyFvt9uaQBYlVWBtkcGZyeaxHFrbfCyeu0jm7SfwmiIg12hKlIHtPTjEZQUX+kkWr8cdaZ8UCgYEAh0Pl+K09QzVc97yC0jmeTnTnlYWvksvdnKUw3nZvYtSukndH75nLhfr524HOs+5xwnUDd+3hCjaJDSEd7yf5lUfmr+1XdoXNTb0igrfxU/JLWbfU4geuqnaaDyACTxHmfLePC4C413ZJ61fxaCDvjsrN+JgTZanGt0EcRT3WC3kCgYEAgf5/GMJxlw0JXbs515a5R8Xl9358Whj/at3KcRsPTeIiNqnkrc54dR9ol60KViMDZ0+VDDobn5pLXzZ26/jzXD1PLHgU4gp18Q6glhAdx/3cNm11gLhtUCA/XLlwVjm0wggZRpgUQIr/IBKe9c3mr8IUS2Uq6e38nKRf+adhst0CgYAM4tvl+U1MPbbz3YzDv8QPepZ7Pglgdfxqfr5OkXA7jNhqTZjSq10B6oClGvirBo1m6f26F02iUKk1n67AuiLlTP/RRZHi1cfq6P9IaXl23PcxJfUMvIxQDS0U+UTFpNXryTw/qNAkSfufN48YzKdGvc8vHrYJyaeemaVlbdJOCw==";
|
||
//
|
||
//
|
||
// try {
|
||
//
|
||
// String data = "测试提交数据";
|
||
//
|
||
// byte[] publicEncryptBytes = RSAUtils.encryptByPublicKey(data.getBytes(), publicKey);
|
||
// System.out.println("公钥加密后的数据:" + Base64.encodeToString(publicEncryptBytes, Base64.URL_SAFE | Base64.NO_WRAP));
|
||
// byte[] privatDecryptBytes = RSAUtils.decryptByPrivateKey(publicEncryptBytes, privateKey);
|
||
// System.out.println("私钥解密后的数据:" + new String(privatDecryptBytes));
|
||
//
|
||
//
|
||
// System.out.println("--------------------");
|
||
//
|
||
// byte[] privateKeyEncryptBytes = RSAUtils.encryptByPrivateKey(data.getBytes(), privateKey);
|
||
// System.out.println("私钥加密后的数据:" + Base64.encodeToString(privateKeyEncryptBytes, Base64.URL_SAFE | Base64.NO_WRAP));
|
||
//
|
||
// String singnData = RSAUtils.sign(data.getBytes(), privateKey);
|
||
// System.out.println("私钥签名后的数据:" + singnData);
|
||
//
|
||
//
|
||
// byte[] publicDecryptBytes = RSAUtils.decryptByPublicKey(privateKeyEncryptBytes, publicKey);
|
||
// System.out.println("公钥解密后的数据:" + new String(publicDecryptBytes));
|
||
//
|
||
// boolean isSign = RSAUtils.verify(data.getBytes(), publicKey, singnData);
|
||
// System.out.println("签名是否正确:" + isSign);
|
||
//
|
||
//
|
||
// } catch (Exception e) {
|
||
// e.printStackTrace();
|
||
// }
|
||
}
|
||
|
||
|
||
/**
|
||
* @param keySize 生成的秘钥长度 一般为1024或2048
|
||
* @return
|
||
* @throws Exception
|
||
*/
|
||
public static Map<String, Object> genKeyPair(int keySize) throws Exception {
|
||
KeyPairGenerator keyPairGen = KeyPairGenerator.getInstance(KEY_ALGORITHM);
|
||
keyPairGen.initialize(keySize);
|
||
KeyPair keyPair = keyPairGen.generateKeyPair();
|
||
RSAPublicKey publicKey = (RSAPublicKey) keyPair.getPublic();
|
||
RSAPrivateKey privateKey = (RSAPrivateKey) keyPair.getPrivate();
|
||
Map<String, Object> keyMap = new HashMap<String, Object>(2);
|
||
keyMap.put(PUBLIC_KEY, publicKey);
|
||
keyMap.put(PRIVATE_KEY, privateKey);
|
||
|
||
// Log.d("DoggieX","publicKey:" + Base64.encodeToString(publicKey.getEncoded(), Base64.URL_SAFE | Base64.NO_WRAP));
|
||
// Log.d("DoggieX","privateKey:" + Base64.encodeToString(privateKey.getEncoded(), Base64.URL_SAFE | Base64.NO_WRAP));
|
||
Log.d("DoggieX","publicKey:" + Base64.encodeToString(publicKey.getEncoded(), Base64.NO_WRAP));
|
||
Log.d("DoggieX","privateKey:" + Base64.encodeToString(privateKey.getEncoded(), Base64.NO_WRAP));
|
||
|
||
return keyMap;
|
||
}
|
||
|
||
|
||
/**
|
||
* 对已加密数据进行签名
|
||
*
|
||
* @param data 已加密的数据
|
||
* @param privateKey 私钥
|
||
* @return 对已加密数据生成的签名
|
||
* @throws Exception
|
||
*/
|
||
|
||
public static String sign(byte[] data, String privateKey) throws Exception {
|
||
byte[] keyBytes = Base64.decode(privateKey, Base64.URL_SAFE | Base64.NO_WRAP);
|
||
PKCS8EncodedKeySpec pkcs8KeySpec = new PKCS8EncodedKeySpec(keyBytes);
|
||
KeyFactory keyFactory = KeyFactory.getInstance(KEY_ALGORITHM);
|
||
PrivateKey privateK = keyFactory.generatePrivate(pkcs8KeySpec);
|
||
Signature signature = Signature.getInstance(SIGNATURE_ALGORITHM);
|
||
signature.initSign(privateK);
|
||
signature.update(data);
|
||
return Base64.encodeToString(signature.sign(), Base64.URL_SAFE);
|
||
}
|
||
|
||
|
||
/**
|
||
* 验签
|
||
*
|
||
* @param data 签名之前的数据
|
||
* @param publicKey 公钥
|
||
* @param sign 签名之后的数据
|
||
* @return 验签是否成功
|
||
* @throws Exception
|
||
*/
|
||
public static boolean verify(byte[] data, String publicKey, String sign) throws Exception {
|
||
byte[] keyBytes = Base64.decode(publicKey, Base64.URL_SAFE | Base64.NO_WRAP);
|
||
X509EncodedKeySpec keySpec = new X509EncodedKeySpec(keyBytes);
|
||
KeyFactory keyFactory = KeyFactory.getInstance(KEY_ALGORITHM);
|
||
PublicKey publicK = keyFactory.generatePublic(keySpec);
|
||
Signature signature = Signature.getInstance(SIGNATURE_ALGORITHM);
|
||
signature.initVerify(publicK);
|
||
signature.update(data);
|
||
return signature.verify(Base64.decode(sign, Base64.URL_SAFE));
|
||
}
|
||
|
||
|
||
/**
|
||
* 用私钥对数据进行解密
|
||
*
|
||
* @param encryptedData 使用公钥加密过的数据
|
||
* @param privateKey 私钥
|
||
* @return 解密后的数据
|
||
* @throws Exception
|
||
*/
|
||
public static byte[] decryptByPrivateKey(byte[] encryptedData, String privateKey) throws Exception {
|
||
byte[] keyBytes = Base64.decode(privateKey, Base64.NO_WRAP);
|
||
PKCS8EncodedKeySpec pkcs8KeySpec = new PKCS8EncodedKeySpec(keyBytes);
|
||
KeyFactory keyFactory = KeyFactory.getInstance(KEY_ALGORITHM);
|
||
Key privateK = keyFactory.generatePrivate(pkcs8KeySpec);
|
||
//Cipher cipher = Cipher.getInstance(keyFactory.getAlgorithm());
|
||
Cipher cipher = Cipher.getInstance("RSA/ECB/PKCS1Padding","BC");
|
||
cipher.init(Cipher.DECRYPT_MODE, privateK);
|
||
Log.d("DoggieX","provider:"+cipher.getProvider().getClass().getName());
|
||
|
||
int inputLen = encryptedData.length;
|
||
ByteArrayOutputStream out = new ByteArrayOutputStream();
|
||
int offSet = 0;
|
||
byte[] cache;
|
||
int i = 0;
|
||
// 对数据分段解密
|
||
while (inputLen - offSet > 0) {
|
||
if (inputLen - offSet > MAX_DECRYPT_BLOCK) {
|
||
cache = cipher.doFinal(encryptedData, offSet, MAX_DECRYPT_BLOCK);
|
||
} else {
|
||
cache = cipher.doFinal(encryptedData, offSet, inputLen - offSet);
|
||
}
|
||
out.write(cache, 0, cache.length);
|
||
i++;
|
||
offSet = i * MAX_DECRYPT_BLOCK;
|
||
}
|
||
byte[] decryptedData = out.toByteArray();
|
||
out.close();
|
||
|
||
|
||
return decryptedData;
|
||
}
|
||
|
||
/**
|
||
* 公钥解密
|
||
*
|
||
* @param encryptedData 使用私钥加密过的数据
|
||
* @param publicKey 公钥
|
||
* @return 解密后的数据
|
||
* @throws Exception
|
||
*/
|
||
public static byte[] decryptByPublicKey(byte[] encryptedData, String publicKey) throws Exception {
|
||
byte[] keyBytes = Base64.decode(publicKey, Base64.URL_SAFE | Base64.NO_WRAP);
|
||
X509EncodedKeySpec x509KeySpec = new X509EncodedKeySpec(keyBytes);
|
||
KeyFactory keyFactory = KeyFactory.getInstance(KEY_ALGORITHM);
|
||
Key publicK = keyFactory.generatePublic(x509KeySpec);
|
||
Cipher cipher = Cipher.getInstance(keyFactory.getAlgorithm());
|
||
cipher.init(Cipher.DECRYPT_MODE, publicK);
|
||
int inputLen = encryptedData.length;
|
||
ByteArrayOutputStream out = new ByteArrayOutputStream();
|
||
int offSet = 0;
|
||
byte[] cache;
|
||
int i = 0;
|
||
// 对数据分段解密
|
||
while (inputLen - offSet > 0) {
|
||
if (inputLen - offSet > MAX_DECRYPT_BLOCK) {
|
||
cache = cipher.doFinal(encryptedData, offSet, MAX_DECRYPT_BLOCK);
|
||
} else {
|
||
cache = cipher.doFinal(encryptedData, offSet, inputLen - offSet);
|
||
}
|
||
out.write(cache, 0, cache.length);
|
||
i++;
|
||
offSet = i * MAX_DECRYPT_BLOCK;
|
||
}
|
||
byte[] decryptedData = out.toByteArray();
|
||
out.close();
|
||
return decryptedData;
|
||
}
|
||
|
||
|
||
/**
|
||
* 公钥加密
|
||
*
|
||
* @param data 需要加密的数据
|
||
* @param publicKey 公钥
|
||
* @return 使用公钥加密后的数据
|
||
* @throws Exception
|
||
*/
|
||
public static byte[] encryptByPublicKey(byte[] data, String publicKey) throws Exception {
|
||
byte[] keyBytes = Base64.decode(publicKey, Base64.NO_WRAP);
|
||
X509EncodedKeySpec x509KeySpec = new X509EncodedKeySpec(keyBytes);
|
||
KeyFactory keyFactory = KeyFactory.getInstance(KEY_ALGORITHM);
|
||
Key publicK = keyFactory.generatePublic(x509KeySpec);
|
||
// 对数据加密
|
||
Cipher cipher = Cipher.getInstance("RSA/ECB/PKCS1Padding","BC");
|
||
cipher.init(Cipher.ENCRYPT_MODE, publicK);
|
||
Log.d("DoggieX",cipher.getProvider().getClass().toString());
|
||
int inputLen = data.length;
|
||
ByteArrayOutputStream out = new ByteArrayOutputStream();
|
||
int offSet = 0;
|
||
byte[] cache;
|
||
int i = 0;
|
||
// 对数据分段加密
|
||
while (inputLen - offSet > 0) {
|
||
if (inputLen - offSet > MAX_ENCRYPT_BLOCK) {
|
||
cache = cipher.doFinal(data, offSet, MAX_ENCRYPT_BLOCK);
|
||
} else {
|
||
cache = cipher.doFinal(data, offSet, inputLen - offSet);
|
||
}
|
||
out.write(cache, 0, cache.length);
|
||
i++;
|
||
offSet = i * MAX_ENCRYPT_BLOCK;
|
||
}
|
||
byte[] encryptedData = out.toByteArray();
|
||
out.close();
|
||
return encryptedData;
|
||
}
|
||
|
||
|
||
/**
|
||
* 私钥加密
|
||
*
|
||
* @param data 待加密的数据
|
||
* @param privateKey 私钥
|
||
* @return 使用私钥加密后的数据
|
||
* @throws Exception
|
||
*/
|
||
public static byte[] encryptByPrivateKey(byte[] data, String privateKey) throws Exception {
|
||
byte[] keyBytes = Base64.decode(privateKey, Base64.URL_SAFE | Base64.NO_WRAP);
|
||
PKCS8EncodedKeySpec pkcs8KeySpec = new PKCS8EncodedKeySpec(keyBytes);
|
||
KeyFactory keyFactory = KeyFactory.getInstance(KEY_ALGORITHM);
|
||
Key privateK = keyFactory.generatePrivate(pkcs8KeySpec);
|
||
Cipher cipher = Cipher.getInstance(keyFactory.getAlgorithm());
|
||
cipher.init(Cipher.ENCRYPT_MODE, privateK);
|
||
int inputLen = data.length;
|
||
ByteArrayOutputStream out = new ByteArrayOutputStream();
|
||
int offSet = 0;
|
||
byte[] cache;
|
||
int i = 0;
|
||
// 对数据分段加密
|
||
while (inputLen - offSet > 0) {
|
||
if (inputLen - offSet > MAX_ENCRYPT_BLOCK) {
|
||
cache = cipher.doFinal(data, offSet, MAX_ENCRYPT_BLOCK);
|
||
} else {
|
||
cache = cipher.doFinal(data, offSet, inputLen - offSet);
|
||
}
|
||
out.write(cache, 0, cache.length);
|
||
i++;
|
||
offSet = i * MAX_ENCRYPT_BLOCK;
|
||
}
|
||
byte[] encryptedData = out.toByteArray();
|
||
out.close();
|
||
return encryptedData;
|
||
}
|
||
|
||
|
||
/**
|
||
* 获取私钥
|
||
*
|
||
* @param keyMap 生成的秘钥对
|
||
* @return
|
||
* @throws Exception
|
||
*/
|
||
public static String getPrivateKey(Map<String, Object> keyMap) throws Exception {
|
||
Key key = (Key) keyMap.get(PRIVATE_KEY);
|
||
return Base64.encodeToString(key.getEncoded(), Base64.URL_SAFE | Base64.NO_WRAP);
|
||
}
|
||
|
||
|
||
/**
|
||
* 获取公钥
|
||
*
|
||
* @param keyMap 生成的秘钥对
|
||
* @return
|
||
* @throws Exception
|
||
*/
|
||
public static String getPublicKey(Map<String, Object> keyMap) throws Exception {
|
||
Key key = (Key) keyMap.get(PUBLIC_KEY);
|
||
return Base64.encodeToString(key.getEncoded(), Base64.URL_SAFE | Base64.NO_WRAP);
|
||
}
|
||
} |