QT MD4 MD5 Sha1等几种加密方式

news2024/10/7 8:25:57

QT MD4 MD5 Sha1等几种加密方式

  • [1] QT MD4 MD5 Sha1等几种加密方式
  • [2] qt MD5 和AES 加密
    • 一 、MD5 加密
    • 二、AES 加密和解密
  • [3] QT中sqlite数据库数据加密/混淆---MD5/SHA1/SHA2/SHA3
    • (1)创建一个加密对象
    • (2)放入要加密的数据
    • (3)获取加密的结果
    • 测试
    • 结果显示
  • [4] [Qt学习笔记]Qt程序加密,实现软件运行次数和硬件信息绑定
    • 1、思路分析
    • 2、实现函数
    • 3、示例代码
    • 4、效果展示
  • [5] Qt--探讨.exe程序加密
    • 一、前言
    • 二、分析
    • 三、实现(核心代码)
    • 四、效果
    • 附:MD5加密注册码
  • [6] Qt之Base64编解码
    • 编码
    • 解码
    • CTK Plugin Framework插件框架学习1--Qt编译CTK


[1] QT MD4 MD5 Sha1等几种加密方式

原文链接:https://blog.csdn.net/xiezhongyuan07/article/details/80594108?spm=1001.2101.3001.6650.2&utm_medium=distribute.pc_relevant.none-task-blog-2%7Edefault%7ECTRLIST%7ERate-2-80594108-blog-121167585.235%5Ev35%5Epc_relevant_increate_t0_download_v2&depth_1-utm_source=distribute.pc_relevant.none-task-blog-2%7Edefault%7ECTRLIST%7ERate-2-80594108-blog-121167585.235%5Ev35%5Epc_relevant_increate_t0_download_v2&utm_relevant_index=3

QT 自带有QCryptographicHash类可以进行很方便的进行 md4 、md5等加密。使用方法很简单,直接提供了一个静态的方法。

QByteArray QCryptographicHash::hash(const QByteArray &data, Algorithm method)

data就是要加密的数据, method是加密的方法,看官方文档选择加密的方式就行。
在这里插入图片描述

QString strCipher = QCryptographicHash::hash(strKey.toLocal8Bit(), QCryptographicHash::Md5);
QString MainWindow::fileMd5(const QString &sourceFilePath) {
 
    QFile sourceFile(sourceFilePath);
    qint64 fileSize = sourceFile.size();
    const qint64 bufferSize = 10240;
 
    if (sourceFile.open(QIODevice::ReadOnly)) {
        char buffer[bufferSize];
        int bytesRead;
        int readSize = qMin(fileSize, bufferSize);
 
        QCryptographicHash hash(QCryptographicHash::Md5);
 
        while (readSize > 0 && (bytesRead = sourceFile.read(buffer, readSize)) > 0) {
            fileSize -= bytesRead;
            hash.addData(buffer, bytesRead);
            readSize = qMin(fileSize, bufferSize);
        }
 
        sourceFile.close();
        return QString(hash.result().toHex());
    }
    return QString();
}

[2] qt MD5 和AES 加密

一 、MD5 加密

#include <QCryptographicHash>
 
 
// MD5 加密
QString Dialog_login::MD5_encryption(const QString &data)
{
    QCryptographicHash md5(QCryptographicHash::Md5);    //使用MD5加密
    md5.addData(data.toUtf8(),data.size());   // 添加数据
    QByteArray bArry = md5.result();    //获取MD5加密后的密码
    QString md5_date;
    md5_data.append(bArry.toHex());
 
    return md5_data;
}

二、AES 加密和解密

https://blog.csdn.net/chen1231985111/article/details/125748534

1. 下载库:https://github.com/bricke/Qt-AES

下载后需要下面5个文件,把这些文件复制出来放到一个文件夹下:
在这里插入图片描述
2. qt 工程添加这些文件
在这里插入图片描述
3. 编写加密解密代码

