#########################Playfair密码#########################  

#约定1:若明文字母数量为奇数,在明文末尾添加一个'Z'
#约定2:'I'作为'J'来处理  

#字母表
letter_list='ABCDEFGHJKLMNOPQRSTUVWXYZ'  

#密码表
T_letter=['','','','','']  

#根据密钥建立密码表
def Create_Matrix(key):
  key=Remove_Duplicates(key)  #移除密钥中的重复字母
  key=key.replace(' ','') #去除密钥中的空格  

  for ch in letter_list:  #根据密钥获取新组合的字母表
    if ch not in key:
      key+=ch  

  j=0
  for i in range(len(key)): #将新的字母表里的字母逐个填入密码表中,组成5*5的矩阵
    T_letter[j]+=key[i]     #j用来定位字母表的行
    if 0==(i+1)%5:
      j+=1  

#移除字符串中重复的字母
def Remove_Duplicates(key):
  key=key.upper() #转成大写字母组成的字符串
  _key=''
  for ch in key:
    if ch=='I':
      ch='J'
    if ch in _key:
      continue
    else:
      _key+=ch
  return _key  

#获取字符在密码表中的位置
def Get_MatrixIndex(ch):
  for i in range(len(T_letter)):
    for j in range(len(T_letter)):
      if ch==T_letter[i][j]:
        return i,j #i为行,j为列  

#加密
def Encrypt(plaintext,T_letter):
  ciphertext=''  

  if len(plaintext) % 2 !=0:  #如果新的明文长度为奇数,在其末尾添上'Z'
    plaintext+='Z'  

  i=0
  while i<len(plaintext): #对明文进行遍历
    if True==plaintext[i].isalpha():  #如果是明文是字母的话,
      j=i+1                           #则开始对该字母之后的明文进行遍历,
      while j<len(plaintext):         #直到遍历到字母,进行加密
        if True==plaintext[j].isalpha():
          if 'I'==plaintext[i].upper():             #
            x=Get_MatrixIndex('J')                  #
          else:                                     #
            x=Get_MatrixIndex(plaintext[i].upper()) #对字符在密码表中的坐标
          if 'I'==plaintext[j].upper():             #进行定位,同时将'I'作为
              y=Get_MatrixIndex('J')                #'J'来处理
          else:                                     #
            y=Get_MatrixIndex(plaintext[j].upper()) #  

          if x[0]==y[0]:    #如果在同一行
            ciphertext+=T_letter[x[0]][(x[1]+1)%5]+T_letter[y[0]][(y[1]+1)%5]
          elif x[1]==y[1]:  #如果在同一列
            ciphertext+=T_letter[(x[1]+1)%5][x[0]]+T_letter[(y[1]+1)%5][y[0]]
          else:             #如果不同行不同列
            ciphertext+=T_letter[x[0]][y[1]]+T_letter[y[0]][x[1]]
          break;  #每组明文对加密完成后,结束本次对明文的遍历
        j+=1
      i=j+1  #每次对明文的遍历是从加密过后的明文的后一个明文开始的,结束本次循环
      continue
    else:
      ciphertext+=plaintext[i]  #如果明文不是字母,直接加到密文上
    i+=1  

  return ciphertext  

#解密
def Decrypt(ciphertext,T_letter):
  plaintext=''
  if len(ciphertext) % 2 !=0:  #如果新的密文长度为奇数,在其末尾添上'Z'
    ciphertext+='Z'  

  i=0
  while i<len(ciphertext): #对密文进行遍历
    if True==ciphertext[i].isalpha():  #如果是密文是字母的话,
      j=i+1                            #则开始对该字母之后的密文进行遍历,
      while j<len(ciphertext):         #直到遍历到字母,进行解密
        if True==ciphertext[j].isalpha():
          if 'I'==ciphertext[i].upper():              #
            x=Get_MatrixIndex('J')                    #
          else:                                       #
            x=Get_MatrixIndex(ciphertext[i].upper())  #对字符在密码表中的坐标
          if 'I'==ciphertext[j].upper():              #进行定位,同时将'I'作为
              y=Get_MatrixIndex('J')                  #'J'来处理
          else:                                       #
            y=Get_MatrixIndex(ciphertext[j].upper())  #  

          if x[0]==y[0]:    #如果在同一行
            plaintext+=T_letter[x[0]][(x[1]-1)%5]+T_letter[y[0]][(y[1]-1)%5]
          elif x[1]==y[1]:  #如果在同一列
            plaintext+=T_letter[(x[1]-1)%5][x[0]]+T_letter[(y[1]-1)%5][y[0]]
          else:             #如果不同行不同列
            plaintext+=T_letter[x[0]][y[1]]+T_letter[y[0]][x[1]]
          break;  #每组密文对解密完成后,结束本次对密文的遍历
        j+=1
      i=j+1  #每次对密文的遍历是从解密过后的密文的后一个密文开始的,结束本次循环
      continue
    else:
      plaintext+=ciphertext[i]  #如果密文不是字母,直接加到明文上
    i+=1  

  return plaintext  

#主函数
if __name__=='__main__':
  print("加密请按D,解密请按E:")
  user_input=input();
  while(user_input!='D' and user_input!='E'):#输入合法性检测
    print("输入有误!请重新输入:")
    user_input=input()  

  print('请输入密钥,密钥由英文字母组成:')
  key=input()  

  Create_Matrix(key)  #建立密码表  

  if user_input=='D': #加密
    print('请输入明文:')
    plaintext=input()
    print("密文为:\n%s" % Encrypt(plaintext,T_letter))
  else:               #解密
    print('请输入密文:')
    ciphertext=input()
    print('明文为:\n%s' % Decrypt(ciphertext,T_letter))  

