近来做作业(老男孩那个9.9元的训练营)我想写一个装逼点的密文输入密码,类似于:

这个东西我先前实现过,忘了获取一个字节的方法是什么,于是去网上找,发现网上的实现方式大部分都有问题。

一、网上(百度)的三种实现方式

网上的实现方式不外乎三种:

  1. 直接明文输入(这是扯淡)
  2. 使用getpass模块 覆盖输入,无法看到位数,没有退格。(不好用)
  3. 通过msvcrt模块的getch和putch实现(有问题)

我们重点研究第三种。

二、第三种实现方式的问题何在?

先看代码:

 def inputPassword():# 密码输入
try:
li = []
while True:
ch = msvcrt.getch()
if ch == b'\r': #回车键,确认输入
return ''.join(li) # 返回密码字符串
elif ch == b'\x08': #退格键,删除最后一个输出
del li[-1] #删除密码列表最后一位
msvcrt.putch(b'\b')
msvcrt.putch(b' ')
msvcrt.putch(b'\b')# 这相当于刷新
elif ch == b'\x1a': # ctrl+z 退出键,抛出异常
raise EOFError
else: #否则,就是密码
li.append(ch.decode()) # 加入列表
msvcrt.putch(b'*') # 遮盖符
except EOFError:
pass

看似没有问题,然而当实际运行的时候,获取到的密码会变成

"1 2 3 1 2 3"之类

而且可以发现,按下一个键后会出现两个*

这是为什么呢?

原来,当按下一个键时,实际上捕获到两个ch,一个是实际按下的键,另一个是b'\x00',这个东西造成了密码输入的不正确

怎么改?

只需要加入一个判断 处理b'\x00'就行了

正常的代码:

 def inputPassword():# 密码输入
try:
li = []
while True:
ch = msvcrt.getch()
if ch == b'\r': #回车键,确认输入
return ''.join(li) # 返回密码字符串
elif ch == b'\x08': #退格键,删除最后一个输出
del li[-1] #删除密码列表最后一位
msvcrt.putch(b'\b')
msvcrt.putch(b' ')
msvcrt.putch(b'\b')# 这相当于刷新
elif ch == b'\x00': #去除空字符
pass
elif ch == b'\x1a': # ctrl+z 退出键,抛出异常
raise EOFError
else: #否则,就是密码
li.append(ch.decode()) # 加入列表
msvcrt.putch(b'*') # 遮盖符
except EOFError:
pass

你只需要注意13行就可以。

事情并没用这么简单

在Python2中,不处理b'\x00'也是可以的,这是为何?

为此,我下载了python2和python3的源码,仔细研究

在..\Python-2.7.14\PC文件夹中(或者python3,也是这个文件夹),有一个名为msvcrtmodule.c的文件,这个就是msvcrt模块的源码

 static PyObject *
msvcrt_getch(PyObject *self, PyObject *args)
{
int ch;
char s[]; if (!PyArg_ParseTuple(args, ":getch"))
return NULL; Py_BEGIN_ALLOW_THREADS
ch = _getch();
Py_END_ALLOW_THREADS
s[] = ch;
return PyString_FromStringAndSize(s, );
}

python2

static int
msvcrt_getch_impl(PyObject *module)
/*[clinic end generated code: output=a4e51f0565064a7d input=37a40cf0ed0d1153]*/
{
int ch; Py_BEGIN_ALLOW_THREADS
ch = _getch();
Py_END_ALLOW_THREADS
return ch;
}

python3

比较这两个getch的代码,_getch方法我没有找到,可能在某个头文件中,这个方法的作用应该是获取输入的内容,python2和python3中,应该没有改变

然而在python2中,getch函数返回的是一个PyString_FromStringAndSize(s, 1)的返回值,而python3直接返回了_getch得到的东西。

但是由于笔者C语言拙劣,没用有找到PyString_FromStringAndSize的函数定义,只能推测,在这个函数中,处理掉了b'\x00',导致python2中压根没有获取到b'\x00',也就不用处理了

至于网上的文章,我怀疑msvcrt是最近更新的,python3的前几个版本中,应该还沿用了python2的这个函数。

为此我下载了python3.3的源码,发现getch函数和python2的完全一样。

因此网上没有处理b‘\x00’的方式,应该至少是3.3以前的版本(至于3.4、3.5我没有用过,也没有下载源码核实)