#include "aesni/qaesencryption.h"
 
 
// 加密
QString Dialog_login::AES_encryption(const QString &data, const QString &key)
{
    QAESEncryption encryption(QAESEncryption::AES_128, QAESEncryption::ECB, QAESEncryption::PKCS7);
    QByteArray enBA = encryption.encode(data.toUtf8(), key.toUtf8());
    return QString::fromLatin1(enBA.toBase64());
}
 
// 解密
QString Dialog_login::AES_decryption(const QString &data, const QString &key)
{
    QAESEncryption encryption(QAESEncryption::AES_128, QAESEncryption::ECB, QAESEncryption::PKCS7);
    QByteArray  enBA = QByteArray::fromBase64(data.toUtf8());
    QByteArray deBA = encryption.decode(enBA, key.toUtf8());
    return QString::fromLatin1(QAESEncryption::RemovePadding(deBA, QAESEncryption::PKCS7));
}

[3] QT中sqlite数据库数据加密/混淆—MD5/SHA1/SHA2/SHA3

原文链接:https://blog.csdn.net/qq_43533553/article/details/123416880

利用分解质因数的运算(质数取模)原理实现的,比如:给你100个数让你乘起来比较容易但是给你一个数让你求它100个质数就会比较难,所以数据加密就用这种方法来实现。
MD5是一种将任意长度的数据进行变换得到128位2进制数据,无法通过128位数据求的原始数据。
在QT中使用:

(1)创建一个加密对象

QCryptographicHash hash(QCryptographicHash::Md5);

(2)放入要加密的数据

void QCryptographicHash::addData(const QByteArray &data);
//hash.addData(password.toUft8());

(3)获取加密的结果

QByteArray QCryptographicHash::result() const;
//hash.result();

测试

//对密码进行MD5加密
//1.创建加密对象
 QCryptographicHash hash(QCryptographicHash::Md5);
 //2.放入要加入的数据
 hash.addData(password.toUtf8());
 //3.获取加密的结果
 QByteArray  arr = hash.result();
 password = arr.toHex();
 qDebug()<<password; 

结果显示

在这里插入图片描述

[4] [Qt学习笔记]Qt程序加密,实现软件运行次数和硬件信息绑定

原文链接:https://blog.csdn.net/fengyaowuhui/article/details/126150459

前言
程序员为了防止自己的成果被白嫖,或者发布的软件在公司层面上要做一些防抄袭处理,这时就需要在软件层面上加锁、加密等操作。
1、单机终端软件
这类软件一般在未联网的情况下使用,所以不能通过网络去判定是否得到授权,一般采用绑定硬件信息来对软件进行加密,这样软件与设备绑定就无法进行随意使用。
2、远程授权监控
终端软件在启动后就会跟服务器通讯来检查当前设备是否已经得到授权,可实现远程锁定程序,设置程序的使用时间和使用次数。
以上两种方法都是最基础的软件加密方法,根据加密的算法的复杂度其破解的难度也不同,不过作为普通的一种加密方法已经够用了,市面上有比较成熟的加密加壳的软件,其安全程度要远高于软件加密的方法,下次抽时间介绍一下软件VMProtect的加、解密过程。

1、思路分析

单机软件想要每次启动前去验证是否有授权,需要从软件的配置中获取授权的验证信息,这里以软件的使用次数来举例,我们需要把软件的可使用次数写入配置文件ini或数据库中。每次在软件重新启动时,我们根据配置文件中信息来判断软件剩余的使用次数。
在这里插入图片描述

这里设定软件的剩余使用次数为2次,不过这里有一个很明显的缺点,就是明文写到配置文件里,就可以自己修改,然后就失去了加密的属性,这里就需要对这个信息进行加密后再写入。这里使用Qt自带的base64对字符串进行加密。

2、实现函数

//加密
QString Widget::Encode(QString row)
{
    QByteArray byteArray = row.toUtf8();
    byteArray = byteArray.toBase64();
    return  byteArray;
}
//解密
QString Widget::Decode(QString passwd)
{
    QByteArray byteArray = passwd.toUtf8();
    byteArray = QByteArray::fromBase64(byteArray);
    return byteArray;
}
//从配置文件获取信息
QString Widget::getInfoFromIni(QString str)
{
    QString info;
    QFile file(str);
    if(!file.exists()) { //如果文件不存在
        QString times_str = "RemainTime:"+time;
        WriteInfo2Ini(str, times_str);
    }
    file.open(QFile::ReadWrite | QFile::Text);
    info = file.readAll(); //读取信息
    file.close();
    return info;
}
//将信息写入到配置文件
void Widget::WriteInfo2Ini(QString str, QString info_text)
{
    QFile file(str);
    file.open(QFile::ReadWrite | QFile::Text | QFile::Truncate);
    file.write(Encode(info_text).toUtf8()); //写入信息
    file.close();
}

