求一段 PHP 和 Python 的 RC4 加解密代码
发布时间:2025-05-17 17:03:56 发布人:远客网络
一、求一段 PHP 和 Python 的 RC4 加解密代码
刚对RC4算法进行了学习,网上发现mory.cn/entry/46753中作者展示了RC4的python实现,但代码缺乏注释,较为晦涩,因此本文对部分代码进行了注释,希望能对学习RC4算法的pythoner有所帮助。
1#/usr/bin/python 2#coding=utf-8 3 import sys,os,hashlib,time,base64 4 def rc4(string, op='encode', public_key='ddd', expirytime= 0): 5 ckey_lenth= 4#定义IV的长度 6 public_key= public_key and public_key or'' 7 key= hashlib.md5(public_key).hexdigest()#将密码public_key进行md5,返回32字节的key 8 keya= hashlib.md5(key[0:16]).hexdigest()#将Key的前16字节md5,返回32字节的keya 9 keyb= hashlib.md5(key[16:32]).hexdigest()#将key的后16字节md5,返回32字节的keyb10 11#当加密时,keyc取time.time()的md5前4字节,用作IV12#当解密时,从密文的前4字节取出IV13 keyc= ckey_lenth and(op=='decode' and string[0:ckey_lenth] or hashlib.md5(str(time.time())).hexdigest()[32- ckey_lenth:32]) or''14 15#真正的密钥cryptkey是由keya拼接keya以及keyc的md5得来的共64字节的字符串16 cryptkey= keya+ hashlib.md5(keya+ keyc).hexdigest()17 key_lenth= len(cryptkey)#6418 19#当加密时,待加密的明文是由10字节的0以及待加密的明文string与keyb的md5值的前前16字节以及明文string拼接而成20#当解密时,密文即为传入的string的前4字节以后的内容并解码21 string= op=='decode' and base64.b64decode(string[4:]) or'0000000000'+ hashlib.md5(string+ keyb).hexdigest()[0:16]+ string22 string_lenth= len(string)23 24 result=''25 box= list(range(256))26 randkey= []27 28 for i in xrange(255):29#随机填充cryptokey中字符的ascii码值,会出现4轮的重复,randkey[0]~randkey[63],randkey[64]~randkey[127],……30 randkey.append(ord(cryptkey[i% key_lenth]))31 32#随机打乱box列表33#cryptkey的真正目的是生成伪随机的box34 for i in xrange(255):35 j= 036 j=(j+ box[i]+ randkey[i])% 25637 tmp= box[i]38 box[i]= box[j]39 box[j]= tmp40 41 for i in xrange(string_lenth):42 a= j= 043 a=(a+ 1)% 25644 j=(j+ box[a])% 25645 tmp= box[a]46 box[a]= box[j]47 box[j]= tmp48#以上再次进行了打乱49 50#真正的明文string逐字节与box中的随机值异或生成加密的result51#不管怎么随机打乱,由于cryptkey以及string_length总是一样的,因此box最终也一样52 result+= chr(ord(string[i]) ^(box[(box[a]+ box[j])% 256]))53#解密时,密文在与box异或则返回明文54 55 if op=='decode':56#result[10:26]== hashlib.md5(result[26:]+ keyb).hexdigest()[0:16],用来验证string的完整性57 if(result[0:10]=='0000000000' or int(result[0:10])- int(time.time())> 0) and result[10:26]== hashlib.md5(result[26:]+ keyb).hexdigest()[0:16]:58 return result[26:]#前十字节是0,再16字节是明文string与keyb的md5前16字节,最后的则是string59 else:60 return None61 else:62#加密,返回IV+result的base64编码63 return keyc+ base64.b64encode(result)64 65 if __name__=='__main__':66#print rc4('我们','encode','98765')67 print rc4('fd09GMhYylNXC5t550VwC5oX9WS4zrB0bI9rs6kvTAMoiGI=','decode','98765')
二、sublime怎么运行python代码
1.首先下载Sublime和Python,安装Python环境
注意:如果不想动手亲自配置Python环境安装的时候环境变量,请在安装的界面给Add Python 3.5 To Path前面打上对号。
安装完成时,Win+R→输入cmd→Enter→调出来命令行,输入python确认安装是否成功。
打开sublime。首选项→浏览程序包→ Python→Python.sublime-build→将下面的代码粘贴到该文件,内容全部替换。(这里使用的已集成部分插件的sublime中文版)
"cmd":["python.exe","-u","$file"],
"path":"C:/Users/csq/AppData/Local/Programs/Python/Python36-32",//注意:路径根据自己的python安装路径而定
"file_regex":"^[ ]*File \"(...*?)\", line([0-9]*)",
点击菜单栏中的工具—>编译系统,勾选Python即可
创建hello.py文件,Ctrl+S保存文件,Ctrl+B执行文件,结果如下图
3.sublime运行python文件的交互环境设置
安装包管理器(package control),查看→显示/隐藏控制台→打开控制台粘贴以下代码→Enter
按ctrl+shift+p快捷键呼出一个输入框,输入Install Package,回车,在新出现的输入框里输入SublimeREPL,在下面的选择列表中选中回车就会开始安装,一般几秒钟就能装完。
在刚刚的python文件的界面里点击上方菜单栏的工具->sublimeREPL->python->python run current file,这时候就像IDLE一样,会弹出一个新的窗口,而且是可交互的,可以输入
键位绑定(每次按照上个步骤执行可交互的python文件比较繁琐,按键绑定可以提高工作效率~)Preferences->按键绑定-用户,粘贴以下代码,并保存。
注意:无论是安装包管理器还是安装包SublimeREPL,执行完成之后需要重启sublime安装才有效。
三、python删除文件夹python删除文件夹里的指定文件
今天小编带着大家来学习Python读写文件的方法,虽然现阶段做项目需要用到文件操作的情况不多了,但是免不了在特殊情况下还是需要用到这个技术的。
今天小编还会给大家讲解Python创建、删除、授权文件夹的方法,这个技术在控制台程序里面用途是非常多的。
Python创建、删除、授权文件夹以及读写文件是依靠os库来实现的,而文件路径则是通过os的path属性对象的方法来处理的,我们来逐个介绍一下。
文件夹操作就是目录操作,在Windows系统里面文件夹叫folder,翻译过来就是文件夹,在Linux系统里面文件夹叫directory,翻译过来就是目录。所以创建、删除、授权文件夹就是创建、删除、授权目录。
读写文件有一些常识需要大家先了解一下。
读写文件可以是本地电脑上面的文件,也可以是远程网络上面的文件,只要授权了就可以操作。文件夹操作可以是本地电脑上面的文件夹,也可以是远程网络上面的文件夹,只要授权了就可以操作。要创建文件需要对创建文件的文件夹有写权限。读写已经存在的文件只需要对文件有权限。文件内容分为普通模式和二进制模式,普通模式通过字符串操作,二进制模式通过字节操作。写文件分为重置写入和追加写入,前者会清空已有内容,后者不会。通过文件指针可以精确控制读写文件内容的具体位置,但是写入只会覆盖已有内容而不会像编辑器一样插入内容。当前文件夹通过“.”表示,上一级文件夹通过“..”表示。任何文件夹都会有“当前文件夹”和“上一级文件夹”。3.文件夹
遍历文件夹用到os库的walk方法,这个方法很方便,直接就把所有的子级、孙级的全部文件夹和文件都遍历出来了。
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"forroot,dirs,filesinos.walk(rootDir):fordirindirs:print("文件夹:"+os.path.abspath(root+"\\"+dir))forfileinfiles:print("文件:"+os.path.abspath(root+"\\"+file))12345673.2不递归遍历
os的walk方法很简单的解决了遍历文件夹的问题,但是并不是任何情况下我们都需要把文件夹里面的子级、孙级全部都找出来的,这时候就用到了os的另外一个方法listdir了。
通过os的listdir可以像PHP那样只遍历一层目录下面的内容,可以通过os.path的isdir、isfile方法判断路径是文件夹还是文件。
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"files=os.listdir(rootDir)forfiinfiles:fullPath=os.path.abspath(rootDir+"\\"+fi)ifos.path.isfile(fullPath):print("文件:"+fullPath)elifos.path.isdir(fullPath):print("文件夹:"+fullPath)1234567893.3普通遍历
既然os的listdir可以遍历一层目录,通过递归函数也就可以遍历所有子级、孙级的目录了。方法和上一课的PHP一样,封装一个函数,在函数里面调用函数自己,实现递归。
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"deflistFolder(dir):files=os.listdir(dir)forfiinfiles:fullPath=os.path.abspath(dir+"\\"+fi)ifos.path.isfile(fullPath):print("文件:"+fullPath)elifos.path.isdir(fullPath):print("文件夹:"+fullPath)listFolder(fullPath)listFolder(rootDir)123456789101112133.4创建
创建文件夹使用os的mkdir方法,创建之后可以通过os的chmod方法授权,通过os的chown方法设置角色。如果创建失败(包括文件夹已经存在的情况)则会抛出异常。
需要注意的是os.chmod和os.chown只有在Linux操作系统里面才能使用。
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"newDir=rootDir+"\\newFolder"os.mkdir(newDir)os.chmod(newDir,0755)os.chown(newDir,100,-1)1234563.5删除
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"newDir=rootDir+"\\newFolder"os.rmdir(newDir)12343.6递归删除
如果文件夹含有子级文件/文件夹则需要先删除子级项目才能删除文件夹,这个时候就体现出os的walk方法的便捷了。
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"newDir=rootDir+"\\newFolder"forroot,dirs,filesinos.walk(newDir):fordirindirs:os.rmdir(root+"\\"+dir)forfileinfiles:os.unlink(root+"\\"+file)123456784.文件4.1创建
Python创建文件只有标准方式一种方法,就是通过open、close实现的。
使用open创建文件需要了解更多知识,首先我们需要指定“文件打开方式”这个概念,使用open实际上知识打开了一个文件,需要我们指定一个打开文件想要干什么的模式,这个就是文件打开方式。
r,读模式r+,可读可写模式,文件指针在文件开头w,写模式,重置文件内容w+,可读可写模式,重置文件内容a,写模式,文件指针在文件末尾a+,可读可写模式,文件指针在文件末尾
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"newFile=rootDir+"\\newFile.py"fo=open(newFile,"w")iffo:writeLen=fo.write("这是福哥新创建的文件")fo.close()12345674.2删除
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"newFile=rootDir+"\\newFile.py"os.unlink(newFile)12344.3读文件
读文件就是以读模式或者读写模式打开文件。
使用open读文件需要用到while语句通过readline循环读取。
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"newFile=rootDir+"\\newFile.py"fc=""fo=open(newFile,"r")iffo:whileTrue:line=fo.readline()ifnotline:breakfc+=linefo.close()1234567891011124.4写文件
写文件就是以写模式或者读写模式打开文件。
使用open写文件需要用到write方法。
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"newFile=rootDir+"\\newFile.py"fo=open(newFile,"w")iffo:fo.write("福哥说这是全部内容了")fo.close()fc=""fo=open(newFile,"r")iffo:whileTrue:line=fo.readline()ifnotline:breakfc+=linefo.close()print(fc)123456789101112131415161718194.5追加写文件
有时候我们不想将文件内容全部重置了,需要在文件现有内容后面追加内容怎么办?
将文件打开方式设置为追加写模式。
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"newFile=rootDir+"\\newFile.py"fo=open(newFile,"a")iffo:fo.write("福哥又说话了")fo.close()fc=""fo=open(newFile,"r")iffo:whileTrue:line=fo.readline()ifnotline:breakfc+=linefo.close()print(fc)123456789101112131415161718194.6文件指针
我们以可读可写模式打开文件后,通过seek移动文件指针,在指定位置进行读写操作,这种方式可以避免将文件全部内容加载到内存当中就可以完成很多情况的读写操作,可以解决针对超大文件内容的格式化数据的编辑的难题。
举例:我们制作一个文件数据库,将用户名和密码存储到一个文件里面,用户名和密码格式如下
[用户名(最长50个字符)][密码(取MD5哈希串,所以固定32个字符)][换行]1然后我们只要保证每一组用户信息都是一样的固定长度就可以通过简单计算知道某一个用户ID对应的用户信息在文件的哪个位置上面,进而通过fseek就可以快速定位并读取用户信息了。
因为我们规定了文件内的用户数据格式,那么通过操控文件指针就可以实现这个文本数据库的增、删、改、查功能了,是不是觉得很神奇?
这是文本数据库TFSimpleTextDb的定义。
classTFSimpleTextDb:rootDir=""fp=""fo=Nonefs=0lastUserId=0lastLoginUserId=0def__init__(self,rootDir):self.rootDir=rootDirself.fp=""self.fo=Noneself.fs=0self.lastUserId=0self.lastLoginUserId=0defstrPad(self,s,padLen,chr):strLen=len(s)padLen-=strLenforiinrange(0,padLen):s+=chrreturnsdefstrRepeat(self,chr,num):s=""foriinrange(0,num):s+=chrreturnsdefconn(self):self.fp=self.rootDir+"/users.tongfu.net.txt"#数据库文件不存在就创建它ifnotos.path.exists(self.fp):withopen(self.fp,"w")asfo:fo.close()try:#打开数据库文件self.fo=open(self.fp,"r+")#记得当前文件尺寸self.fs=os.path.getsize(self.fp)returnTrueexceptExceptionase:print(e)returnFalsedefadd(self,user,pwd):iflen(user)>50:returnfalse#移动文件指针到文件末尾self.fo.seek(self.fs)#写入一行数据保存用户名和密码#用户名用strpad补位,保证数据结构一致line="%s%s\n"%(self.strPad(user,50,''),md5(pwd).hexdigest())self.fo.write(line)self.fs+=len(line)#立即写入新内容到磁盘self.fo.flush()#新用户ID就是最新一行的行号self.lastUserId=int(self.fs/(50+32+1))returnTruedefmod(self,userId,pwd):#移动文件指针到指定userId对应的用户信息位置self.fo.seek((50+32+1)*(userId-1))#按约定数据长度读取数据栏位上的数据userInDb=(self.fo.read(50)).strip()pwdInDb=self.fo.read(32)ifuserInDb=="":returnFalse#修改密码#后退32个字符,在密码栏位开始处开始写self.fo.seek(self.fo.tell()-32)self.fo.write(md5(pwd).hexdigest())#立即写入新内容到磁盘self.fo.flush()returnTruedefdelete(self,userId):#移动文件指针到指定userId对应的用户信息位置self.fo.seek((50+32+1)*(userId-1))#按约定数据长度读取数据栏位上的数据userInDb=(self.fo.read(50)).strip()pwdInDb=self.fo.read(32)ifuserInDb=="":returnFalse#修改密码#后退82个字符,在用户名栏位开始处开始写#写入82个空字符表示用户已经被删除了self.fo.seek(self.fo.tell()-82)self.fo.write(self.strRepeat("",82))#立即写入新内容到磁盘self.fo.flush()returnTruedeflogin(self,user,pwd=""):fo=open(self.fp,"r")iffo:whileTrue:#解析出用户名和密码userInDb=fo.read(50)ifnotuserInDb:breakpwdInDb=fo.read(32)ifnotpwdInDb:breakLF=fo.read(1)ifnotLF:breakuserInDb=userInDb.strip()#验证用户名ifuser==userInDb:#验证密码ifmd5(pwd).hexdigest()==pwdInDb:#计算登录用户IDself.lastLoginUserId=int(fo.tell()/(50+32+1))return0else:return1#密码错误fo.close()return2#用户名不存在defclose(self):self.fo.close()defgetLastUserId(self):returnself.lastUserIddefgetLastLoginUserId(self):returnself.lastLoginUserId1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601614.6.3文本数据库测试
下面是对这个TFSimpleTextDb的测试代码。
myTFSimpleTextDb=TFSimpleTextDb(rootDir)myTFSimpleTextDb.conn()ifmyTFSimpleTextDb.login("福哥")==2:print("创建用户福哥\n")myTFSimpleTextDb.add("福哥","123456")print(myTFSimpleTextDb.getLastUserId())ifmyTFSimpleTextDb.login("鬼谷子叔叔")==2:print("创建用户鬼谷子叔叔\n")myTFSimpleTextDb.add("鬼谷子叔叔","abcdef")print(myTFSimpleTextDb.getLastUserId())ifmyTFSimpleTextDb.login("同福大哥")==2:print("创建用户同福大哥\n")myTFSimpleTextDb.add("同福大哥","123456")print(myTFSimpleTextDb.getLastUserId())print("修改鬼谷子叔叔的密码\n")print(myTFSimpleTextDb.mod(2,"123456"))print("使用新密码登录鬼谷子叔叔\n")print(myTFSimpleTextDb.login("鬼谷子叔叔","123456"))print(myTFSimpleTextDb.getLastLoginUserId())print("删除鬼谷子叔叔\n")print(myTFSimpleTextDb.delete(2))print("再次登录鬼谷子叔叔\n")print(myTFSimpleTextDb.login("鬼谷子叔叔","123456"))myTFSimpleTextDb.close()1234567891011121314151617181920212223242526275.超大文件读写超大文件读写需要避免将文件里面的内容全部读出来放到变量里这种行为,因为变量都是存在于内存当中的,如果变量里面的数据太多会把内存用光,就会导致系统瘫痪,这个太可怕了~~
5.将一个10G文件里的“福哥”换成“鬼谷子叔叔”
这个时候我们可以通过开启两个文件指针,一个文件指针负责读现有文件,一个文件指针负责写新的临时文件,完成后删除现有文件,再将临时文件重命名为原来的文件,就搞定了~~
rootDir=os.path.dirname(os.path.abspath(__file__))+"\\TFSE"newFile=rootDir+"\\newFile.py"tmpFile=rootDir+"\\newFile.py.tmp"foNew=open(newFile,"r")foTmp=open(tmpFile,"w")iffoNewandfoTmp:whileTrue:tmpLine=foNew.readline()ifnottmpLine:breaktmpLine=tmpLine.replace("福哥","鬼谷子叔叔")foTmp.write(tmpLine)foNew.close()foTmp.close()os.unlink(newFile)os.rename(tmpFile,newFile)fc=""withopen(newFile,"r")asfo:fc+=fo.readline()print(fc)123456789101112131415161718192021226.总结好了,今天童鞋们跟着福哥系统地将Python语言操作文件夹、操作文件的方法学习了一遍,有了这些技术之后,今后在项目当中处理各种有关文件夹/文件的问题就不会发怵了!
要注意一点哦,文件夹操作、文件操作属于IO操作,是有一定风险的,一定不要把文件夹/文件的路径搞错了,要不把系统文件或者程序文件写坏了,系统有可能就完蛋了~~