<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          手把手教你用Python破解加密 zip 文件的密碼

          共 3480字,需瀏覽 7分鐘

           ·

          2021-05-25 22:51


          今天的文章來自【盞茶作酒】同學。這位同學在老電腦中發(fā)現(xiàn)了一個加密的 zip 文件,于是用 Python 破解了文件密碼。在破解的過程中出現(xiàn)了內(nèi)存爆炸的問題,通過閱讀 Python 源代碼找到了解決方案。

          點擊上方“Python爬蟲與數(shù)據(jù)挖掘”,進行關注

          回復“書籍”即可獲贈Python從入門到進階共10本電子書

          置酒長安道,同心與我違。


          攝影:產(chǎn)品經(jīng)理
          海鮮咖喱泡飯

          之前在家里的老電腦中,發(fā)現(xiàn)一個加密zip壓縮包,由于時隔太久忘記密碼了,依稀記得密碼是6位字母加數(shù)字,網(wǎng)上下載了很多破解密碼的軟件都沒有效果,于是想到自己用Python寫一個暴力破解密碼的腳本。

          Python有一個內(nèi)置模塊zipfile可以干這個事情,測試一波,一個測試文件,設置解壓密碼為123。

          import zipfile

          # 創(chuàng)建文件句柄
          file = zipfile.ZipFile("測試.zip"'r')
          # 提取壓縮文件中的內(nèi)容,注意密碼必須是bytes格式,path表示提取到哪
          file.extractall(path='.', pwd='123'.encode('utf-8'))

          運行效果如下圖所示,提取成功。

          好了開始破解老文件的密碼,為了提高速度我加了多線程最初的代碼:

          import zipfile
          import itertools
          from concurrent.futures import ThreadPoolExecutor

          def extract(file, password):
              if not flag: return
              file.extractall(path='.', pwd=''.join(password).encode('utf-8'))


          def result(f):
              exception = f.exception()
              if not exception:
                  # 如果獲取不到異常說明破解成功
                  print('密碼為:', f.pwd)
                  global flag
                  flag = False


          if __name__ == '__main__':
              # 創(chuàng)建一個標志用于判斷密碼是否破解成功
              flag = True
              # 創(chuàng)建一個線程池
              pool = ThreadPoolExecutor(100)
              nums = [str(i) for i in range(10)]
              chrs = [chr(i) for i in range(6591)]
              # 生成數(shù)字+字母的6位數(shù)密碼
              password_lst = itertools.permutations(nums + chrs, 6)
              # 創(chuàng)建文件句柄
              zfile = zipfile.ZipFile("加密文件.zip"'r')
              for pwd in password_lst:
                  if not flag: break
                  f = pool.submit(extract, zfile, pwd)
                  f.pwd = pwd
                  f.pool = pool
                  f.add_done_callback(result)

          這個代碼有個問題,跑一會兒內(nèi)存就爆了!原因:ThreadPoolExecutor默認使用的是無界隊列,嘗試密碼的速度跟不上生產(chǎn)密碼的速度,會把生產(chǎn)任務無限添加到隊列中。導致內(nèi)存被占滿。內(nèi)存直接飆到95:

          然后程序奔潰:

          看了一下源碼發(fā)現(xiàn)ThreadPoolExecutor內(nèi)部使用的是無界隊列,所以導致內(nèi)存直接飆滿,重寫ThreadPoolExecutor類中的_work_queue屬性,將無界隊列改成有界隊列,這樣就不會出現(xiàn)內(nèi)存爆滿的問題,看代碼:

          import queue
          from concurrent.futures import ThreadPoolExecutor


          class BoundedThreadPoolExecutor(ThreadPoolExecutor):
              def __init__(self, max_workers=None, thread_name_prefix=''):
                  super().__init__(max_workers, thread_name_prefix)
                  self._work_queue = queue.Queue(self._max_workers * 2# 設置隊列大小

          最后破解成功,如下圖所示。

          ------------------- End -------------------

          往期精彩文章推薦:

          歡迎大家點贊,留言,轉(zhuǎn)發(fā),轉(zhuǎn)載,感謝大家的相伴與支持

          想加入Python學習群請在后臺回復【入群

          萬水千山總是情,點個【在看】行不行

          /今日留言主題/

          隨便說一兩句吧~~

          瀏覽 54
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  国产日韩+欧美在线观看 | 手机操逼网站 | 夜夜躁狠狠躁日日躁 | 青青视频网站 | www.丁香五月 |