25、Qt设备识别(简单的密钥生成器)

一、说明

在很多商业软件中,需要提供一些可以试运行的版本,这样就需要配套密钥机制来控制,纵观大部分的试用版软件,基本上采用以下几种机制来控制。

1、远程联网激活,每次启动都联网查看使用时间等,这种方法最完美,缺点是没法联网的设备就歇菜了。

2、通过获取本地的硬盘、CPU等硬件的编号,做一个运算,生成一个激活码,超过半数的软件会采用此方法,缺点是不能自由控制软件的其他参数,比如软件中添加的设备数量的控制。

3、设定一个运行到期时间+数量限制+已运行时间的密钥文件,发给用户配套软件使用,缺点是如果仅仅设置的是运行到期时间,用户可以更改电脑时间来获取更长的使用时间,在电脑不联网的情况下。

注:本文实现第2种方式;

二、功能描述

1、软件A(KeyDemo)首次运行弹出输入注册码(密钥)对话框,点击获取机器码,把机器码(系统的UUID)发送给商家;

2、商家使用密钥生成器(Key)生成注册码(密钥),给软件A;

3、软件A输入注册码,点击确认,软件正常启动;

4、第二次运行软件A时,若还是同一台电脑,则直接正常运行;若不是同一台电脑则弹出输入注册码对话框;

三、查看windows系统的UUID

1、win+r,打开命令提示符,输入wmic,点击确定

2、输入csproduct,回车(enter),UUID即为一会要用到的机器码

3、运行wmic错误提示

注:若cmd运行wmic出现如下提示

‘wmic‘ 不是内部或外部命令,也不是可运行的程序或批处理文件

或者提示:

Windows 找不到文件’wmic’,请确定文件名是否正确,再试一次

解决办法

win+r,打开命令提示符,输入sysdm.cpl,点击确定

选择 高级 ”->“ 环境变量

系统变量 ”->“Path”-> 编辑 

新建 ,输入 “C:\Windows\System32\wbem” ,点击 确定 ,之后就可以正常运行 wmic

四、Qt使用AES加密解密

访问下面的链接,下载Qt-AES相关文件,会用到压缩包里的qaesencryption.cpp和qaesencryption.h两个文件

GitHub - bricke/Qt-AES: Native Qt AES encryption class

五、创建工程KeyDemo

1、新建Qt Widgets应用,名称为KeyDemo,基类选择QMainWindow;

2、添加新的Qt设计师界面类,名称为KeyDialog,基类选择QDialog,界面样式如下

3、添加新的C++类,名称为KeyVerify,基类为空

4、把下载好的qaesencryption.cpp和qaesencryption.h添加到工程中

更改KeyDialog.h代码

#ifndef KEYDIALOG_H
#define KEYDIALOG_H

#include <QDialog>

namespace Ui {
class KeyDialog;
}

class KeyDialog : public QDialog
{
    Q_OBJECT

public:
    explicit KeyDialog(QWidget *parent = nullptr);
    ~KeyDialog();

    static QString g_key;

private slots:
    void on_cancelButton_clicked();

    void on_okButton_clicked();

    void on_uuidButton_clicked();

private:
    Ui::KeyDialog *ui;
};

#endif // KEYDIALOG_H

更改KeyDialog.cpp代码

#include "KeyDialog.h"
#include "ui_KeyDialog.h"
#include <QProcess>
#include <QMessageBox>
#include <QFile>
QString KeyDialog::g_key = "";

KeyDialog::KeyDialog(QWidget *parent) :
QDialog(parent),
ui(new Ui::KeyDialog)
{
    ui->setupUi(this);
    ui->uuidEdit->setVisible(false);
    ui->tipsLabel->setVisible(false);
}

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

void KeyDialog::on_cancelButton_clicked()
{
    exit(0);
}

void KeyDialog::on_okButton_clicked()
{
    g_key = ui->keyEdit->text();
    this->close();
}