这种方法有一种弊端,就是如果重新覆盖配置文件可以跳出限制,比如这里如果事前拷贝一份ini文件,当软件次数为0后,重新拷贝原始ini文件后又获得初始的软件使用次数。
如果考虑不使用ini文件来记录授权信息,而是使用注册表来记录授权信息,这样软件使用者就不容易去发现授权信息的位置

void Widget::WriteInfo2Registry(QString str)
{
    //写入注册表
    QSettings settings("HKEY_CURRENT_USER\\Software\\Code_Encryption\\Settings",QSettings::NativeFormat);
    settings.setValue("remain_times",Encode(str).toUtf8());
}

QString Widget::getInfoFromRegistry()
{
    QString info;
    //通过写入注册表来判断
    QSettings settings("HKEY_CURRENT_USER\\Software\\Code_Encryption\\Settings",QSettings::NativeFormat);
    info = settings.value("remain_times").toString();
    return info;
}

3、示例代码

widget.cpp

#pragma execution_character_set("utf-8")
#include "widget.h"
#include "ui_widget.h"

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);
    QString str = getCpuId();
    qDebug()<<"cpu id is"<<str;
//    QString times_str = "RemainTime:"+time;
//    WriteInfo2Registry(times_str);
    Judge_Authorize_times(1);
}

Widget::~Widget()
{
    delete ui;
}

QString Widget::getWMIC(const QString &cmd)
{
    //获取cpu名称:wmic cpu get Name
    //获取cpu核心数:wmic cpu get NumberOfCores
    //获取cpu线程数:wmic cpu get NumberOfLogicalProcessors
    //查询cpu序列号:wmic cpu get processorid
    //查询主板序列号:wmic baseboard get serialnumber
    //查询BIOS序列号:wmic bios get serialnumber
    //查看硬盘:wmic diskdrive get serialnumber
    QProcess p;
    p.start(cmd);
    p.waitForFinished();
    QString result = QString::fromLocal8Bit(p.readAllStandardOutput());
    QStringList list = cmd.split(" ");
    result = result.remove(list.last(), Qt::CaseInsensitive);
    result = result.replace("\r", "");
    result = result.replace("\n", "");
    result = result.simplified();
    return result;
}

QString Widget::getCpuId() //获取CPU序列号
{
    return getWMIC("wmic cpu get processorid");
}

QString Widget::getDiskNum() //获取硬盘序列号
{
    return getWMIC("wmic diskdrive where index=0 get serialnumber");
}

QString Widget::Encode(QString row)
{
    QByteArray byteArray = row.toUtf8();
    byteArray = byteArray.toBase64();
    return  byteArray;
}
QString Widget::Decode(QString passwd)
{
    QByteArray byteArray = passwd.toUtf8();
    byteArray = QByteArray::fromBase64(byteArray);
    return byteArray;
}

QString Widget::getInfoFromIni(QString str)
{
    QString info;
    QFile file(str);
    if(!file.exists()) { //如果文件不存在
        QString times_str = "RemainTime:"+time;
        WriteInfo2Ini(str, times_str);
    }
    file.open(QFile::ReadWrite | QFile::Text);
    info = file.readAll(); //读取信息
    file.close();
    return info;
}

void Widget::WriteInfo2Ini(QString str, QString info_text)
{
    QFile file(str);
    file.open(QFile::ReadWrite | QFile::Text | QFile::Truncate);
    file.write(Encode(info_text).toUtf8()); //写入信息
    file.close();
}

