zoukankan      html  css  js  c++  java
  • 在python实现加密的方式总结

    基础知识扫盲

    对称加密

    对称密钥加密 , 又叫私钥加密。即信息发送的方和接受方用一个密钥去加密和揭秘数据。 最大的优势是 加解密速度快,适合对大量数据进行加密, 对称加密的缺点是密钥的管理和分配, 换句话说就是 如何把密钥发送到需要解密你的消息的人手里的问题。在发送密钥的过程中, 密钥有很大的风险被黑客拦截。 现实中的做法是将对称加密的密钥进行非对称加密然后传给需要他的人。

    非对称加密

    非对称加密系统, 又称公钥密钥加密。 非对称加密为数据的加密与解密提供了一种非常安全的方式。她使用了一对密钥, 私钥和公钥。 私钥只能有一方安全保管, 不能外泄, 而公钥可以发给任何请求她的人。非对称加密使用这对密钥中的一个进行加密, 而解密却需要一个另外一个密钥。 比如你去银行 你向银行请求公钥,银行将公钥发给你,你使用公钥对消息加密,那么只有私钥的持有人–银行才能对你的消息解密。 与对称加密的不同之处是, 银行不需要将私钥通过网络发送出去。因此安全性大大提高。 目前最常用的非对称加密算法是RSA算法。公钥机制灵活,但加密和解密速度却比对称密钥加密慢得多。 公钥机制灵活, 但是加密和解密速度却要比堆成加密慢很多。

    1) Alice需要在银行的网站做一笔交易,她的浏览器首先生成了一个随机数作为对称密钥。
    (2) Alice的浏览器向银行的网站请求公钥。
    (3) 银行将公钥发送给Alice。
    (4) Alice的浏览器使用银行的公钥将自己的对称密钥加密。
    (5) Alice的浏览器将加密后的对称密钥发送给银行。
    (6) 银行使用私钥解密得到Alice浏览器的对称密钥。
    (7) Alice与银行可以使用对称密钥来对沟通的内容进行加密与解密了。

    总结
    (1) 对称加密加密与解密使用的是同样的密钥,所以速度快,但由于需要将密钥在网络传输,所以安全性不高。
    (2) 非对称加密使用了一对密钥,公钥与私钥,所以安全性高,但加密与解密速度慢。
    (3) 解决的办法是将对称加密的密钥使用非对称加密的公钥进行加密,然后发送出去,接收方使用私钥进行解密得到对称加密的密钥,然后双方可以使用对称加密来进行沟通。

    python中加密的注意点

    我们所说的加密方式,都是对二进制编码的格式进行加密的,对应到Python中,则是我们的Bytes。

    所以当我们在Python中进行加密操作的时候,要确保我们操作的是Bytes,否则就会报错。

    将字符串和Bytes互相转换可以使用encode()和decode()方法。

    Base64

    Base64是一种用64个字符来表示任意二进制数据的方法。

    Base64编码可以称为密码学的基石。可以将任意的二进制数据进行Base64编码。所有的数据都能被编码为并只用65个字符就能表示的文本文件。( 65字符:A~Z a~z 0~9 + / = )编码后的数据~=编码前数据的4/3,会大1/3左右。

    原理

    1.将所有字符转化为ASCII码。
    2.将ASCII码转化为8位二进制 。
    3.将二进制3个归成一组(不足3个在后边补0)共24位,再拆分成4组,每组6位。
    4.统一在6位二进制前补两个0凑足8位。
    5.将补0后的二进制转为十进制。
    6.从Base64编码表获取十进制对应的Base64编码。
    

    说明

    1. 转换的时候,将三个byte的数据,先后放入一个24bit的缓冲区中,先来的byte占高位。
    2. 数据不足3byte的话,于缓冲区中剩下的bit用0补足。然后,每次取出6个bit,按照其值选择查表选择对应的字符作为编码后的输出。
    3. 不断进行,直到全部输入数据转换完成。
    4. 如果最后剩下两个输入数据,在编码结果后加1个“=”。
    5. 如果最后剩下一个输入数据,编码结果后加2个“=”。
    6. 如果没有剩下任何数据,就什么都不要加,这样才可以保证资料还原的正确性。

    Base64加密与解密

    注意:
    1.用于base64编码的,要么是ASCII包含的字符,要么是二进制数据
    2.base64 是对称加密

    import base64
    
    s = 'hello, world'
    s = "你好"
    # 加密
    bs = base64.b64encode(s.encode("utf8"))
    print(bs)
    
    # 解密
    decode = base64.b64decode(bs)
    print(decode)
    print(decode.decode("utf8"))
    

    MD5

    message-digest algorithm 5(信息-摘要算法)。经常说的“MD5加密”,就是信息摘要算法。

    md5,其实就是一种算法。可以将一个字符串,或文件,或压缩包,执行md5后,就可以生成一个固定长度为128bit的串。这个串,基本上是唯一的。

    说明

    • 压缩性:任意长度的数据,算出的MD5值长度都是固定的。
    • 容易计算:从原数据计算出MD5值很容易。
    • 抗修改性:对原数据进行任何改动,哪怕只修改1个字节,所得到的MD5值都有很大区别。
    • 强抗碰撞:已知原数据和其MD5值,想找到一个具有相同MD5值的数据(即伪造数据)是非常困难的。
    • 不可逆性:每个人都有不同的指纹,看到这个人,可以得出他的指纹等信息,并且唯一对应,但你只看一个指纹,是不可能看到或读到这个人的长相或身份等信息。

    MD5加密与解密

    由于MD5模块在python3中被移除,在python3中使用hashlib模块进行md5操作

    import hashlib
    
    # 待加密信息
    str = '这是一个测试'
    
    # 创建md5对象
    hl = hashlib.md5()
    
    # 此处必须声明encode
    # 若写法为hl.update(str)  报错为: Unicode-objects must be encoded before hashing
    hl.update(str.encode(encoding='utf-8'))
    
    print('MD5加密前为 :' + str)
    print('MD5加密后为 :' + hl.hexdigest())
    

    输出结果

    MD5加密前为 :这是一个测试
    MD5加密后为 :cfca700b9e09cf664f3ae80733274d9f
    

    md5的长度,默认为128bit,也就是128个0和1的二进制串。这样表达是很不友好的。所以将二进制转成了16进制,每4个bit表示一个16进制,所以128/4 = 32 换成16进制表示后,为32位了。

    为什么网上还有md5是16位的呢?

    其实16位的长度,是从32位md5值来的。是将32位md5去掉前八位,去掉后八位得到的。

    DES

    介绍PyCrypto

    PyCrypto是 Python 中密码学方面最有名的第三方软件包,提供了许多加密算法的使用。可惜的是,它的开发工作于2012年就已停止。

    幸运的是,有一个该项目的分支PyCrytodome 取代了 PyCrypto 。

    安装与导入

    安装之前需要先安装Microsoft Visual c++ 2015。

    在Linux上安装,可以使用以下 pip 命令:

    pip install pycryptodome
    

    导入:

    import Crypto
    

    在Windows 系统上安装则稍有不同:

    pip install pycryptodomex
    

    导入:

    import Cryptodome
    

    DES介绍

    DES算法为密码体制中的对称密码体制,又被称为美国数据加密标准。
    DES是一个分组加密算法,典型的DES以64位为分组对数据加密,加密和解密用的是同一个算法。
    DES算法的入口参数有三个:Key、Data、Mode。其中Key为7个字节共56位,是DES算法的工作密钥;Data为8个字节64位,是要被加密或被解密的数据;Mode为DES的工作方式,有两种:加密或解密。
    密钥长64位,密钥事实上是56位参与DES运算(第8、16、24、32、40、48、56、64位是校验位,使得每个密钥都有奇数个1),分组后的明文组和56位的密钥按位替代或交换的方法形成密文组。

    加密原理

    DES 使用一个 56 位的密钥以及附加的 8 位奇偶校验位,产生最大 64 位的分组大小。这是一个迭代的分组密码,使用称为 Feistel 的技术,其中将加密的文本块分成两半。使用子密钥对其中一半应用循环功能,然后将输出与另一半进行“异或”运算;接着交换这两半,这一过程会继续下去,但最后一个循环不交换。DES 使用 16 个循环,使用异或,置换,代换,移位操作四种基本运算。

    算法步骤

    1)初始置换

    其功能是把输入的64位数据块按位重新组合,并把输出分为L0、R0两部分,每部分各长3 2位,其置换规则为将输入的第58位换到第一位,第50位换到第2位……依此类推,最后一位是原来的第7位。L0、R0则是换位输出后的两部分,L0是输出的左32位,R0是右32位,例:设置换前的输入值为D1D2D3……D64,则经过初始置换后的结果为:L0=D58D50……D8;R0=D57D49……D7。
    其置换规则见下表:
    58,50,42,34,26,18,10,2,60,52,44,36,28,20,12,4,
    62,54,46,38,30,22,14,6,64,56,48,40,32,24,16,8,
    57,49,41,33,25,17,9,1,59,51,43,35,27,19,11,3,
    61,53,45,37,29,21,13,5,63,55,47,39,31,23,15,7,

    2)逆置换

    经过16次迭代运算后,得到L16、R16,将此作为输入,进行逆置换,逆置换正好是初始置换的逆运算,由此即得到密文输出。
    此算法是对称加密算法体系中的代表,在计算机网络系统中广泛使用.

    DES加密与解密

    # 导入DES模块
    from Cryptodome.Cipher import DES
    import binascii
    
    # 这是密钥
    key = b'abcdefgh'
    # 需要去生成一个DES对象
    des = DES.new(key, DES.MODE_ECB)
    # 需要加密的数据
    text = 'python spider!'
    text = text + (8 - (len(text) % 8)) * '='
    
    # 加密的过程
    encrypto_text = des.encrypt(text.encode())
    encrypto_text = binascii.b2a_hex(encrypto_text)
    print(encrypto_text)
    

    3DES

    3DES(或称为Triple DES)是三重数据加密算法(TDEA,Triple Data Encryption Algorithm)块密码的通称。它相当于是对每个数据块应用三次DES加密算法。

    由于计算机运算能力的增强,原版DES密码的密钥长度变得容易被暴力破解。3DES即是设计用来提供一种相对简单的方法,即通过增加DES的密钥长度来避免类似的攻击,而不是设计一种全新的块密码算法。

    3DES(即Triple DES)是DES向AES过渡的加密算法(1999年,NIST将3-DES指定为过渡的加密标准),加密算法,其具体实现如下:设Ek()和Dk()代表DES算法的加密和解密过程,K代表DES算法使用的密钥,M代表明文,C代表密文,这样:

    3DES加密过程为:C=Ek3(Dk2(Ek1(M)))

    3DES解密过程为:M=Dk1(EK2(Dk3(C)))

    AES

    高级加密标准(英语:Advanced Encryption Standard,缩写:AES),在密码学中又称Rijndael加密法,是美国联邦政府采用的一种区块加密标准。这个标准用来替代原先的DES,已经被多方分析且广为全世界所使用。经过五年的甄选流程,高级加密标准由美国国家标准与技术研究院(NIST)于2001年11月26日发布于FIPS PUB 197,并在2002年5月26日成为有效的标准。2006年,高级加密标准已然成为对称密钥加密中最流行的算法之一。

    AES在软件及硬件上都能快速地加解密,相对来说较易于实作,且只需要很少的存储器。作为一个新的加密标准,目前正被部署应用到更广大的范围。

    特点

    1. 抵抗所有已知的攻击。
    2. 在多个平台上速度快,编码紧凑。
    3. 设计简单。

    AES加密与解码

    AES为分组密码,分组密码也就是把明文分成一组一组的,每组长度相等,每次加密一组数据,直到加密完整个明文。在AES标准规范中,分组长度只能是128位,也就是说,每个分组为16个字节(每个字节8位)。密钥的长度可以使用128位、192位或256位。密钥的长度不同,推荐加密轮数也不同。

    一般常用的是128位

    from Cryptodome.Cipher import AES
    from Cryptodome import Random
    
    from binascii import a2b_hex
    
    # 要加密的明文
    data = '南来北往'
    # 密钥key必须为 16(AES-128), 24(AES-192), 32(AES-256)
    key = b'this is a 16 key'
    # 生成长度等于AES 块大小的不可重复的密钥向量
    iv = Random.new().read(AES.block_size)
    print(iv)
    # 使用 key 和iv 初始化AES 对象, 使用MODE_CFB模式
    mycipher = AES.new(key, AES.MODE_CFB, iv)
    print(mycipher)
    # 加密的明文长度必须为16的倍数, 如果长度不为16的倍数, 则需要补足为16的倍数
    # 将iv(密钥向量)加到加密的密钥开头, 一起传输
    ciptext = iv + mycipher.encrypt(data.encode())
    # 解密的话需要用key 和iv 生成的AES对象
    print(ciptext)
    mydecrypt = AES.new(key, AES.MODE_CFB, ciptext[:16])
    # 使用新生成的AES 对象, 将加密的密钥解密
    decrytext = mydecrypt.decrypt(ciptext[16:])
    
    print(decrytext.decode())
    

    RSA

    RSA加密算法是一种非对称加密算法。在公开密钥加密和电子商业中RSA被广泛使用。

    该算法基于一个十分简单的数论事实:将两个大素数相乘十分容易,但那时想要对其乘积进行因式分解却极其困难,因此可以将乘积公开作为加密密钥,即公钥,而两个大素数组合成私钥。公钥是可发布的供任何人使用,私钥则为自己所有,供解密之用。

    非对称加密

    典型的如RSA等,常见方法,使用openssl ,keytools等工具生成一对公私钥对,使用被公钥加密的数据可以使用私钥来解密,反之亦然(被私钥加密的数据也可以被公钥解密) 。

    在实际使用中私钥一般保存在发布者手中,是私有的不对外公开的,只将公钥对外公布,就能实现只有私钥的持有者才能将数据解密的方法。 这种加密方式安全系数很高,因为它不用将解密的密钥进行传递,从而没有密钥在传递过程中被截获的风险,而破解密文几乎又是不可能的。

    但是算法的效率低,所以常用于很重要数据的加密,常和对称配合使用,使用非对称加密的密钥去加密对称加密的密钥。

    RSA加密与解密

    首先我们需要安装一个rsa模块:

    pip install rsa
    

    而且,因为RSA加密算法的特性,RSA的公钥私钥都是10进制的,但公钥的值常常保存为16进制的格式,所以需要将其用int()方法转换为10进制格式。

    import rsa
    import binascii
    
    # 使用网页中获得的n和e值,将明文加密
    def rsa_encrypt(rsa_n, rsa_e, message):
        # 用n值和e值生成公钥
        key = rsa.PublicKey(rsa_n, rsa_e)
        # 用公钥把明文加密
        message = rsa.encrypt(message.encode(), key)
        # 转化成常用的可读性高的十六进制
        message = binascii.b2a_hex(message)
        # 将加密结果转化回字符串并返回
        return message.decode()
    
    # RSA的公钥有两个值n和e,我们在网站中获得的公钥一般就是这样的两个值。
    # n常常为长度为256的十六进制字符串
    # e常常为十六进制‘10001’
    pubkey_n = '8d7e6949d411ce14d7d233d7160f5b2cc753930caba4d5ad24f923a505253b9c39b09a059732250e56c594d735077cfcb0c3508e9f544f101bdf7e97fe1b0d97f273468264b8b24caaa2a90cd9708a417c51cf8ba35444d37c514a0490441a773ccb121034f29748763c6c4f76eb0303559c57071fd89234d140c8bb965f9725'
    pubkey_e = '10001'
    # 需要将十六进制转换成十进制
    rsa_n = int(pubkey_n, 16)
    rsa_e = int(pubkey_e, 16)
    # 要加密的明文
    message = '南北今天很忙'
    
    print("公钥n值长度:", len(pubkey_n))
    print(rsa_encrypt(rsa_n, rsa_e, message))
    

    运行结果:

    公钥n值长度: 256
    480f302eed822c8250256511ddeb017fcb28949cc05739ae66440eecc4ab76e7a7b2f1df398aefdfef2b9bfce6d6152bf6cc1552a0ed8bebee9e094a7ce9a52622487a6412632144787aa81f6ec9b96be95890c4c28a31b3e8d9ea430080d79297c5d75cd11df04df6e71b237511164399d72ccb2f4c34022b1ea7b76189a56e
    
  • 相关阅读:
    FastDFS分布式文件系统
    Nginx负载均衡
    Linux系统:第六章:Linux服务
    libphp5.so可能遇到的问题(转摘)
    apache和tomcat的关系
    linux有些sh文件,为什么要用 ./ 来执行
    Linux上安装apache
    解决centos6系统上python3—flask模块的安装问题
    常见的消息队列中间件介绍
    Linux上部署Tomcat+Nginx负载均衡
  • 原文地址:https://www.cnblogs.com/xxpythonxx/p/12210391.html
Copyright © 2011-2022 走看看