//获取UUID
void KeyDialog::on_uuidButton_clicked()
{
    QString cmd = "wmic csproduct get uuid";
    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();

    p.kill();
    p.close();

    ui->tipsLabel->setVisible(true);
    ui->uuidEdit->setVisible(true);
    ui->uuidEdit->setText(result);
}

更改KeyVerify.h代码

#ifndef KEYVERIFY_H
#define KEYVERIFY_H

#include <QObject>

class KeyVerify
{
public:
    KeyVerify();

private slots:
    void initSystem(); //初始化

    QString getUUID(); //获取唯一标识

    bool enterKey(); //输入密钥

    QString encodedText(QString, QString key = "zxcvbnm"); //加密

    QString decodedText(QString, QString key = "zxcvbnm"); //解密
};

#endif // KEYVERIFY_H

更改KeyVerify.cpp代码

#include "KeyVerify.h"
#include <QCryptographicHash>
#include <QDebug>
#include <QFile>
#include <QProcess>
#include <QMessageBox>
#include <QMutex>
#include <QApplication>
#include "qaesencryption.h"
#include "KeyDialog.h"

KeyVerify::KeyVerify()
{
    initSystem();
}

void KeyVerify::initSystem()
{
    QString keyName = qApp->applicationDirPath() + "/key.db";
    QFile keyFile(keyName);

    //密钥文件不存在
    if(!keyFile.exists() || keyFile.size() == 0)
    {
        //弹出输入密钥界面
        if(!enterKey()) //密钥不对
        {
        exit(0);
        }
    }
    else
    {
        //读取密钥文件
        keyFile.open(QFile::ReadOnly);
        QByteArray keyData = keyFile.readAll();
        keyFile.close();

        QString decodedStr = decodedText(QString::fromLatin1(keyData));

        QString uuid = getUUID();

        if(uuid != decodedStr) //密钥不对
        {
            int ret = QMessageBox::critical(nullptr, "错误", "密钥文件不对,请联系供应商!",             QMessageBox::Ok, QMessageBox::Cancel);
            if( ret == QMessageBox::Ok)
            {
                if(!enterKey()) //密钥不对
                {
                    keyFile.close();
                    exit(0);
                }
            }
            else
            {
                keyFile.close();
                exit(0);
            }
        }
        keyFile.close();
    }
}

//获取电脑UUID
QString KeyVerify::getUUID()
{
    QString cmd = "wmic csproduct get uuid";
    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();

    p.kill();
    p.close();

    return result;
}

//输入密钥
bool KeyVerify::enterKey()
{
    //弹出输入密钥界面
    KeyDialog keyDialog;
    keyDialog.exec();

    if(KeyDialog::g_key != "")
    {
        //解密
        QString decodedStr = decodedText(KeyDialog::g_key);
        //获取本地uuid
        QString uuid = getUUID();
        //对比
        if(uuid != decodedStr)
        {
            QMessageBox::critical(nullptr, "错误", "密钥文件不对,请联系供应商!");
            return false;
        }
        else
        {
            QFile keyFile(qApp->applicationDirPath() + "/key.db");
            //存储
            keyFile.open(QFile::WriteOnly);
            keyFile.write(KeyDialog::g_key.toLatin1());
            keyFile.close();
        }
    }
    else
    {
        QMessageBox::critical(nullptr, "错误", "密钥不能为空");
        return false;
    }
    return true;
}

//加密
QString KeyVerify::encodedText(QString data, QString key)
{
    QAESEncryption encryption(QAESEncryption::AES_128, QAESEncryption::ECB,     QAESEncryption::ZERO);
    QByteArray hashKey = QCryptographicHash::hash(key.toUtf8(), QCryptographicHash::Md5);
    QByteArray encodedText = encryption.encode(data.toUtf8(), hashKey);
    QString encodeTextStr = QString::fromLatin1(encodedText.toBase64());
    //qDebug()<< "encodedText:"<< encodeTextStr;

    return encodeTextStr;
}