void Widget::WriteInfo2Registry(QString str)
{
    //写入注册表
    QSettings settings("HKEY_CURRENT_USER\\Software\\Code_Encryption\\Settings",QSettings::NativeFormat);
    settings.setValue("remain_times",Encode(str).toUtf8());
}

QString Widget::getInfoFromRegistry()
{
    QString info;
    //通过写入注册表来判断
    QSettings settings("HKEY_CURRENT_USER\\Software\\Code_Encryption\\Settings",QSettings::NativeFormat);
    info = settings.value("remain_times").toString();
    return info;
}

void Widget::Judge_Authorize_times(int type)
{
    QString times_info;
    if(type == 0)
    {
        //通过ini配置文件进行判断
        times_info = getInfoFromIni("System.ini");
    }
    else
    {
        //通过写入注册表来判断
        times_info = getInfoFromRegistry();
    }

    if(times_info.isEmpty())
    {
        QMessageBox::about(this,"提示","授权信息为空,请检查!");
        exit(0);
    }
    else
    {
        QString info = Decode(times_info); //解码
        if(info.contains(':')) {	//信息正确
            QStringList list = info.split(':');
            if(list.length()>=1) {
                if(list[1].toInt() <= 0) {	//程序剩余使用次数不足
                    QMessageBox::about(this,"提示","请注册后再使用!");
                    exit(0); //程序退出
                }else {	//程序还有剩余使用次数
                    QString time_remain = QString::number(list[1].toInt()-1); //程序剩余使用次数减1
                    QString str = "RemainTime:"+time_remain;
                    if(type == 0) WriteInfo2Ini("System.ini", str);
                    else {
                        //写入注册表
                        WriteInfo2Registry(str);
                    }

                    QMessageBox::about(this,"提示","程序剩余使用次数:"+time_remain+"次");
                }
            }
        }

    }

}

widget.h

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QFile>
#include <QMessageBox>
#include <QProcess>
#include <QDebug>
#include <QSettings>

namespace Ui {
class Widget;
}

class Widget : public QWidget
{
    Q_OBJECT

public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();

private:
    Ui::Widget *ui;

    QString getWMIC(const QString &cmd);
    QString getCpuId();
    QString getDiskNum();
    //加密
    QString Encode(QString row);
    //解密
    QString Decode(QString passwd);
    //判断授权信息  0:使用ini方法判断;1:使用注册表方法判断
    void Judge_Authorize_times(int type);
    //从ini获取授权信息
    QString getInfoFromIni(QString str);
    //写入授权信息到ini
    void WriteInfo2Ini(QString str, QString info_text);
    //从注册表获取授权信息
    QString getInfoFromRegistry();
    //写入授权信息到注册表
    void WriteInfo2Registry(QString str);
    //授权程序使用次数为2次
    QString time = "2";

};

#endif // WIDGET_H

4、效果展示

在这里插入图片描述

[5] Qt–探讨.exe程序加密

原文链接:https://blog.csdn.net/Mr_robot_strange/article/details/108776517

一、前言

程序员随着代码写的多,难免会接到一些私活。但是又怕对方白嫖,困难总比办法多,哦不,错了,办法总比困难多。为了避免被别人白嫖,我们可以给自己的程序留后手,网上查阅资料,大概有以下两种:

1、采用远程控制:UDP指令远程锁死程序;

2、采用本地加密:提取CPU和硬盘序列号,加密生成注册码;

以上两种方法各有利弊,方法1可以本人自由控制,但是一旦设备没网,就GG了;方法2则不依赖网络,且每台设备的注册码唯一,但是本人无法直接控制。个人认为还是方法2简单有效,所以下面就方法2进行具体分析。

二、分析

采用注册码机制,我们首先要生成注册码,Qt下获取CPU和硬盘等系统信息(针对windows系统)的方法如下:

windows下执行命令除了用cmd之外,还有个东西叫WMIC,非常强大,可以通过他获取很多信息,包括硬件信息。

