主题色

第5章 字符串

字符判断、字符串方法、切片技巧

5-1 判断字符类型

题目:从键盘输入一个字符,判断是数字、英文字母还是其他字符。

ch = input()  # 请输入一个字符
if ch.isdigit():
    print("输入的是数字")
elif ch.isalpha():
    print("输入的是英文字母")
else:
    print("输入的是其他字符")
行号代码白话解释
1ch = input()从键盘读取输入,存到变量 ch 里。input() 读进来的永远是字符串
2if ch.isdigit():调用 isdigit() 方法,判断 ch 是不是纯数字(如 '5')。
4elif ch.isalpha():否则,判断 ch 是不是纯英文字母(如 'a''Z')。
6else:前两关都没过,说明是标点、空格、中文等其他字符。

拓展:也可以用 ASCII 范围判断:if '0' <= ch <= '9':elif 'a' <= ch <= 'z' or 'A' <= ch <= 'Z':,但 isdigit() / isalpha() 更简洁。


5-2 字符串处理

题目:给定 s = 'http://sports.sina.com.cn/',用表达式完成 7 项操作。

先给字符串编号(理解切片的命脉):

h  t  t  p  :  /  /  s  p  o  r  t  s  .  s  i  n  a  .  c  o  m  .  c  n  /
0  1  2  3  4  5  6  7  8  9  10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
题号需求答案白话解释
1计算 t 出现次数s.count('t')count 就像拿着放大镜数项链上有几颗同样的珠子。
2计算 "com" 出现的位置s.find('com')从左往右找,返回第一次出现的起始座位号(结果为 21)。找不到返回 -1
3. 替换为 -s.replace('.', '-')"全部替换",结果是 http://sports-sina-com-cn/
4提取 "sports"s[7:13]7~12 号座位是 sports,终点 13 要 +1(包前不包后)。
4提取 "sina"s[14:18]14~17 号座位是 sina
5字母全变大写s.upper()一键 Caps Lock。
6输出总字符数len(s)数一数这条"字符项链"共有多少颗珠子,结果是 26
7后拼接 "index"s + 'index'+ 把两条字符串首尾相接。

字符串方法速查口诀:count 数一数,find 找一找,replace 换掉它,upper 变大写,len 求长度,加号粘一起。


5-3 字符串切片

题目:填写切片参数,使程序产生期望输出。

s = '接天莲叶无穷碧,映日荷花别样红。'

给字符串编号(从 0 开始):

接  天  莲  叶  无  穷  碧  ,  映  日  荷  花  别  样  红  。
0   1   2   3   4   5   6   7   8   9   10  11  12  13  14  15

参考答案:

s = '接天莲叶无穷碧,映日荷花别样红。'

print(s[3:])       # '叶无穷碧,映日荷花别样红。'
print(s[:8])       # '接天莲叶无穷碧,'
print(s[3::2])     # '叶穷,日花样。'
print(s[::-1])     # '。红样别花荷日映,碧穷无叶莲天接'
print(s[10:3:-2])  # '荷映碧无'
print(s[::3])      # '接叶碧日别。'
print(s[8::2])     # '映荷别红'
题号期望输出答案解析
1叶无穷碧,映日荷花别样红。s[3:]从 3 号 "叶" 开始,取到末尾。
2接天莲叶无穷碧,s[:8]从 0 开始,取到 8 号之前(0~7),刚好包含
3叶穷,日花样。s[3::2]"叶"(3号)出发,步长 2,隔一个取一个:3→5→7→9→11→13→15。
4。红样别花荷日映,碧穷无叶莲天接s[::-1]步长 -1,字符串反转神技。
5荷映碧无s[10:3:-2]"荷"(10号)倒着走,步长 -2,到 3 号之前停:10→8→6→4。
6接叶碧日别。s[::3]从 0 开始,步长 3,跳两格取一个:0→3→6→9→12→15。
7映荷别红s[8::2]"映"(8号)开始,步长 2:8→10→12→14。

切片万能公式: s[start:end:step]

  • start 省略 = 从头来;end 省略 = 取到尾;step 省略 = 1。
  • 包前不包后[2:5] 取 2、3、4,不取 5。
  • 步长为负 = 倒着取,此时终点是不包含的"挡路石"。

附录:字符串核心口诀

切片三参数,起止和步长;
起点 inclusive,终点 exclusive;
步长为正正向走,步长为负倒着取。

编程题

7-1 到底有多二

题目:一个整数"犯二的程度"定义为该数字中包含 2 的个数与其位数的比值。如果这个数是负数,则程度增加 0.5 倍;如果还是个偶数,则再增加 1 倍。

输入格式:一个不超过 50 位的整数 N。

输出格式:犯二程度,保留小数点后两位。

输入样例输出样例
-1314222333681.82%
n = input().strip()

# 统计2的个数
count2 = n.count('2')

# 计算有效位数(负数去掉负号)
if n[0] == '-':
    length = len(n) - 1
    ratio = count2 / length * 1.5
else:
    length = len(n)
    ratio = count2 / length

# 偶数再翻倍
if int(n[-1]) % 2 == 0:
    ratio *= 2

print(f"{ratio * 100:.2f}%")
行号代码白话解释
3n.count('2')数一数字符串里有多少个字符 '2'
6-9if n[0] == '-':负数时,有效位数减 1(去掉负号),比值先乘 1.5。
12-13int(n[-1]) % 2 == 0取最后一位判断奇偶。% 2 == 0 就是偶数,再乘 2。
15f"{ratio * 100:.2f}%"换算成百分比,保留两位小数,末尾加 %

7-2 密码验证

题目:密码必须满足:至少 8 个字符、包含至少一个大写字母、包含至少一个数字。符合要求输出"密码设置成功",否则提示重新输入。输入 Q(不区分大小写)强制结束。

while True:
    pwd = input()
    if pwd.lower() == 'q':
        break
    if (len(pwd) >= 8 and
        any(c.isupper() for c in pwd) and
        any(c.isdigit() for c in pwd)):
        print("密码设置成功")
    else:
        print("密码不符合要求,请重新输入")
行号代码白话解释
1while True:无限循环,直到遇到 break
3pwd.lower() == 'q'统一转小写再比较,实现"不区分大小写"。
5len(pwd) >= 8长度检查。
6any(c.isupper() for c in pwd)any 检查密码里至少有一个大写字母。
7any(c.isdigit() for c in pwd)同理,至少有一个数字。

7-3 身份证掩码

题目:输入 18 位身份证号,用 8 个 * 替换出生年月部分(第 7-14 位)后输出。不足 18 位输出 error

输入样例输出样例
120102198802016598120102********6598
12010219880201659error
s = input()
if len(s) != 18:
    print("error")
else:
    print(s[:6] + "********" + s[14:])
行号代码白话解释
1s = input()读取身份证号。
2len(s) != 18先验长度,不对直接报错。
4s[:6] + "********" + s[14:]前 6 位保留,中间 8 位替换成 *,后 4 位保留。

7-4 身份证号验证

题目:验证 18 位身份证号格式:前 17 位为数字,最后一位为数字或 X,且长度必须为 18。

输入样例输出样例
370101199001011234Valid
s = input().strip()
if (len(s) == 18 and
    s[:17].isdigit() and
    (s[17].isdigit() or s[17].upper() == 'X')):
    print("Valid")
else:
    print("Invalid")
行号代码白话解释
1s = input().strip()strip() 去掉首尾空格,防止误伤。
2len(s) == 18长度检查。
3s[:17].isdigit()前 17 位必须全是数字。
4s[17].isdigit() or s[17].upper() == 'X'最后一位是数字或者 X(大小写均可)。