0

我需要编写一个验证 13 位 ISBN 的函数。它需要以 978 或 979 开头,以单个数字结尾,其余部分至少需要 1 位长度。我需要一些帮助才能完成这项工作,我不明白为什么它永远不会返回 true

def validate(s)
 lst = s.split("-")
 isbn= False
 if lst[0] == "978" or lst[0] == "979":
     if len(lst[1])>=1 and len(lst[2])>=1:
         if len(lst[3])==1:
            isbn= True
return isbn
4

4 回答 4

2

ISBN-13 要求 13 位有效。您的代码不会检查所有字符是否都是数字(不包括-分隔符),也不会检查实际长度。此外,需要五个部分,您可以验证校验位。

具体来说,您的代码永远无法返回True,因为第四段 ( lst[3]) 只检查一个字符 ( if len(lst[3])==1:),但是,该元素通常会超过 1 位数字。

通过 PyPI 可以使用可以验证 ISBN 代码的 python 库。这是一个使用示例isbnlib

>>> import isbnlib
>>> isbnlib.is_isbn13('978-3-16-148410-0')
True
>>> isbnlib.is_isbn13('978-3-16-148410-5')
False
>>> isbnlib.is_isbn13('978-3-16-148410-A')
False
>>> isbnlib.is_isbn13('979-3-16-148410-9')
True

另一个更轻量级的库是pyisbn

>>> import pysisbn
>>> pyisbn.validate('979-3-16-148410-9')
True
>>> pyisbn.validate('979-3-16-148410-0')
False

使用这些库的优点是,除了省去您自己解析 ISBN 字符串的麻烦之外,它们还提供了额外的功能,例如从 ISBN-13 转换为 ISBN-10。

于 2016-06-30T00:15:32.953 回答
1

您应该使用正则表达式,这正是它用于:

>>> import re
>>> def validate(isbn):
        isbn_regex = re.compile('^(978|979)-\d+-\d+-\d$')
        return isbn_regex.search(isbn) is not None

>>> print validate('978-12-12-2')
    True

注意:这按照您在上面代码中的逻辑工作(除了您没有检查它是否是数字)。

于 2016-06-29T23:36:03.813 回答
0

错误 1:在“def”语句的末尾缺少一个冒号。

错误2:'return isbn'语句没有缩进;它在函数之外,但应该在函数内部。

错误3:当lst中有四个以上的元素时,检查lst[3]长度的行不检查isbn字符串的最后一个元素。

split 命令在 lst 中为您的第一个字符串 978-3-16-148410-0 创建五个元素;但 lst[3] 有 6 位数字,长度测试失败。

split 命令在 lst 中为您的第二个字符串 978-3-16-148410 创建四个元素;但 lst[3] 有 6 位数字,长度测试失败。

考虑使用 lst[-1] 来指定 lst 的最后一个元素,而不管它包含多少元素。如果您的 i/p 格式正确,则 o/p 应该是正确的。

于 2016-06-30T00:30:14.440 回答
0

ISBN-13 由五组数字组成,最后一位是校验位。这是一个确保有五个组,正好是 13 位的函数,并验证校验位。它适用于您的样品:

import re

def validate(s):
    d = re.findall(r'\d',s)
    if len(d) != 13:
        return False
    if not re.match(r'97[89](?:-\d+){3}-\d$',s):
        return False

    # The ISBN-13 check digit, which is the last digit of the ISBN, must range from 0 to 9
    # and must be such that the sum of all the thirteen digits, each multiplied by its
    # (integer) weight, alternating between 1 and 3, is a multiple of 10.
    odd = [int(x) for x in d[::2]]
    even = [int(x)*3 for x in d[1::2]]
    return (sum(odd)+sum(even)) % 10 == 0

trials = '''\
978-3-16-148410-0
978-3-16-148410
978-0-306-40615-7
978-0306-40615-7
979-11111-11-11-2
978-7654-321-12-4
977-7654-321-12-4
978-7654-321-1-41
978-7654-321-1-4
978-7654-321-122-4
'''.splitlines()

for trial in trials:
    print(validate(trial),trial)

输出:

True 978-3-16-148410-0
False 978-3-16-148410        # too few numbers and groups       
True 978-0-306-40615-7
False 978-0306-40615-7       # too few groups
True 979-11111-11-11-2
False 978-7654-321-12-4      # wrong check digit
False 977-7654-321-12-4      # didn't start with 978 or 979
False 978-7654-321-1-41      # didn't end in one digit.
False 978-7654-321-1-4       # too few digits
False 978-7654-321-122-4     # too many digits
于 2016-06-30T00:15:00.413 回答