QString Widget::getWMIC(const QString &cmd)
{
    //获取cpu名称:wmic cpu get Name
    //获取cpu核心数:wmic cpu get NumberOfCores
    //获取cpu线程数:wmic cpu get NumberOfLogicalProcessors
    //查询cpu序列号:wmic cpu get processorid
    //查询主板序列号:wmic baseboard get serialnumber
    //查询BIOS序列号:wmic bios get serialnumber
    //查看硬盘:wmic diskdrive get serialnumber
    QProcess p;
    p.start(cmd);
    p.waitForFinished();
    QString result = QString::fromLocal8Bit(p.readAllStandardOutput());
    QStringList list = cmd.split(" ");
    result = result.remove(list.last(), Qt::CaseInsensitive);
    result = result.replace("\r", "");
    result = result.replace("\n", "");
    result = result.simplified();
    return result;
}
 
QString Widget::getCpuName()
{
    return getWMIC("wmic cpu get name");
}
 
QString Widget::getCpuId()
{
    return getWMIC("wmic cpu get processorid");
}
 
QString Widget::getDiskNum()
{
    return getWMIC("wmic diskdrive where index=0 get serialnumber");
}

获取注册码之后,就可以着手程序的加密了,我的策略是这样的:

如果程序没有注册,默认可以使用3次;
如果程序注册了,使用次数无限制;
那怎么限制程序的使用次数呢?我的策略是这样的:

生成一个System.ini配置文件,里面存放程序的剩余运行次数RemainTime:3;
但是RemainTime:3当然不能直接存在System.ini中,不然一眼看出来就可以自己修改次数了;
所以在对System.ini写之前,进行加密;对System.ini读之后,进行解密;
加/解密算法可以自行选取,但是要采用可逆算法;
然后判断剩余运行次数,如果小于等于0,则程序提示需注册后使用,并退出;
否则,剩余运行次数减1,程序正常运行;

三、实现(核心代码)

QString Widget::getWMIC(const QString &cmd)
{
    //获取cpu名称:wmic cpu get Name
    //获取cpu核心数:wmic cpu get NumberOfCores
    //获取cpu线程数:wmic cpu get NumberOfLogicalProcessors
    //查询cpu序列号:wmic cpu get processorid
    //查询主板序列号:wmic baseboard get serialnumber
    //查询BIOS序列号:wmic bios get serialnumber
    //查看硬盘:wmic diskdrive get serialnumber
    QProcess p;
    p.start(cmd);
    p.waitForFinished();
    QString result = QString::fromLocal8Bit(p.readAllStandardOutput());
    QStringList list = cmd.split(" ");
    result = result.remove(list.last(), Qt::CaseInsensitive);
    result = result.replace("\r", "");
    result = result.replace("\n", "");
    result = result.simplified();
    return result;
}

QString Widget::getCpuId() //获取CPU序列号
{
    return getWMIC("wmic cpu get processorid");
}

QString Widget::getDiskNum() //获取硬盘序列号
{
    return getWMIC("wmic diskdrive where index=0 get serialnumber");
}

QString Widget::Encode(QString row) //加密,由于加/解密不是本文重点,故简单处理一下
{
    QByteArray byteArray = row.toUtf8();
    byteArray = byteArray.toBase64();
    return  byteArray;
}
QString Widget::Decode(QString passwd) //解密
{
    QByteArray byteArray = passwd.toUtf8();
    byteArray = QByteArray::fromBase64(byteArray);
    return byteArray;
}

void Widget::Judge_Valid() //判断程序合法性
{
    QString filePath = "System.ini";
    QFile file(filePath);

    if(!file.exists()) { //如果文件不存在
        file.open(QIODevice::ReadWrite | QIODevice::Text); //新建文件
        QString info = "RemainTime:"+time; //time是全局变量,QString time = "3";
        file.write(Encode(info).toUtf8()); //写入信息
        file.close();
         goto Here;
    }else {	//如果文件存在
        Here:if(file.open(QFile::ReadWrite | QFile::Text)) { //打开文件
            QString info = file.readAll(); //读取信息
            if(info.isEmpty()) { //如果文件为空
                QString time_temp = "1";
                QString info = "RemainTime:"+time_temp;
                file.write(Encode(info).toUtf8()); //写入信息
                info = file.readAll();
            }
            info = Decode(info); //解码
            if(info.contains(':')) {	//信息正确
                QStringList list = info.split(':');
                if(list.length()>=1) {
                    if(list[1].toInt() <= 0) {	//程序剩余使用次数不足
                        QMessageBox::about(this,"提示","请注册后再使用!");
                        exit(0); //程序退出
                    }else {	//程序还有剩余使用次数
                        file.close();
                        file.open(QFile::ReadWrite | QFile::Text | QFile::Truncate); //清空文件内容
                        QString time_remain = QString::number(list[1].toInt()-1); //程序剩余使用次数减1
                        QString info = "RemainTime:"+time_remain;
                        file.write(Encode(info).toUtf8()); //更新程序剩余使用次数
                        QMessageBox::about(this,"提示","程序剩余使用次数:"+time_remain+"次");
                    }
                }
            }
      }//Here
      file.close();

   }//如果文件存在
}

