【开发实践】使用AES算法加密 .sql文件

一、需求分析

笔者在开发遇到一个小需求,导出数据的sql文件,由于安全监管的需要,对sql文件进行加密处理。实现将sql文件进行加密,最终呈现如下效果:

二、加密文件

1、加密模块

    /*** AES加密算法** @param sourceFilePath    文件路径* @param encryptedFilePath 目标文件路径* @param secretKey         密钥*/public static void encrypt(String sourceFilePath, String encryptedFilePath, String secretKey) {try {// 生成密钥Key key = generateKey(secretKey);// 初始化加密器Cipher cipher = Cipher.getInstance("AES");cipher.init(Cipher.ENCRYPT_MODE, key);// 读取文件内容int index = 0;// 读取文件内容List<byte[]> somwFileData = getSomwFileData(sourceFilePath);for (byte[] bytes : somwFileData) {// 加密文件byte[] outputBytes = cipher.doFinal(bytes);// 保存加密后的文件FileOutputStream fos = new FileOutputStream(encryptedFilePath + "\\" + fileName[index++]);fos.write(outputBytes);fos.close();}} catch (Exception e) {e.printStackTrace();}}

 笔者的文件分块:

 完整加密解密类 代码:

package util;import javax.crypto.*;
import javax.crypto.spec.SecretKeySpec;
import java.io.*;
import java.security.Key;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;/*** @author : 枫蜜柚子茶* @date : 2023/11/7*/
public class FileEncryption {//自己导出文件的名称数组private static String[] fileName = {"f1", "f2", "f3", "f4", "f5", "f6"};/*** 设置文件路径*/public static void setFileList(String[] strings) {fileName = strings;}/*** 获取文件的字节数据** @param sourceFilePath 文件路径* @param cipher         加密器* @return 字节数据*/private static byte[] getAllFileData(String sourceFilePath, Cipher cipher) throws IOException, IllegalBlockSizeException, BadPaddingException {List<byte[]> res = new ArrayList<>();for (String s : fileName) {File file = new File(sourceFilePath + s);if (file.exists()) {FileInputStream fis = new FileInputStream(file);byte[] inputBytes = new byte[(int) fis.available()];fis.read(inputBytes);fis.close();res.add(cipher.doFinal(inputBytes));}}return concatAll(res);}/*** 获取文件的分片字节数据** @param sourceFilePath 文件路径* @return 字节数据*/private static List<byte[]> getSomwFileData(String sourceFilePath) throws IOException, IllegalBlockSizeException, BadPaddingException {FileInputStream fis = new FileInputStream(new File(sourceFilePath));byte[] inputBytes = new byte[(int) fis.available()];fis.read(inputBytes);fis.close();//加密数据return splitArr(inputBytes, fileName.length);}/*** 合并字节数据** @param list 字节数组* @return 单个字节数组*/private static byte[] concatAll(List<byte[]> list) {if (list == null || list.isEmpty()) {return null;}int totalLength = 0;for (byte[] bytes : list) {totalLength += bytes.length;}byte[] result = Arrays.copyOf(list.get(0), totalLength);int offset = list.get(0).length;for (int i = 1; i < list.size(); i++) {byte[] array = list.get(i);System.arraycopy(array, 0, result, offset, array.length);offset += array.length;}return result;}/*** 对字节数组进行分片** @param array 字节数组* @param num   分片数量* @return 分片数组*/private static List<byte[]> splitArr(byte[] array, int num) {
//        System.out.println(array.length);int everySize = array.length / num;List<byte[]> arrayList = new ArrayList<>();for (int i = 0; i < num; i++) {int index = i * everySize;everySize = i != num - 1 ? everySize : array.length - index;byte[] arr = new byte[everySize];int j = 0;while (j < everySize && index < array.length) {arr[j] = array[index++];j++;}arrayList.add(arr);}return arrayList;}/*** AES加密算法** @param sourceFilePath    文件路径* @param encryptedFilePath 目标文件路径* @param secretKey         密钥*/public static void encrypt(String sourceFilePath, String encryptedFilePath, String secretKey) {try {// 生成密钥Key key = generateKey(secretKey);// 初始化加密器Cipher cipher = Cipher.getInstance("AES");cipher.init(Cipher.ENCRYPT_MODE, key);// 读取文件内容int index = 0;// 读取文件内容List<byte[]> somwFileData = getSomwFileData(sourceFilePath);for (byte[] bytes : somwFileData) {// 加密文件byte[] outputBytes = cipher.doFinal(bytes);// 保存加密后的文件FileOutputStream fos = new FileOutputStream(encryptedFilePath + "\\" + fileName[index++]);fos.write(outputBytes);fos.close();}
//            System.out.println("文件加密成功!");} catch (Exception e) {e.printStackTrace();}}/*** AES解密算法** @param sourceFilePath    文件路径* @param decryptedFilePath 目标文件路径* @param secretKey         密钥*/public static void decrypt(String sourceFilePath, String decryptedFilePath, String secretKey) {try {// 生成密钥Key key = generateKey(secretKey);// 初始化加密器Cipher cipher = Cipher.getInstance("AES");cipher.init(Cipher.DECRYPT_MODE, key);// 读取文件内容byte[] allFileData = getAllFileData(sourceFilePath, cipher);// 加密文件
//            byte[] outputBytes = cipher.doFinal(allFileData);// 保存加密后的文件FileOutputStream fos = new FileOutputStream(decryptedFilePath);fos.write(allFileData);fos.close();
//            System.out.println("文件解密成功!");} catch (Exception e) {e.printStackTrace();}}/*** 密钥对象** @param secretKey 密钥* @return 对象*/private static Key generateKey(String secretKey) throws Exception {KeyGenerator keyGenerator = KeyGenerator.getInstance("AES");keyGenerator.init(128);SecretKey key = new SecretKeySpec(secretKey.getBytes(), "AES");return key;}
}

三、解密文件

解密算法已在完整代码中呈现,这里不在赘述。

解密器的图形化界面代码:

import config.SystemConfig;
import util.FileEncryption;
import util.FileNameCheck;import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.File;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import javax.swing.*;/*** @author : 枫蜜柚子茶* @date : 2023/11/8*/
public class EncryptFrame extends JFrame {// 定义组件JPanel jp1, jp2, jp4;JLabel label;JButton start, reset, scanFile;JTextField textField;//当前文件类File curr;// 构造函数public EncryptFrame() {EncryptFrame encryptFrame = this;jp1 = new JPanel();jp2 = new JPanel();jp4 = new JPanel();label = new JLabel("请选择文件:");textField = new JTextField(20);// 文本域textField.setEnabled(false);scanFile = new JButton("浏览");// 钮1start = new JButton("开始");reset = new JButton("重置");//设置布局为4行1列GridLayout layout = new GridLayout(4, 1);this.setLayout(layout);// 加入各个组件jp2.add(label);jp2.add(textField);jp2.add(scanFile);jp4.add(start);jp4.add(reset);// 加入到JFramethis.add(jp1);this.add(jp2);this.add(jp4);//设置参数this.setSize(500, 260);this.setTitle("解密器");this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);this.setVisible(true);//使得窗口在屏幕中心显示Toolkit kit = Toolkit.getDefaultToolkit();Dimension screenSize = kit.getScreenSize();this.setBounds((screenSize.width - 500) / 2, (screenSize.height - 260) / 2, 500, 260);//获取所选择的文件scanFile.addActionListener(new ActionListener() {@Overridepublic void actionPerformed(ActionEvent e) {JFileChooser chooser = new JFileChooser();chooser.setFileSelectionMode(JFileChooser.FILES_AND_DIRECTORIES);JLabel jLabel = new JLabel();chooser.showDialog(jLabel, "选择");curr = chooser.getSelectedFile();if (curr != null) {textField.setText(curr.getAbsoluteFile().toString());}}});//重置当前选择文件reset.addActionListener(new ActionListener() {@Overridepublic void actionPerformed(ActionEvent e) {textField.setText("");}});//开始处理数据start.addActionListener(new ActionListener() {public void actionPerformed(ActionEvent e) {try {String sourcePath = textField.getText() + "\\";Date date = new Date();SimpleDateFormat format = new SimpleDateFormat("yy年MM月dd日");String decryptedFilePath = "sql\\" + format.format(date) + "数据文件.sql";
//                    System.out.println(sourcePath);if (sourcePath == null || "".equals(sourcePath)) {JOptionPane.showMessageDialog(encryptFrame, "文件路径不能为空!", "警告", JOptionPane.WARNING_MESSAGE);} else if (!curr.exists()) {JOptionPane.showMessageDialog(encryptFrame, "当前文件或者文件夹不存在!", "警告", JOptionPane.WARNING_MESSAGE);} else if (curr.isFile()) {JOptionPane.showMessageDialog(encryptFrame, "请选择文件夹!", "警告", JOptionPane.WARNING_MESSAGE);} else if (curr.listFiles() == null || curr.listFiles().length == 0) {JOptionPane.showMessageDialog(encryptFrame, "该文件夹为空!", "警告", JOptionPane.WARNING_MESSAGE);} else {//遍历当前文件夹下的文件File[] files = curr.listFiles();//收集文件名Set<String> fileName = new HashSet<>();for (File file : files) {fileName.add(file.getName());}if (!FileNameCheck.checkFileName(fileName)) {JOptionPane.showMessageDialog(encryptFrame, "文件不符合解密对象要求!", "警告", JOptionPane.WARNING_MESSAGE);} else {File target = new File("sql");if (!target.exists()) {target.mkdirs();}FileEncryption.decrypt(sourcePath, decryptedFilePath, SystemConfig.secretKey);JOptionPane.showMessageDialog(encryptFrame, "解密成功,已导出到当前文件夹下!", "成功", JOptionPane.INFORMATION_MESSAGE);}}} catch (Exception ex) {JOptionPane.showMessageDialog(encryptFrame, "导出错误!", "失败", JOptionPane.ERROR_MESSAGE);}}});}
}

【运行结果】 

成功将6个乱码文件解密为 .sql文件。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/211289.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

飞翔的鸟游戏

一.准备工作 首先创建一个新的Java项目命名为“飞翔的鸟”&#xff0c;并在src中创建一个包命名为“com.qiku.bird"&#xff0c;在这个包内分别创建4个类命名为“Bird”、“BirdGame”、“Column”、“Ground”&#xff0c;并向需要的图片素材导入到包内。 二.代码呈现 pa…

VBA经典应用69例:Find方法的应用

《VBA经典应用69例》&#xff08;版权10178981&#xff09;&#xff0c;是我推出的第九套教程&#xff0c;教程是专门针对初级、中级学员在学习VBA过程中可能遇到的案例展开&#xff0c;这套教程案例众多&#xff0c;紧贴“实战”&#xff0c;并做“战术总结”&#xff0c;以便…

白银现货怎么开户,需要投入多少钱?

在白银投资市场中&#xff0c;有一种交易产品种类&#xff0c;它能够提供给人们稳定的理财环境&#xff0c;还具有辨别的交易环境。这种白银理财产品就是现货白银&#xff0c;也有一部分投资者将其称为白银现货。为了人们对这一产品具有更加清晰的认识&#xff0c;今天万洲金业…

赛迪生电源充电模块维修CHR-22005 RCU-202A

通信电源维修品牌&#xff1a;英可瑞,许继,艾默生,通合,动力源,九洲,华隆,合欣,泰坦,赛迪生等 直流屏模块故障和解决办法&#xff1a; 1、针对各类变电站直流屏&#xff0c;若显示交流空开跳闸&#xff0c;但并没有动作。应当检查三处地方是否正确&#xff1a; 接线是否正确…

CMS垃圾收集器深入解析

1.CMS的两种模式与一种特殊策略 1.1Backgroud CMS(没有并发失败的情况) 1.1.1并发标记还能被整理成两个流程 (1)初始标记 (2)并发标记(3)(4)在这个阶段发生(3)并发预处理 (4)可中止的预处理 (5)重新标记 (6)并发清除 1.1.2为什么我们的并发标记细化之后还会额外有两个流…

Colmap三维重建详解与使用方法

图片捕获过程&#xff0c;请遵循以下指导方针以获得最佳重建结果: 1 捕捉具有良好纹理的图像。避免使用完全没有质感的图片(例如&#xff0c;白色的墙或空桌子)。如果场景本身没有足够的纹理&#xff0c;你可以放置额外的背景对象&#xff0c;如海报等。 2 在相似的照明条件下…

Docker 快速搭建 Gitlab 服务

linux环境&#xff1a; 使用 vim 编辑 hosts 文件&#xff1a; vim /etc/hosts按 I 进入编辑模式&#xff0c;在文件末行追加上虚拟机的 IP 和要设置的域名&#xff1a; 192.168.1.17 gitlab.kunwu.toplwindows环境&#xff1a; Windows 系统的 hosts 文件位于 C:\Windows\S…

如何做一个简单的深度集成学习框架

使用同一个框架&#xff0c;独立在一个数据集上面&#xff0c;分别训练多次&#xff0c;每个单独模型训练超参数可以一样&#xff0c;也可以不一样&#xff0c;最后若干个训练好的独立模型在测试集数据上面做最后集中决策。 实例代码如下&#xff1a; class MyEnsemble(nn.Modu…

linux通过串口传输文件

简介 在嵌入式调试过程中&#xff0c;我们经常会使用调试串口来查看Log或者执行指令&#xff0c;其实&#xff0c;调试串口还有另一种功能&#xff0c;就是传输文件&#xff0c;本文说明使用MobaXterm串口工具来传输文件。 环境要求 嵌入式系统需要安装lsz和lrz&#xff0c;…

【Java】定时器的简单应用

在写代码的过程中&#xff0c;如果我们遇到了隔一段时间就要进行一项任务时&#xff0c;采用定时器会提高我们的效率。下面对定时器的使用进行简单说明 1、应用说明 首先我们要创建一个Timer类 Timer timer new Timer(); 然后在timer中调用schedule()方法添加任务 timer.…

将对象转成URL参数

背景 有的时候前端跳转到其他平台的页面需要携带额外的参数&#xff0c;需要将对象转成用 & 连接的字符串拼接在路径后面。 实现方法