Re: [问题] 新手for+判断式问题

楼主: outshaker (out)   2020-02-25 22:08:03
※ 引述《yvmi (鞭策自己社社长)》之铭言:
: 这是一个在 codewars 里面的小练习:
: 目标是做一个函式判断一个字串里面是否有包含所有英文字母。
: 我的作法是把所有的英文转成 ascii 码,建成一个 list
: 然后将 list 中非英文的 ascii 码以外的删除。
: 但是实测时发现有部分数值在 97-122 以外的值并不会被删除。
: 比方说如果投入的 s = 'abcdef!!!!'
: ^^→这两码惊叹号会被删除,后面两码不会。
: 如果是在字母的中间则是连续第二个字符不会被删除。
: 请问这个判断式的问题在哪里?
: 感谢大家。
: def is_pangram(s):
: s = s.lower()
: s_list = list(s)
: ascii_i = []
: for i in s_list:
: ascii_i.append(ord(i))
: for j in ascii_i:
: if j<97 or j>122:
: ascii_i.remove(j)
我把你的资料改成 s="[email protected]#$" 跑过同样的程式码变成 "[email protected]$"
这表示!和#有被程式读取到,但@和$应该是跳过没有被读取到
推测原因是你使用remove删除了资料影响原本的读取
一般这种loop是按照顺序(或称index)去读资料的
读到第4个字符,发现是!,移除掉该元素,清单变成"[email protected]#$"
123456
下一次抓的是第5个字符,是#,移除该元素,清单变成"[email protected]$"
12345
下一次抓第6个字符,这时候清单已经抓不到字符了,于是退出loop
要检查程式码运作其实很简单,加一段print(ascii_i) 就可以知道资料会怎么变化
然后再加一些推理就可以猜出原因
最后真的不建议在循环里面增加或删除资料
改变清单的大小,程式会有很多不预期的结果
原PO推文已经有建议用set或counter了,都是现成且方便的解决方案
这边我也给一个做法,先把字母挑出来做成一张表,然后检查a-z是否都有出现在里面
def test(s):
t = []
for c in s:
if 'a' <= c <= 'z' or 'A' <= c <= 'Z':
t.append(c.lower()) # 把字母加进清单
a = "abcdefghijklmnopqrstuvwxyz"
return all([x in t for x in a])
all() 是用来检查所有条件都成立的
[x in t for x in a] 是缩写,等同于以下内容:
list_ = []
for x in a: # x 代表a-z的字母
list_.append(x in t) # 检查字母是否在清单内
然后把字母加入清单的部分可以用 filter() 的写法,有兴趣你再找相关资料
以上
楼主: outshaker (out)   2019-02-25 22:08:00
打这么久没有稿费QQ
作者: Schottky (顺风相送)   2020-02-25 22:32:00
没板主的板,发文不会给 P 币
作者: cuteSquirrel (松鼠)   2020-02-25 23:15:00
push
作者: bibo9901 (function(){})()   2020-02-25 23:25:00
不知道有什么好推的 槽点一堆
作者: jiyu520 (不要鲫鱼我)   2020-02-26 00:55:00
其实可以用 copy or deepcopy
作者: uranusjr (←這人是超級笨蛋)   2020-02-26 02:19:00
其实你可以 if c in string.ascii_letters:
楼主: outshaker (out)   2020-02-26 07:36:00
当初想说用原生的功能打就好了 还是谢谢你的建议

Links booklink

Contact Us: admin [ a t ] ucptt.com