//解密
QString KeyVerify::decodedText(QString data, QString key)
{
    QAESEncryption encryption(QAESEncryption::AES_128, QAESEncryption::ECB,     QAESEncryption::ZERO);
    QByteArray hashKey = QCryptographicHash::hash(key.toUtf8(), QCryptographicHash::Md5);
    QByteArray decodedText = encryption.decode(QByteArray::fromBase64(data.toLatin1()), hashKey);
    QString decodedTextStr = QString::fromLatin1(decodedText);
    //qDebug()<<"decodedText:"<< decodedTextStr;
    
    return decodedTextStr;
}

更改main.cpp代码

#include "MainWindow.h"
#include <QApplication>
#include "KeyVerify.h"

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);

    //密钥验证
    KeyVerify k;

    MainWindow w;
    w.show();

    return a.exec();
}

六、创建工程Key

1、新建Qt Widgets应用,名称为Key,基类选择QMainWindow,界面设计如下

2、把下载好的qaesencryption.cpp和qaesencryption.h添加到工程中

3、代码演示

更改MainWindow.h代码

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>

namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = nullptr);
    ~MainWindow();

private slots:
    void on_keyButton_clicked();

    QString encodedText(QString data, QString key);

private:
    Ui::MainWindow *ui;
};

#endif // MAINWINDOW_H

更改MainWindow.cpp代码

#include "MainWindow.h"
#include "ui_MainWindow.h"
#include <QMessageBox>
#include <QCryptographicHash>
#include <QDebug>
#include "qaesencryption.h"

MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MainWindow)
{
    ui->setupUi(this);
}

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

void MainWindow::on_keyButton_clicked()
{
    if(ui->uuidEdit->text().isEmpty())
    {
        QMessageBox::critical(this, tr("提示"), tr("机器码不能为空"));
        return;
    }

    if(ui->passwordEdit->text().isEmpty())
    {
        QMessageBox::critical(this, tr("提示"), tr("密码不能为空"));
        return;
    }

    //加密
    QString key = encodedText(ui->uuidEdit->text(), ui->passwordEdit->text());
    ui->keyEdit->setText(key);
    ui->keyEdit->setFocus();
}

//加密
QString MainWindow::encodedText(QString data, QString key)
{
    //使用QCryptographicHash(只能加密不能解密)类对密码进行加密
    QByteArray hashKey = QCryptographicHash::hash(key.toUtf8(), QCryptographicHash::Md5);

    //使用AES加密
    QAESEncryption encryption(QAESEncryption::AES_128, QAESEncryption::ECB,     QAESEncryption::ZERO);

    QByteArray encodedText = encryption.encode(data.toUtf8(), hashKey);
    QString encodeTextStr = QString::fromLatin1(encodedText.toBase64());
    //qDebug()<< "encodedText:"<< encodeTextStr << encodedText;

    return encodeTextStr;
}

七、运行测试

1、先运行KeyDemo,弹出输入密钥对话框 

2、点击“获取机器码”,把机器码给商家(软件Key) 

3、运行软件Key,输入机器码和密码,点击生成密钥,Ctrl+a、Ctrl+v复制密钥,把密钥发送给KeyDemo软件;

注:密码需要和KeyDemo工程中KeyVerify.h的“zxcvbnm”一样

QString encodedText(QString, QString key = "zxcvbnm"); //加密

QString decodedText(QString, QString key = "zxcvbnm"); //解密

4、KeyDemo软件输入密钥后,点击“确定”,软件正常运行,弹出主界面

5、KeyDemo软件若还在刚才的电脑上第二次运行,会直接弹出主界面;若不是在同一台电脑上则弹出输入密钥对话框;

6、KeyDemo程序可执行文件exe的同级目录下用key.db文件,里面保存着输入的密钥

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

顶配版SAM:由分割一切迈向感知一切