四、效果

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

附:MD5加密注册码

//获取设备信息字符串
const QString get_SerialNumber()
{
    QString cpu_id = "";
    QProcess p(0);
    p.start("wmic CPU get ProcessorID");  //其它信息也类似
    p.waitForStarted();
    p.waitForFinished();
    cpu_id = QString::fromLocal8Bit(p.readAllStandardOutput());
    cpu_id = cpu_id.remove("ProcessorId").trimmed();

    QString lpRootPathName = "C:\\";
    LPTSTR lpVolumeNameBuffer=new TCHAR[12];//磁盘卷标
    DWORD nVolumeNameSize=12;// 卷标的字符串长度
    DWORD VolumeSerialNumber;//硬盘序列号
    DWORD MaximumComponentLength;// 最大的文件长度
    LPTSTR lpFileSystemNameBuffer=new TCHAR[10];// 存储所在盘符的分区类型的长指针变量
    DWORD nFileSystemNameSize=10;// 分区类型的长指针变量所指向的字符串长度
    DWORD FileSystemFlags;// 文件系统的一此标志
    GetVolumeInformation((LPTSTR)lpRootPathName.utf16(),
                         lpVolumeNameBuffer, nVolumeNameSize,
                         &VolumeSerialNumber, &MaximumComponentLength,
                         &FileSystemFlags,
                         lpFileSystemNameBuffer, nFileSystemNameSize);
    return (cpu_id.mid(0,4) + "D-"+ cpu_id.mid(4,4) + "R-" +
            cpu_id.mid(8,4) + "E-" + cpu_id.mid(12,4) + "A-" +
            QString::number(VolumeSerialNumber,10).mid(0,4)+"M");
}

//使用MD5对设备信息字符串进行加密
const QString hash_Encryption(const QString temp)
{
    QByteArray byte_array;
    byte_array.append(temp);
    QByteArray hash_byte_array = QCryptographicHash::hash(byte_array,QCryptographicHash::Md5);
    return hash_byte_array.toHex().toUpper();
}

//将加密后的字符串序列化为注册码形式
const QString format_HASHString(const QString hashtemp)
{
    QString retemp = "";
    for(int i = 0; i < 7; i++)
    {
        retemp += hashtemp.mid(4*i,4) + "-";
    }
    retemp += hashtemp.mid(28,4);
    return retemp;
}

//去除注册码格式,转化为MD5加密后的字符串
const QString remove_Format(const QString formathash)
{
    QString temp = "";
    for(int i = 0; i < 8; i++)
    {
        temp += formathash.mid(5*i,4);
    }
    return temp;
}

[6] Qt之Base64编解码

原文链接

编码

QByteArray QByteArray::toBase64() const
QByteArray QByteArray::toBase64(QByteArray::Base64Options options) const

示例

QByteArray text("Hello world");
text.toBase64(); 
/* 输出: SGVsbG8gd29ybGQ= */

解码

[static] QByteArray QByteArray::fromBase64(const QByteArray &base64)
[static] QByteArray QByteArray::fromBase64(const QByteArray &base64, 
                                           QByteArray::Base64Options options)

示例

QByteArray::fromBase64("SGVsbG8gd29ybGQ="); 
/* 输出: Hello world */

在这里插入图片描述

CTK Plugin Framework插件框架学习1–Qt编译CTK