在python标准库网站中,我也没有看到什么解释。。(https://docs.python.org/3.7/library/msvcrt.html)

2019/6/8 更新

最近读C++的书籍,发现C的字符串(或者叫 char )类型,本质是一个数组。最后一个元素通常是以一个 “结束符” 结尾的。但是C++的字符串(C-style string)是一个对象。

或许python2 python3 之间处理数据的差距就在这里吧……

浅论Python密文输入密码的方法的更多相关文章

  1. DAY2 Python 标准库 -> Getpass 模块 -> 命令行下输入密码的方法.

    getpass 模块 getpass 模块提供了平台无关的在命令行下输入密码的方法. getpass(prompt) 会显示提示字符串, 关闭键盘的屏幕反馈, 然后读取密码. 如果提示参数省略, 那么 ...

  2. 开发技术--浅谈python数据类型

    开发|浅谈python数据类型 在回顾Python基础的时候,遇到最大的问题就是内容很多,而我的目的是回顾自己之前学习的内容,进行相应的总结,所以我就不玩基础了,很多在我实际生活中使用的东西,我会在文 ...

  3. 浅谈Python在信息学竞赛中的运用及Python的基本用法

    浅谈Python在信息学竞赛中的运用及Python的基本用法 前言 众所周知,Python是一种非常实用的语言.但是由于其运算时的低效和解释型编译,在信息学竞赛中并不用于完成算法程序.但正如LRJ在& ...

  4. 开发技术--浅谈Python函数

    开发|浅谈Python函数 函数在实际使用中有很多不一样的小九九,我将从最基础的函数内容,延伸出函数的高级用法.此文非科普片~~ 前言 目前所有的文章思想格式都是:知识+情感. 知识:对于所有的知识点 ...

  5. 开发技术--浅谈python基础知识

    开发|浅谈python基础知识 最近复习一些基础内容,故将Python的基础进行了总结.注意:这篇文章只列出来我觉得重点,并且需要记忆的知识. 前言 目前所有的文章思想格式都是:知识+情感. 知识:对 ...

  6. python进阶之内置方法

    python进阶之内置方法 字符串类型的内置方法 常用操作与方法: 按索引取值 str[index] 切片 ste[start:stop:step] 长度 len(str) 成员运算in和not in ...

  7. python 类属性与方法

    Python 类属性与方法 标签(空格分隔): Python Python的访问限制 Python支持面向对象,其对属性的权限控制通过属性名来实现,如果一个属性有双下划线开头(__),该属性就无法被外 ...

  8. Python执行系统命令的方法 os.system(),os.popen(),commands

    os.popen():用python执行shell的命令,并且返回了结果,括号中是写shell命令 Python执行系统命令的方法: https://my.oschina.net/renwofei42 ...

  9. 浅谈Python时间模块

    浅谈Python时间模块 今天简单总结了一下Python处理时间和日期方面的模块,主要就是datetime.time.calendar三个模块的使用.希望这篇文章对于学习Python的朋友们有所帮助 ...

随机推荐

  1. Java遍历List集合的4种方式

    public class Test { public static void main(String[] args) { // 循环遍历List的4中方法 List<String> str ...

  2. 三分钟学会.NET微服务之Polly

    熔断降级是一个非常重要的概念,我们先说一下什么是熔断降级,咱们都知道服务发现,一个有问题的服务器没来得急注销过一会就崩溃掉了,那么我们的请求就有可能访问一个已经崩溃的服务器,那么就会请求失败,因为已经 ...

  3. 《前端之路》之 JavaScript原型及原型链详解

    05:JS 原型链 在 JavaScript 的世界中,万物皆对象! 但是这各种各样的对象其实具体来划分的话就 2 种. 一种是 函数对象,剩下的就是 普通对象.其中 Function 和 Objec ...

  4. spring boot 集成 zookeeper 搭建微服务架构

    PRC原理 RPC 远程过程调用(Remote Procedure Call) 一般用来实现部署在不同机器上的系统之间的方法调用,使得程序能够像访问本地系统资源一样,通过网络传输去访问远程系统资源,R ...

  5. [翻译] GCC 内联汇编 HOWTO

    目录 GCC 内联汇编 HOWTO 原文链接与说明 1. 简介 1.1 版权许可 1.2 反馈校正 1.3 致谢 2. 概览 3. GCC 汇编语法 4. 基本内联 5. 扩展汇编 5.1 汇编程序模 ...

  6. SpringBoot轻松整合ElasticSearch

    完整项目代码地址(https://github.com/fonxian/spring-elasticsearch-example/tree/master/spring-elasticsearch-ex ...

  7. 前端知识复习:Html DIV 图文混排(文字放在图片下边)

    Html知识复习之图文混排 练习练习基础 先上效果图: 废话不多说,直接贴代码: <!DOCTYPE html> <html xmlns="http://www.w3.or ...

  8. Python3 日期与时间戳相互转换

    开发中经常会对时间格式处理,对于时间数据,比如2019-02-28 10:23:29,有时需要日期与时间戳进行相互转换,在Python3中主要用到time模块,相关的函数如下: 其中unix_time ...

  9. Java设计模式---桥接Bridge模式

    参考于 : 大话设计模式 马士兵设计模式视频 写在开头: 桥接模式主要用于一件事物分成了两个维度,进行排列组合,比如礼物,可以分成优雅的礼物(抽象),花(具体),排列组合优雅的花! 1.为什么使用桥接 ...

  10. html 微信video放大后无法返回问题

    android  video播放视频放大后无法返回,先debug下debugx5.qq.com 发现用的不是X5内核 直接激活  debugmm.qq.com/?forcex5=true  问题解决 ...