文章目录 0. 前言1. 论文地址1.1 项目&代码1.2 模型地址1.3 Demo 2. 模型介绍2.1 亮点2.2 方法 3. 量化结果、可视化展示Reference 0. 前言 现有的视觉分割基础模型&#xff0c;如 SAM 及其变体&#xff0c;集中优势在形状、边缘等初级定位感知&#xff0c;或依赖外部模型…

【Android】使用android studio查看内置数据库信息

背景 需要用到android db 逻辑存储用户信息等等。 使用 在 App inspection 工具中查看该 app 内的 db 数据 sql执行 在新的查询框内解析查询即可知道当前的数据信息。 官方文档-使用 Database Inspector 调试数据库

【计算机毕业设计】SSM医疗药品采购系统

项目介绍 ssm医疗药品采购系统。主要功能有&#xff1a; 用户管理&#xff1a;管理员列表&#xff1b; 采购管理&#xff1a;采购列表&#xff1b; 药品出库&#xff1a;药品出库&#xff1b; 库存管理&#xff1a;库存统计&#xff1b; 数据维护&#xff1a;药品列表、仓库…

【Unity入门】PlayerPrefs的简介与使用

目录 PlayerPrefs储存位置用例注意事项 PlayerPrefs PlayerPrefs 是Unity内置的一个静态类&#xff0c;可以用于存储一些简单的数据类型&#xff1a;int ,string ,float。 分别对应的函数为&#xff1a; SetInt()&#xff1a;保存整型数据GetInt()&#xff1a;读取整形数据Se…

MSF(Metasploit Framework)详细教程

一. 简介 Metasploit 是一个开源的渗透测试开源软件&#xff0c;也是一个逐步发展成熟的漏洞研究与渗透测试代码开发平台&#xff0c;此外也将成为支持整个渗透测试过程的安全技术集成开发与应用环境&#xff0c;2009年10月&#xff0c;Metasploit项目被一家渗透测试技术领域的…

GO学习记录

一、Go语言的源文件的拓展是.go 开发环境和工具&#xff1a;GOLAND 个人版开发&#xff1a; 企业版开发&#xff1a; 二、Go语言结构 1、package main 定义一个名为main的包名 2、import "fmt" 添加fmt包 3、func main() 是程序开始执行的函数 4、定义变量&a…

《对话品牌》——活到老“养”到老

本期节目《对话品牌》栏目组邀请到了深圳壹常青健康管理有限公司董事长邬锡娣女士参加栏目录制&#xff0c;分享其企业故事&#xff0c;树立品牌形象&#xff0c;提升品牌价值&#xff01; 节目嘉宾&#xff1a;邬锡娣女士 节目主持人&#xff1a;董倩 节目播出平台&#xf…

【K8S 二进制部署】部署单Master Kurbernetes集群

目录 一、基本架构和系统初始化 1、集群架构&#xff1a; 2、操作系统初始化配置&#xff1a; 2.1、关闭防火墙和安全机制&#xff1a; 2.2、关闭swap 2.3、根据规划设置主机名 2.4、三台主机全部互相映射 2.5、调整内核参数 3、时间同步&#xff08;所有节点时间必须同…

iframe展示pdf、png、jpg

iframe展示pdf、png、jpg&#xff1a; 1、前端定义div&#xff1a; <div id"pdf-container"></div>/*dpf的div*/ <iframe id"imageFrame"></iframe>/*图片的div*/2、后端查询base64的流&#xff0c;前端页面初始化js方法&#x…

upset 绘制

好久没有更新,今天来一个upset图的绘制 1.1 安装包 #绘制upset的包现在看来有三个 ## UpSet ### 最基本的upsetR包,使用方便,但是扩展不方便 devtools::install_github("hms-dbmi/UpSetR") ## complex-upset ### UpSet的升级款 支持ggplot2 devtools::install_git…

什么是服务器迁移?

服务器迁移一般来说是将物理服务器从一个地点&#xff08;物理机房&#xff09;移动到另一个地点&#xff0c;或将数据从一台服务器移动到另一台服务器的过程。 机房搬迁&#xff1a;当公司办公场所发生变化&#xff0c;原有机房无法继续使用时&#xff0c;需要将服务器迁移到…