https://wangjichuan.blog.csdn.net/article/details/128547331?spm=1001.2014.3001.5502

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

嘉立创EDA PDF文件的输出教程

在PCB生产调试期间&#xff0c;为了方便查看文件或者查询相关元件信息&#xff0c;会把PCB设计文件转换成PDF文件。下面介绍常规PDF文件的输出方式。 前期工作是需要在电脑上安装PDF阅读器&#xff0c;准备充足后按照以下步骤进行操作。 1&#xff09;执行菜单命令“导出-PDF…

企业财务管理为何需要数字化转型?

为什么企业财务管理需要数字化转型&#xff1f; 许多企业在推动各大业务部门进行数字化转型时&#xff0c;往往会忽略财务部门。然而&#xff0c;作为掌握公司核心资源与数据和推动企业数字化建设的部门&#xff0c;财务也应成为企业数字化转型的重要突破口。 这篇就用几个案…

用Leangoo领歌Scrum敏捷开发工具管理产品路线图?

那我们来看下&#xff0c;如何利用Leangoo管理产品路线图&#xff1f; 首先什么是产品路线图&#xff1f; 产品路线图是一个高层次的战略计划&#xff0c;它描述了产品在未来一段时间可能会如何发展和壮大。 产品路线图确保整个产品团队持续关注产品的目标&#xff0c;帮助产…

07-微服务部署2023系列-centos+docker部署nacos

1、创建nacos的数据库配置 1.1、创建数据库 1.2、创建用户nacos并授权数据库 参考 06 mysql 创建账户部分内容 1.3、执行数据库脚本 导入官方nacos sql语句。如果是历史有历史数据&#xff0c; 从历史数据sql导入 2、创建 bridge 网络并指定 IP 区间 2.1、创建自定义网络…

第十章 使用Samba或NFS实现共享文件

文章目录 第十章 使用Samba或NFS实现共享文件一、SAMBA文件共享服务1、SAMBA简介2、相关配置&#xff08;1&#xff09;、安装软件&#xff08;2&#xff09;、删除多余配置文件&#xff08;3&#xff09;、Samba服务验证方式 3、配置共享资源&#xff08;1&#xff09;、用于设…

认识 AbortController控制器对象 及其应用

参考文章1 参考文章2 一、什么是AbortController (abort 意为 中止/废弃) AbortController是一个控制器对象&#xff08;DOM API&#xff09;&#xff0c;可通过new构造函数的方式&#xff0c;生成控制器实例对象&#xff0c;根据需要终止/取消一个或多个Web请求/监听事件 通…

微信开发者工具突然打不开问题解决

今天微信小程序开发者工具 好好的在电脑里 突然就打不开了 这个问题已经上百度了 想必并不是很少遇到 可能是版本太旧 或者 其中依赖文件丢失 这个基本不用抱什么幻想 还是得重新装 先将快捷打开方式删掉 然后找到工具所在目录 把它删了 然后访问如下地址 https://mp.weixi…

日期时间选择器el-date-picker,限制可选范围,以后端接口某个时间字段为时间节点

哈喽 大家好啊 今天需要做一个时间选择器的限制&#xff0c;不然用户选择某个时间节点之前的时间 比如用户选择发货时间不允许早于收货时间&#xff08;来自后端接口&#xff09; picker-options当前时间日期选择器特有的选项参考下表object 首先申明一个时间可选对象 因为我…

黑马程序员-学成在线项目总结

黑马程序员-学成在线项目总结 收获 基础公共样式 清除默认样式&#xff0c;例如内边距、外边距、项目符号等等 设置通用样式&#xff0c;例如&#xff1a;文字样式 项目结构 注:多个css文件引入顺序&#xff0c;先清除&#xff0c;后设置 版心效果 许多网页整体都有版心居…

Java框架学习01(Spring框架)

1、什么是Spring框架&#xff1f; Spring 是一款开源的轻量级 Java 开发框架&#xff0c;旨在提高开发人员的开发效率以及系统的可维护性。 我们一般说 Spring 框架指的都是 Spring Framework&#xff0c;它是很多模块的集合&#xff0c;使用这些模块可以很方便地协助我们进行…

