Python:如何生成随机电话号码?

Bri*_*ian 5 python phone-number

不是制作随机电话号码 xxx-xxx-xxxx的副本

我的项目使用python-phonenumbersdjango-phonenumber-field进行电话号码验证。项目中有大量的自定义验证规则列表,对于这种简单的方法是不够的:

>>> import functools
>>> import random
>>> a = functools.partial(random.randint, 0, 9)
>>> gen = lambda: "+{}-{}{}{}-{}{}{}-{}{}{}{}".format(a(), a(), a(), a(), a(), a(), a(), a(), a(), a(), a())
>>> gen()
'+2-758-702-0180'  # Obviously wrong
>>> gen()
'+1-911-555-0180'  # Obviously wrong, it has 911 in it
Run Code Online (Sandbox Code Playgroud)

那么,如果不诉诸无上限的蛮力 while 循环,也不为这些琐碎的问题引入上限,有什么更好的方法来生成验证器本身接受的有效电话号码?

from phonenumber_field.validators import validate_international_phonenumber
from django.core.exceptions import ValidationError

def generate_valid_number():
    while True:  # While loops are not desired, even with an upper bound!
        try:
            number = gen()
            validate_international_phonenumber(number)
            return number
        except ValidationError:
            pass
Run Code Online (Sandbox Code Playgroud)

muc*_*le6 7

我认为 davidn 对这个问题的回答应该有效,特别是因为您已经在使用 python-phonenumbers

我建议使用 phonenumbers 包,它是 Google 的 libphonenumber 的 python 端口,它现在包含移动运营商的数据集:

import phonenumbers 
from phonenumbers import carrier
from phonenumbers.phonenumberutil import number_type

number = "+49 176 1234 5678"
carrier._is_mobile(number_type(phonenumbers.parse(number))) 
Run Code Online (Sandbox Code Playgroud)

如果 number 是手机号码,这将返回 True,否则返回 False。注意号码必须是有效的国际号码,否则会抛出异常。您还可以使用电话号码来解析给定区域提示的电话号码。


bdo*_*leu 7

如果您正在使用该包,faker您可以编写一个可以在您的项目中重复使用的自定义提供程序。

import phonenumbers
from faker.providers.phone_number.en_US import Provider

class CustomPhoneProvider(Provider):
    def phone_number(self):
        while True:
            phone_number = self.numerify(self.random_element(self.formats))
            parsed_number = phonenumbers.parse(phone_number, 'US')
            if phonenumbers.is_valid_number(parsed_number):
                return phonenumbers.format_number(
                    parsed_number,
                    phonenumbers.PhoneNumberFormat.E164
                )
Run Code Online (Sandbox Code Playgroud)

用于factory_boy

import factory
from faker import Faker

from .models import User
from .providers import CustomPhoneProvider

fake = Faker()
fake.add_provider(CustomPhoneProvider)

class UserFactory(factory.DjangoModelFactory):
    class Meta:
        model = User

    first_name = factory.Faker('first_name')
    last_name = factory.Faker('last_name')
    phone_number = factory.LazyAttribute(lambda _: fake.phone_number())
Run Code Online (Sandbox Code Playgroud)