Java数字证书对文件、加密、解密、签名、校验签名(一)

本文中的Base64Utils.java在其他随笔中已经贴出。Java证书生成命令如下,不做过多解释,可先到网上查询下资料,本文仅提供工具类代码:
把生成的密钥库和证书都放到类的同包下。

keytool - validity 365 - genkey - v - aliaswww.asdc.com.cn - keyalg RSA - keystore D :\ key \ asdc . keystore - dname " CN=172.25.67.98,OU=stos,O=asdc,L=Haidian,ST=Beijing,c=cn " - storepass 123456 - keypass 123456

keytool
- export - v - aliaswww.asdc.com.cn - keystore D :\ key \asdc . keystore - storepass 123456 - rfc - fileD :\ key \asdc . cer



CertificateUtils.java

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.MappedByteBuffer;
import java.nio.channels.FileChannel;
import java.security.KeyStore;
import java.security.PrivateKey;
import java.security.PublicKey;
import java.security.Signature;
import java.security.cert.Certificate;
import java.security.cert.CertificateFactory;
import java.security.cert.X509Certificate;
import java.util.Date;

import javax.crypto.Cipher;

Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public class CertificateUtils {

Java数字证书对文件、加密、解密、签名、校验签名(一)

public static final String KEY_STORE = "JKS";

public static final String X509 = "X.509";

Java数字证书对文件、加密、解密、签名、校验签名(一)

private static final int CACHE_SIZE = 2048;

Java数字证书对文件、加密、解密、签名、校验签名(一)

private static final int MAX_ENCRYPT_BLOCK = 117;

Java数字证书对文件、加密、解密、签名、校验签名(一)

private static final int MAX_DECRYPT_BLOCK = 128;

Java数字证书对文件、加密、解密、签名、校验签名(一)

private static PrivateKey getPrivateKey(String keyStorePath,String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
KeyStore keyStore
=getKeyStore(keyStorePath,password);
PrivateKey privateKey
= (PrivateKey) keyStore.getKey(alias,password.toCharArray());
return privateKey;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

private static KeyStore getKeyStore(String keyStorePath,String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
FileInputStream in
=new FileInputStream(keyStorePath);
KeyStore keyStore
=KeyStore.getInstance(KEY_STORE);
keyStore.load(in, password.toCharArray());
in.close();
return keyStore;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

private static PublicKey getPublicKey(StringcertificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
Certificate certificate
= getCertificate(certificatePath);
PublicKey publicKey
=certificate.getPublicKey();
return publicKey;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

private static Certificate getCertificate(StringcertificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
CertificateFactory certificateFactory
= CertificateFactory.getInstance(X509);
FileInputStream in
=new FileInputStream(certificatePath);
Certificate certificate
= certificateFactory.generateCertificate(in);
in.close();
return certificate;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

private static Certificate getCertificate(String keyStorePath,String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
KeyStore keyStore
=getKeyStore(keyStorePath,password);
Certificate certificate
= keyStore.getCertificate(alias);
return certificate;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]encryptByPrivateKey(byte[]data, String keyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 取得私钥
PrivateKey privateKey= getPrivateKey(keyStorePath, alias,password);
Cipher cipher
=Cipher.getInstance(privateKey.getAlgorithm());
cipher.init(Cipher.ENCRYPT_MODE, privateKey);
int inputLen = data.length;
ByteArrayOutputStream out
= new ByteArrayOutputStream();
int offSet = 0;
byte[] cache;
int i = 0;
// 对数据分段加密
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (inputLen - offSet > 0) {
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (inputLen - offSet > MAX_ENCRYPT_BLOCK) {
cache
= cipher.doFinal(data, offSet,MAX_ENCRYPT_BLOCK);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
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;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]encryptFileByPrivateKey(String filePath, String keyStorePath,String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] data = fileToByte(filePath);
return encryptByPrivateKey(data, keyStorePath, alias,password);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static void encryptFileByPrivateKey(String srcFilePath,String destFilePath, String keyStorePath, String alias, Stringpassword)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 取得私钥
PrivateKey privateKey= getPrivateKey(keyStorePath, alias,password);
Cipher cipher
=Cipher.getInstance(privateKey.getAlgorithm());
cipher.init(Cipher.ENCRYPT_MODE, privateKey);
File srcFile
=new File(srcFilePath);
FileInputStream in
=new FileInputStream(srcFile);
File destFile
=new File(destFilePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (!destFile.getParentFile().exists()){
destFile.getParentFile().mkdirs();
}

destFile.createNewFile();
OutputStream out
=new FileOutputStream(destFile);
byte[] data = new byte[MAX_ENCRYPT_BLOCK];
byte[] encryptedData; // 加密块
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (in.read(data) != -1) {
encryptedData
=cipher.doFinal(data);
out.write(encryptedData,
0,encryptedData.length);
out.flush();
}

out.close();
in.close();
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static String encryptFileToBase64ByPrivateKey(StringfilePath, String keyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] encryptedData = encryptFileByPrivateKey(filePath, keyStorePath,alias, password);
return Base64Utils.encode(encryptedData);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]decryptByPrivateKey(byte[]encryptedData, String keyStorePath, String alias, Stringpassword)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 取得私钥
PrivateKey privateKey= getPrivateKey(keyStorePath, alias,password);
Cipher cipher
=Cipher.getInstance(privateKey.getAlgorithm());
cipher.init(Cipher.DECRYPT_MODE, privateKey);
// 解密byte数组最大长度限制: 128
int inputLen = encryptedData.length;
ByteArrayOutputStream out
= new ByteArrayOutputStream();
int offSet = 0;
byte[] cache;
int i = 0;
// 对数据分段解密
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (inputLen - offSet > 0) {
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (inputLen - offSet > MAX_DECRYPT_BLOCK) {
cache
= cipher.doFinal(encryptedData, offSet,MAX_DECRYPT_BLOCK);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
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;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]encryptByPublicKey(byte[]data, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 取得公钥
PublicKey publicKey= getPublicKey(certificatePath);
Cipher cipher
=Cipher.getInstance(publicKey.getAlgorithm());
cipher.init(Cipher.ENCRYPT_MODE, publicKey);
int inputLen = data.length;
ByteArrayOutputStream out
= new ByteArrayOutputStream();
int offSet = 0;
byte[] cache;
int i = 0;
// 对数据分段加密
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (inputLen - offSet > 0) {
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (inputLen - offSet > MAX_ENCRYPT_BLOCK) {
cache
= cipher.doFinal(data, offSet,MAX_ENCRYPT_BLOCK);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
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;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]decryptByPublicKey(byte[]encryptedData, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
PublicKey publicKey
=getPublicKey(certificatePath);
Cipher cipher
=Cipher.getInstance(publicKey.getAlgorithm());
cipher.init(Cipher.DECRYPT_MODE, publicKey);
int inputLen = encryptedData.length;
ByteArrayOutputStream out
= new ByteArrayOutputStream();
int offSet = 0;
byte[] cache;
int i = 0;
// 对数据分段解密
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (inputLen - offSet > 0) {
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (inputLen - offSet > MAX_DECRYPT_BLOCK) {
cache
= cipher.doFinal(encryptedData, offSet,MAX_DECRYPT_BLOCK);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
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;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static void decryptFileByPublicKey(String srcFilePath,String destFilePath, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
PublicKey publicKey
=getPublicKey(certificatePath);
Cipher cipher
=Cipher.getInstance(publicKey.getAlgorithm());
cipher.init(Cipher.DECRYPT_MODE, publicKey);
File srcFile
=new File(srcFilePath);
FileInputStream in
=new FileInputStream(srcFile);
File destFile
=new File(destFilePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (!destFile.getParentFile().exists()){
destFile.getParentFile().mkdirs();
}

destFile.createNewFile();
OutputStream out
=new FileOutputStream(destFile);
byte[] data = new byte[MAX_DECRYPT_BLOCK];
byte[] decryptedData; // 解密块
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (in.read(data) != -1) {
decryptedData
=cipher.doFinal(data);
out.write(decryptedData,
0,decryptedData.length);
out.flush();
}

out.close();
in.close();
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]sign(byte[]data, String keyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(keyStorePath, alias, password);
// 获取私钥
KeyStore keyStore= getKeyStore(keyStorePath, password);
// 取得私钥
PrivateKey privateKey= (PrivateKey) keyStore.getKey(alias,password.toCharArray());
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initSign(privateKey);
signature.update(data);
return signature.sign();
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static String signToBase64(byte[]data, String keyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
return Base64Utils.encode(sign(data, keyStorePath,alias, password));
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static String signFileToBase64WithEncrypt(StringfilePath, String keyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] encryptedData = encryptFileByPrivateKey(filePath, keyStorePath,alias, password);
return signToBase64(encryptedData, keyStorePath,alias, password);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

@Deprecated
public static byte[]signFile(String filePath, String keyStorePath, String alias, Stringpassword)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] sign = new byte[0];
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(keyStorePath, alias, password);
// 获取私钥
KeyStore keyStore= getKeyStore(keyStorePath, password);
// 取得私钥
PrivateKey privateKey= (PrivateKey) keyStore.getKey(alias,password.toCharArray());
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initSign(privateKey);
File file
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (file.exists()) {
FileInputStream in
=new FileInputStream(file);
FileChannel fileChannel
= in.getChannel();
MappedByteBuffer byteBuffer
= fileChannel.map(FileChannel.MapMode.READ_ONLY,0, file.length());
signature.update(byteBuffer);
fileChannel.close();
in.close();
sign
= signature.sign();
}

return sign;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]generateFileSign(String filePath, String keyStorePath, Stringalias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] sign = new byte[0];
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(keyStorePath, alias, password);
// 获取私钥
KeyStore keyStore= getKeyStore(keyStorePath, password);
// 取得私钥
PrivateKey privateKey= (PrivateKey) keyStore.getKey(alias,password.toCharArray());
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initSign(privateKey);
File file
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (file.exists()) {
FileInputStream in
=new FileInputStream(file);
byte[] cache = new byte[CACHE_SIZE];
int nRead = 0;
Java数字证书对文件、加密、解密、签名、校验签名(一)
while ((nRead = in.read(cache)) != -1) {
signature.update(cache,
0,nRead);
}

in.close();
sign
= signature.sign();
}

return sign;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static String signFileToBase64(String filePath, StringkeyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
return Base64Utils.encode(generateFileSign(filePath,keyStorePath, alias, password));
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static boolean verifySign(byte[]data, String sign, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(certificatePath);
// 获得公钥
PublicKey publicKey= x509Certificate.getPublicKey();
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initVerify(publicKey);
signature.update(data);
return signature.verify(Base64Utils.decode(sign));
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

@Deprecated
public static boolean verifyFileSign(String filePath, String sign,String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
boolean result = false;
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(certificatePath);
// 获得公钥
PublicKey publicKey= x509Certificate.getPublicKey();
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initVerify(publicKey);
File file
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (file.exists()) {
byte[] decodedSign = Base64Utils.decode(sign);
FileInputStream in
=new FileInputStream(file);
FileChannel fileChannel
= in.getChannel();
MappedByteBuffer byteBuffer
= fileChannel.map(FileChannel.MapMode.READ_ONLY,0, file.length());
signature.update(byteBuffer);
in.close();
result
= signature.verify(decodedSign);
}

return result;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static boolean validateFileSign(String filePath, String sign,String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
boolean result = false;
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(certificatePath);
// 获得公钥
PublicKey publicKey= x509Certificate.getPublicKey();
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initVerify(publicKey);
File file
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (file.exists()) {
byte[] decodedSign = Base64Utils.decode(sign);
FileInputStream in
=new FileInputStream(file);
byte[] cache = new byte[CACHE_SIZE];
int nRead = 0;
Java数字证书对文件、加密、解密、签名、校验签名(一)
while ((nRead = in.read(cache)) != -1) {
signature.update(cache,
0,nRead);
}

in.close();
result
= signature.verify(decodedSign);
}

return result;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static boolean verifyBase64Sign(String base64String, Stringsign, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] data = Base64Utils.decode(base64String);
return verifySign(data, sign, certificatePath);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static boolean verifyBase64SignWithDecrypt(Stringbase64String, String sign, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] encryptedData = Base64Utils.decode(base64String);
byte[] data = decryptByPublicKey(encryptedData,certificatePath);
return verifySign(data, sign, certificatePath);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static boolean verifyFileSignWithDecrypt(StringencryptedFilePath, String sign, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] encryptedData = fileToByte(encryptedFilePath);
byte[] data = decryptByPublicKey(encryptedData,certificatePath);
return verifySign(data, sign, certificatePath);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(Certificatecertificate) {
return verifyCertificate(new Date(), certificate);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(Date date, Certificatecertificate) {
boolean isValid = true;
Java数字证书对文件、加密、解密、签名、校验签名(一)
try {
X509Certificate x509Certificate
= (X509Certificate) certificate;
x509Certificate.checkValidity(date);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
catch (Exception e) {
isValid
=false;
}

return isValid;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(Date date, StringcertificatePath) {
Certificate certificate;
Java数字证书对文件、加密、解密、签名、校验签名(一)
try {
certificate
=getCertificate(certificatePath);
return verifyCertificate(certificate);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
catch (Exception e) {
e.printStackTrace();
return false;
}

}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(Date date, StringkeyStorePath, String alias, String password) {
Certificate certificate;
Java数字证书对文件、加密、解密、签名、校验签名(一)
try {
certificate
=getCertificate(keyStorePath, alias,password);
return verifyCertificate(certificate);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
catch (Exception e) {
e.printStackTrace();
return false;
}

}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(String keyStorePath, Stringalias, String password) {
return verifyCertificate(new Date(), keyStorePath, alias, password);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(StringcertificatePath) {
return verifyCertificate(new Date(), certificatePath);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static byte[]fileToByte(String filePath) throws Exception {
byte[] data = new byte[0];
File file
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (file.exists()) {
FileInputStream in
=new FileInputStream(file);
ByteArrayOutputStream out
= new ByteArrayOutputStream(2048);
byte[] cache = new byte[CACHE_SIZE];
int nRead = 0;
Java数字证书对文件、加密、解密、签名、校验签名(一)
while ((nRead = in.read(cache)) != -1) {
out.write(cache,
0,nRead);
out.flush();
}

out.close();
in.close();
data
= out.toByteArray();
}

return data;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static void byteArrayToFile(byte[]bytes, String filePath) throws Exception {
InputStream in
=new ByteArrayInputStream(bytes);
File destFile
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (!destFile.getParentFile().exists()){
destFile.getParentFile().mkdirs();
}

destFile.createNewFile();
OutputStream out
=new FileOutputStream(destFile);
byte[] cache = new byte[CACHE_SIZE];
int nRead = 0;
Java数字证书对文件、加密、解密、签名、校验签名(一)
while ((nRead = in.read(cache)) != -1) {
out.write(cache,
0,nRead);
out.flush();
}

out.close();
in.close();
}


}
本文中的Base64Utils.java在其他随笔中已经贴出。Java证书生成命令如下,不做过多解释,可先到网上查询下资料,本文仅提供工具类代码:
把生成的密钥库和证书都放到类的同包下。

keytool - validity 365 - genkey - v - aliaswww.asdc.com.cn - keyalg RSA - keystore D :\ key \ asdc . keystore - dname " CN=172.25.67.98,OU=stos,O=asdc,L=Haidian,ST=Beijing,c=cn " - storepass 123456 - keypass 123456

keytool
- export - v - aliaswww.asdc.com.cn - keystore D :\ key \asdc . keystore - storepass 123456 - rfc - fileD :\ key \asdc . cer



CertificateUtils.java

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.MappedByteBuffer;
import java.nio.channels.FileChannel;
import java.security.KeyStore;
import java.security.PrivateKey;
import java.security.PublicKey;
import java.security.Signature;
import java.security.cert.Certificate;
import java.security.cert.CertificateFactory;
import java.security.cert.X509Certificate;
import java.util.Date;

import javax.crypto.Cipher;

Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public class CertificateUtils {

Java数字证书对文件、加密、解密、签名、校验签名(一)

public static final String KEY_STORE = "JKS";

public static final String X509 = "X.509";

Java数字证书对文件、加密、解密、签名、校验签名(一)

private static final int CACHE_SIZE = 2048;

Java数字证书对文件、加密、解密、签名、校验签名(一)

private static final int MAX_ENCRYPT_BLOCK = 117;

Java数字证书对文件、加密、解密、签名、校验签名(一)

private static final int MAX_DECRYPT_BLOCK = 128;

Java数字证书对文件、加密、解密、签名、校验签名(一)

private static PrivateKey getPrivateKey(String keyStorePath,String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
KeyStore keyStore
=getKeyStore(keyStorePath,password);
PrivateKey privateKey
= (PrivateKey) keyStore.getKey(alias,password.toCharArray());
return privateKey;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

private static KeyStore getKeyStore(String keyStorePath,String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
FileInputStream in
=new FileInputStream(keyStorePath);
KeyStore keyStore
=KeyStore.getInstance(KEY_STORE);
keyStore.load(in, password.toCharArray());
in.close();
return keyStore;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

private static PublicKey getPublicKey(StringcertificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
Certificate certificate
= getCertificate(certificatePath);
PublicKey publicKey
=certificate.getPublicKey();
return publicKey;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

private static Certificate getCertificate(StringcertificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
CertificateFactory certificateFactory
= CertificateFactory.getInstance(X509);
FileInputStream in
=new FileInputStream(certificatePath);
Certificate certificate
= certificateFactory.generateCertificate(in);
in.close();
return certificate;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

private static Certificate getCertificate(String keyStorePath,String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
KeyStore keyStore
=getKeyStore(keyStorePath,password);
Certificate certificate
= keyStore.getCertificate(alias);
return certificate;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]encryptByPrivateKey(byte[]data, String keyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 取得私钥
PrivateKey privateKey= getPrivateKey(keyStorePath, alias,password);
Cipher cipher
=Cipher.getInstance(privateKey.getAlgorithm());
cipher.init(Cipher.ENCRYPT_MODE, privateKey);
int inputLen = data.length;
ByteArrayOutputStream out
= new ByteArrayOutputStream();
int offSet = 0;
byte[] cache;
int i = 0;
// 对数据分段加密
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (inputLen - offSet > 0) {
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (inputLen - offSet > MAX_ENCRYPT_BLOCK) {
cache
= cipher.doFinal(data, offSet,MAX_ENCRYPT_BLOCK);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
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;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]encryptFileByPrivateKey(String filePath, String keyStorePath,String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] data = fileToByte(filePath);
return encryptByPrivateKey(data, keyStorePath, alias,password);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static void encryptFileByPrivateKey(String srcFilePath,String destFilePath, String keyStorePath, String alias, Stringpassword)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 取得私钥
PrivateKey privateKey= getPrivateKey(keyStorePath, alias,password);
Cipher cipher
=Cipher.getInstance(privateKey.getAlgorithm());
cipher.init(Cipher.ENCRYPT_MODE, privateKey);
File srcFile
=new File(srcFilePath);
FileInputStream in
=new FileInputStream(srcFile);
File destFile
=new File(destFilePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (!destFile.getParentFile().exists()){
destFile.getParentFile().mkdirs();
}

destFile.createNewFile();
OutputStream out
=new FileOutputStream(destFile);
byte[] data = new byte[MAX_ENCRYPT_BLOCK];
byte[] encryptedData; // 加密块
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (in.read(data) != -1) {
encryptedData
=cipher.doFinal(data);
out.write(encryptedData,
0,encryptedData.length);
out.flush();
}

out.close();
in.close();
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static String encryptFileToBase64ByPrivateKey(StringfilePath, String keyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] encryptedData = encryptFileByPrivateKey(filePath, keyStorePath,alias, password);
return Base64Utils.encode(encryptedData);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]decryptByPrivateKey(byte[]encryptedData, String keyStorePath, String alias, Stringpassword)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 取得私钥
PrivateKey privateKey= getPrivateKey(keyStorePath, alias,password);
Cipher cipher
=Cipher.getInstance(privateKey.getAlgorithm());
cipher.init(Cipher.DECRYPT_MODE, privateKey);
// 解密byte数组最大长度限制: 128
int inputLen = encryptedData.length;
ByteArrayOutputStream out
= new ByteArrayOutputStream();
int offSet = 0;
byte[] cache;
int i = 0;
// 对数据分段解密
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (inputLen - offSet > 0) {
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (inputLen - offSet > MAX_DECRYPT_BLOCK) {
cache
= cipher.doFinal(encryptedData, offSet,MAX_DECRYPT_BLOCK);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
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;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]encryptByPublicKey(byte[]data, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 取得公钥
PublicKey publicKey= getPublicKey(certificatePath);
Cipher cipher
=Cipher.getInstance(publicKey.getAlgorithm());
cipher.init(Cipher.ENCRYPT_MODE, publicKey);
int inputLen = data.length;
ByteArrayOutputStream out
= new ByteArrayOutputStream();
int offSet = 0;
byte[] cache;
int i = 0;
// 对数据分段加密
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (inputLen - offSet > 0) {
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (inputLen - offSet > MAX_ENCRYPT_BLOCK) {
cache
= cipher.doFinal(data, offSet,MAX_ENCRYPT_BLOCK);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
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;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]decryptByPublicKey(byte[]encryptedData, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
PublicKey publicKey
=getPublicKey(certificatePath);
Cipher cipher
=Cipher.getInstance(publicKey.getAlgorithm());
cipher.init(Cipher.DECRYPT_MODE, publicKey);
int inputLen = encryptedData.length;
ByteArrayOutputStream out
= new ByteArrayOutputStream();
int offSet = 0;
byte[] cache;
int i = 0;
// 对数据分段解密
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (inputLen - offSet > 0) {
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (inputLen - offSet > MAX_DECRYPT_BLOCK) {
cache
= cipher.doFinal(encryptedData, offSet,MAX_DECRYPT_BLOCK);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
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;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static void decryptFileByPublicKey(String srcFilePath,String destFilePath, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
PublicKey publicKey
=getPublicKey(certificatePath);
Cipher cipher
=Cipher.getInstance(publicKey.getAlgorithm());
cipher.init(Cipher.DECRYPT_MODE, publicKey);
File srcFile
=new File(srcFilePath);
FileInputStream in
=new FileInputStream(srcFile);
File destFile
=new File(destFilePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (!destFile.getParentFile().exists()){
destFile.getParentFile().mkdirs();
}

destFile.createNewFile();
OutputStream out
=new FileOutputStream(destFile);
byte[] data = new byte[MAX_DECRYPT_BLOCK];
byte[] decryptedData; // 解密块
Java数字证书对文件、加密、解密、签名、校验签名(一)
while (in.read(data) != -1) {
decryptedData
=cipher.doFinal(data);
out.write(decryptedData,
0,decryptedData.length);
out.flush();
}

out.close();
in.close();
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]sign(byte[]data, String keyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(keyStorePath, alias, password);
// 获取私钥
KeyStore keyStore= getKeyStore(keyStorePath, password);
// 取得私钥
PrivateKey privateKey= (PrivateKey) keyStore.getKey(alias,password.toCharArray());
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initSign(privateKey);
signature.update(data);
return signature.sign();
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static String signToBase64(byte[]data, String keyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
return Base64Utils.encode(sign(data, keyStorePath,alias, password));
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static String signFileToBase64WithEncrypt(StringfilePath, String keyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] encryptedData = encryptFileByPrivateKey(filePath, keyStorePath,alias, password);
return signToBase64(encryptedData, keyStorePath,alias, password);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

@Deprecated
public static byte[]signFile(String filePath, String keyStorePath, String alias, Stringpassword)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] sign = new byte[0];
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(keyStorePath, alias, password);
// 获取私钥
KeyStore keyStore= getKeyStore(keyStorePath, password);
// 取得私钥
PrivateKey privateKey= (PrivateKey) keyStore.getKey(alias,password.toCharArray());
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initSign(privateKey);
File file
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (file.exists()) {
FileInputStream in
=new FileInputStream(file);
FileChannel fileChannel
= in.getChannel();
MappedByteBuffer byteBuffer
= fileChannel.map(FileChannel.MapMode.READ_ONLY,0, file.length());
signature.update(byteBuffer);
fileChannel.close();
in.close();
sign
= signature.sign();
}

return sign;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static byte[]generateFileSign(String filePath, String keyStorePath, Stringalias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] sign = new byte[0];
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(keyStorePath, alias, password);
// 获取私钥
KeyStore keyStore= getKeyStore(keyStorePath, password);
// 取得私钥
PrivateKey privateKey= (PrivateKey) keyStore.getKey(alias,password.toCharArray());
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initSign(privateKey);
File file
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (file.exists()) {
FileInputStream in
=new FileInputStream(file);
byte[] cache = new byte[CACHE_SIZE];
int nRead = 0;
Java数字证书对文件、加密、解密、签名、校验签名(一)
while ((nRead = in.read(cache)) != -1) {
signature.update(cache,
0,nRead);
}

in.close();
sign
= signature.sign();
}

return sign;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static String signFileToBase64(String filePath, StringkeyStorePath, String alias, String password)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
return Base64Utils.encode(generateFileSign(filePath,keyStorePath, alias, password));
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static boolean verifySign(byte[]data, String sign, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(certificatePath);
// 获得公钥
PublicKey publicKey= x509Certificate.getPublicKey();
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initVerify(publicKey);
signature.update(data);
return signature.verify(Base64Utils.decode(sign));
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

@Deprecated
public static boolean verifyFileSign(String filePath, String sign,String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
boolean result = false;
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(certificatePath);
// 获得公钥
PublicKey publicKey= x509Certificate.getPublicKey();
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initVerify(publicKey);
File file
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (file.exists()) {
byte[] decodedSign = Base64Utils.decode(sign);
FileInputStream in
=new FileInputStream(file);
FileChannel fileChannel
= in.getChannel();
MappedByteBuffer byteBuffer
= fileChannel.map(FileChannel.MapMode.READ_ONLY,0, file.length());
signature.update(byteBuffer);
in.close();
result
= signature.verify(decodedSign);
}

return result;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static boolean validateFileSign(String filePath, String sign,String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
boolean result = false;
// 获得证书
X509Certificatex509Certificate =(X509Certificate)getCertificate(certificatePath);
// 获得公钥
PublicKey publicKey= x509Certificate.getPublicKey();
// 构建签名
Signature signature= Signature.getInstance(x509Certificate.getSigAlgName());
signature.initVerify(publicKey);
File file
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (file.exists()) {
byte[] decodedSign = Base64Utils.decode(sign);
FileInputStream in
=new FileInputStream(file);
byte[] cache = new byte[CACHE_SIZE];
int nRead = 0;
Java数字证书对文件、加密、解密、签名、校验签名(一)
while ((nRead = in.read(cache)) != -1) {
signature.update(cache,
0,nRead);
}

in.close();
result
= signature.verify(decodedSign);
}

return result;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static boolean verifyBase64Sign(String base64String, Stringsign, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] data = Base64Utils.decode(base64String);
return verifySign(data, sign, certificatePath);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static boolean verifyBase64SignWithDecrypt(Stringbase64String, String sign, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] encryptedData = Base64Utils.decode(base64String);
byte[] data = decryptByPublicKey(encryptedData,certificatePath);
return verifySign(data, sign, certificatePath);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

public static boolean verifyFileSignWithDecrypt(StringencryptedFilePath, String sign, String certificatePath)
Java数字证书对文件、加密、解密、签名、校验签名(一)
throws Exception {
byte[] encryptedData = fileToByte(encryptedFilePath);
byte[] data = decryptByPublicKey(encryptedData,certificatePath);
return verifySign(data, sign, certificatePath);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(Certificatecertificate) {
return verifyCertificate(new Date(), certificate);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(Date date, Certificatecertificate) {
boolean isValid = true;
Java数字证书对文件、加密、解密、签名、校验签名(一)
try {
X509Certificate x509Certificate
= (X509Certificate) certificate;
x509Certificate.checkValidity(date);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
catch (Exception e) {
isValid
=false;
}

return isValid;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(Date date, StringcertificatePath) {
Certificate certificate;
Java数字证书对文件、加密、解密、签名、校验签名(一)
try {
certificate
=getCertificate(certificatePath);
return verifyCertificate(certificate);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
catch (Exception e) {
e.printStackTrace();
return false;
}

}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(Date date, StringkeyStorePath, String alias, String password) {
Certificate certificate;
Java数字证书对文件、加密、解密、签名、校验签名(一)
try {
certificate
=getCertificate(keyStorePath, alias,password);
return verifyCertificate(certificate);
Java数字证书对文件、加密、解密、签名、校验签名(一) }
catch (Exception e) {
e.printStackTrace();
return false;
}

}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(String keyStorePath, Stringalias, String password) {
return verifyCertificate(new Date(), keyStorePath, alias, password);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static boolean verifyCertificate(StringcertificatePath) {
return verifyCertificate(new Date(), certificatePath);
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static byte[]fileToByte(String filePath) throws Exception {
byte[] data = new byte[0];
File file
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (file.exists()) {
FileInputStream in
=new FileInputStream(file);
ByteArrayOutputStream out
= new ByteArrayOutputStream(2048);
byte[] cache = new byte[CACHE_SIZE];
int nRead = 0;
Java数字证书对文件、加密、解密、签名、校验签名(一)
while ((nRead = in.read(cache)) != -1) {
out.write(cache,
0,nRead);
out.flush();
}

out.close();
in.close();
data
= out.toByteArray();
}

return data;
}


Java数字证书对文件、加密、解密、签名、校验签名(一)

Java数字证书对文件、加密、解密、签名、校验签名(一)
public static void byteArrayToFile(byte[]bytes, String filePath) throws Exception {
InputStream in
=new ByteArrayInputStream(bytes);
File destFile
=new File(filePath);
Java数字证书对文件、加密、解密、签名、校验签名(一)
if (!destFile.getParentFile().exists()){
destFile.getParentFile().mkdirs();
}

destFile.createNewFile();
OutputStream out
=new FileOutputStream(destFile);
byte[] cache = new byte[CACHE_SIZE];
int nRead = 0;
Java数字证书对文件、加密、解密、签名、校验签名(一)
while ((nRead = in.read(cache)) != -1) {
out.write(cache,
0,nRead);
out.flush();
}

out.close();
in.close();
}


}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值