网络漏洞管理

网络漏洞是硬件、软件或流程中的漏洞或缺陷&#xff0c;可能威胁到网络的正常运行。漏洞会使您的业务和客户的敏感数据面临风险&#xff0c;导致黑客轻松进入、销售额下降、声誉损失和处罚。网络漏洞可以分为硬件、固件、软件和人类漏洞。如果这些实体中的任何一个没有得到适当…

关于Kerberos认证的一些攻击手法学习总结

Kerberos认证流程 前言 本文主要分享最近学习的关于域内Kerberos认证的一些攻击手法&#xff0c;以自我的理解为主&#xff0c;从原理理解切入到基本工具利用来阐述&#xff0c;个人的理解分析较为啰嗦&#xff0c;嫌太兀长的可以跳着看就好&#xff0c;还请各位谅解。如有错误…

唐毅:带领和数集团,做好科技成果与创新需求的“摆渡人”

近年来&#xff0c;人类已经进入有史以来科技创新爆发最密集最迅猛的大时代。 作为人工智能、区块链、交互技术、游戏引擎及数字孪生技术、综合智能网络、物联网等若干前沿科技领域陆续高速发展又碰撞聚变后形成的元宇宙&#xff0c;成为最具引领性的力量。在人工智能、物联网…

网安笔记06 数字签名基本概念

数字签名基本概念 R1:receiver确认、证实sender的签名&#xff0c;这个签名不能被伪造S:sender发送出签名的教习给receiver&#xff0c;不能否认他签发的消息R2:receiver堆收到的签名消息不能否认&#xff0c;收报认证T:第三方可以确认手法收发双方之间的消息传输&#xff0c;…

Django框架之视图HttpRequest对象

本文主要是记录视图的HttpRequest对象属性、方法及示例。 概述 服务器接受http请求后&#xff0c;会根据报文创建httpRequest对象&#xff0c;包含所有请求中必须的数据&#xff1b; 视图方法的第一个参数就是HttpRequest对象&#xff1b;Django创建对象后&#xff0c;调用视…

本地提权漏洞分析【网络安全】

0. 前言 CVE-2023-21752 是 2023 年开年微软第一个有 exploit 的漏洞&#xff0c;原本以为有利用代码会很好分析&#xff0c;但是结果花费了很长时间&#xff0c;难点主要了两个&#xff1a;漏洞点定位和漏洞利用代码分析&#xff0c;欢迎指正。 1. 漏洞简介 根据官方信息&a…

chatgpt中文意思——预训练生成聊天模型

"ChatGPT"相关的英文表达 ChatGPT 这个名字由两部分组成&#xff1a;Chat 和 GPT。 Chat 是聊天&#xff0c;而GPT是Generative Pretrained Transformer的缩写&#xff1a;预训练生成模型。因此&#xff0c;ChatGPT的意思是预训练生成聊天模型。 简单来说就是一个预先…

第五十一章 Unity Input Manager 输入系统(下)

本章节我们就来使用水平轴和垂直轴来控制游戏物体的移动和旋转。我们之前大致讲过&#xff0c;游戏物体移动最重要的是方向&#xff0c;速度和时间三个要素&#xff0c;同样旋转也是。接下来&#xff0c;我们将之前创建的地形场景导入进来&#xff0c;如下所示 然后将之前的“M…

OpenAI又火一个新项目,已开源...

大家好&#xff0c;我是 Jack。 OpenAI 又有新动作了&#xff0c;开源发布 Shap-E。 今天&#xff0c;我继续手把手教学。 算法原理、环境搭建、效果测试&#xff0c;一条龙服务&#xff0c;尽在下文&#xff01; 一、Shap-E 效果 Shap-E 算法的功能&#xff0c;简单来讲就…

第五十二章 Unity Input System 新输入系统

新输入系统InputSystem是2019年Unity新推出的插件。请注意&#xff0c;Unity默认使用旧的Input Manager&#xff0c;新的Input System处于未启用状态。当你安装Input System组件时&#xff0c;Unity会询问你是否启用新的输入系统。如果你选择Yes&#xff0c;Unity会启用新的并禁…