计算机报错x3daudio1_7.dll怎么修复,其实很简单

游戏已经成为了人们休闲娱乐的重要方式之一。然而&#xff0c;有时候我们在玩游戏的过程中会遇到一些错误提示&#xff0c;比如“玩游戏报错x3daudio1_7.dll怎么办”。这个问题可能会导致游戏无法正常运行等问题。x3daudio1_7.dll是DirectX Audio API的一部分&#xff0c;它是D…

腾讯云跨云迁移工具案例实践:阿里云迁移到腾讯云

对于阿里云批量迁移到腾讯云&#xff0c;HyperMotion可以支持批量一键式安装Agent软件&#xff0c;做到了操作步骤简单化、自动化&#xff0c;可以满足常见源端操作系统类型。 例如&#xff1a;Windows 2003-2019&#xff0c;CentOS、RedHat 6.x-7.x、Ubuntu 14.x - 16.x、SUS…

为什么TCP会粘包

硬核图解|tcp为什么会粘包&#xff1f;背后的原因让人暖心 数据包报文格式&#xff08;IP包、TCP报头、UDP报头&#xff09; TCP&#xff0c;Transmission Control Protocol。传输控制协议&#xff0c;是一种面向连接的、可靠的、基于字节流的传输层通信协议。 TCP粘包是指发…

【五】【C语言\动态规划】删除并获得点数、粉刷房子、买卖股票的最佳时机含冷冻期,三道题目深度解析

动态规划 动态规划就像是解决问题的一种策略&#xff0c;它可以帮助我们更高效地找到问题的解决方案。这个策略的核心思想就是将问题分解为一系列的小问题&#xff0c;并将每个小问题的解保存起来。这样&#xff0c;当我们需要解决原始问题的时候&#xff0c;我们就可以直接利…

fork函数详解【Linux】

fork函数详解【Linux】 fork函数的概念fork调用后的底层细节解释fork学习中的一些笔记和问题fork的写实拷贝深拷贝的策略 fork调用失败的原因 fork函数的概念 调用fork函数可以在已存在的进程中创建一个子进程&#xff0c;此时&#xff0c;新进程叫做子进程&#xff0c;原进程叫…

k8s二进制部署--部署高可用

连接上文 notready是因为没有网络&#xff0c;因此无法创建pod k8s的CNI网络插件模式 1.pod内部&#xff0c;容器与容器之间的通信。 在同一个pod中的容器共享资源和网络&#xff0c;使用同一个网络命名空间。 2.同一个node节点之内&#xff0c;不同pod之间的通信。 每个pod都…

element ui Checkbox 多选框组件 lable不支持Object类型的值的问题

浅浅记录一下&#xff0c;遇到这个问题的心理路程吧&#xff0c;首先我遇到的问题是多选框的值回显不打对勾&#xff0c;&#xff08;例如&#xff1a;你新增的时候多选&#xff0c;然后点击编辑的时候选过的值没有被勾选&#xff0c;其实是被勾选上了&#xff0c;但是没有显示…

Linux操作系统——进程(六) 进程地址空间

进程地址空间 C/C程序员一般将我们所写的程序看成如下这种结构&#xff1a; 我们所写的程序通过编译编译之后就可以以这样的方式进行分布. 我们先通过编写一段C语言代码来进行验证&#xff1a; 运行结果&#xff1a; 我们可以看出来上述地址遵循的就是我们上面画的一种结构。…

【附视频解析】Jmeter接口之间关联调用(获取上一个接口的返回值作为下一个接口的请求参数)

正则表达式&#xff1a; 具体如何操作&#xff1a; 1. 草稿保存&#xff0c; 此请求的响应数据的id 为发布总结的请求参数draft_id 2. 草稿保存的响应数据 3.在草稿保存的请求中&#xff0c;添加后置处理器- 正则表达式提取器&#xff0c; 提取响应数据的id信息 4. 发布总结请…