python playfair的更多相关文章

  1. 信息安全-1:python之playfair密码算法详解[原创]

    转发注明出处: http://www.cnblogs.com/0zcl/p/6105825.html 一.基本概念 古典密码是基于字符替换的密码.加密技术有:Caesar(恺撒)密码.Vigenere ...

  2. Playfair加密

    前面讲的不管是单码加密还是多码加密都属于单图加密,什么是单图加密和多图加密呢,简单来说单图加密就是一个字母加密一个字母,而多图加密就是一个字符组加密一个字符组.比如双图加密就是两个字母加密两个字母,这 ...

  3. Python中的多进程与多线程(一)

    一.背景 最近在Azkaban的测试工作中,需要在测试环境下模拟线上的调度场景进行稳定性测试.故而重操python旧业,通过python编写脚本来构造类似线上的调度场景.在脚本编写过程中,碰到这样一个 ...

  4. Python高手之路【六】python基础之字符串格式化

    Python的字符串格式化有两种方式: 百分号方式.format方式 百分号的方式相对来说比较老,而format方式则是比较先进的方式,企图替换古老的方式,目前两者并存.[PEP-3101] This ...

  5. Python 小而美的函数

    python提供了一些有趣且实用的函数,如any all zip,这些函数能够大幅简化我们得代码,可以更优雅的处理可迭代的对象,同时使用的时候也得注意一些情况   any any(iterable) ...

  6. JavaScript之父Brendan Eich,Clojure 创建者Rich Hickey,Python创建者Van Rossum等编程大牛对程序员的职业建议

    软件开发是现时很火的职业.据美国劳动局发布的一项统计数据显示,从2014年至2024年,美国就业市场对开发人员的需求量将增长17%,而这个增长率比起所有职业的平均需求量高出了7%.很多人年轻人会选择编 ...

  7. 可爱的豆子——使用Beans思想让Python代码更易维护

    title: 可爱的豆子--使用Beans思想让Python代码更易维护 toc: false comments: true date: 2016-06-19 21:43:33 tags: [Pyth ...

  8. 使用Python保存屏幕截图(不使用PIL)

    起因 在极客学院讲授<使用Python编写远程控制程序>的课程中,涉及到查看被控制电脑屏幕截图的功能. 如果使用PIL,这个需求只需要三行代码: from PIL import Image ...

  9. Python编码记录

    字节流和字符串 当使用Python定义一个字符串时,实际会存储一个字节串: "abc"--[97][98][99] python2.x默认会把所有的字符串当做ASCII码来对待,但 ...

随机推荐

  1. 轻量级SaaS在线作图工具(继之前介绍后完整介绍)

    俗话说“一图胜千言”,在办公应用领域,流程图是一个非常好的表现企业业务流程或工作岗位规范等内容的展现形式,比如去给客户做调研,回来后都要描述出客户的关键业务流程,谁.什么时候.在什么地方.负责什么事情 ...

  2. CSS3边框温故

    1.简介:border属性在CSS1中就已经定义了,用来设置元素边框风格,设置不同的边框.颜色.粗细 2.基本属性,包括三个类型值:(1)border-width:设置元素边框的粗细,默认3~4px( ...

  3. HTML5中的音视频处理

    * 音视频处理 * 视频处理 * 基本内容 * 使用Flash技术处理HTML页面中的视频内容 * 包含音频.动画.网页游戏等 * 特点 * 浏览器原生不支持(IE浏览器要求安装ActiveX组件) ...

  4. Error message when you try to modify or to delete an alternate access mapping in Windows SharePoint Services 3.0: "An update conflict has occurred, and you must re-try this action"

    Article ID: 939308 - View products that this article applies to. Expand all | Collapse all Symptoms ...

  5. 第一个JSP程序

    本文介绍如何写出第一个JSP程序 1.配置服务器 (1)在eclipse中选择Server视图,(ps:很多童鞋说找不到Server,那是因为eclipse的版本问题,请下载JEE版本的eclipse ...

  6. IOS 网络浅析-(六 网络图片获取之三方SDWebImage)

    网络图片获取是大多数app所能用到的,由于实际app开发中原生api很少用到,在这里就先不介绍了,以后有时间会给大家介绍.这篇文章会给大家介绍一个三方-SDWebImage.SDWebImage 是一 ...

  7. iOS多线程-03-NSOperation与NSOperationQueue

    简介 通过NSOperation与NSOperationQueue的组合也能实现多线程 通常将任务封装成NSOperation对象,并将对象添加到NSOperationQueue中实现 NSOpera ...

  8. MongoDb的bin目录下文件mongod,mongo,mongostat命令的说明及使用

    MongoDB的下载地址:http://www.mongodb.org/downloads. 下载好直接解压安装包,即可使用. bin目录下的几个文件说明: mongo 客户端程序,连接MongoDB ...

  9. (ios实战)单个ViewControl适配不同ios版本xib文件实现

    xcode5 中的界面布局 根据sdk 分成ios7.0 and Later 和 ios6.1 and Earlier 两种,那如何xib同时支持 ios6 和ios7 的界面呢 方法如下: 在xco ...

  10. OSX下VirtualBox安装CentOS

    1.OSX上下载安装VirtualBox 2.新建虚拟机(所有选项默认即可) 3.启动虚拟机,选择CentOS安装镜像 CentOS-6.7-x86_64-minimal.iso 此处下载的是最小镜像 ...