Преобразовать шестнадцатеричную строку с прямым порядком байтов в IP-адрес в Python - PullRequest
8 голосов
/ 04 февраля 2010

Какой лучший способ превратить строку в этой форме в IP-адрес: "0200A8C0".«Октеты», присутствующие в строке, расположены в обратном порядке, т.е. данная строка примера должна генерировать 192.168.0.2.

Ответы [ 4 ]

32 голосов
/ 04 февраля 2010

Работа с сетевым адресом обеспечивается модулем сокета.

socket.inet_ntoa(packed_ip)

Преобразовать 32-разрядный упакованный IPv4-адрес (строка длиной четыре символа) в его стандартное представление в виде строки из четырех точек (например, 123.45.67.89 ’). Это полезно при разговоре с программой, которая использует стандартную библиотеку C и нуждается в объектах типа struct in_addr, который является типом C для 32-разрядных упакованных двоичных данных, которые эта функция принимает в качестве аргумента.

Вы можете перевести свою шестнадцатеричную строку в packed ip, используя struct.pack() и прямой байтовый формат без знака.

>>> import socket
>>> import struct
>>> addr_long = int("0200A8C0",16)
>>> hex(addr_long)
'0x200a8c0'
>>> struct.pack("<L", addr_long)
'\xc0\xa8\x00\x02'

>>> socket.inet_ntoa(struct.pack("<L", addr_long))
'192.168.0.2'
>>> 
5 голосов
/ 04 февраля 2010
>>> s = "0200A8C0"
>>> bytes = ["".join(x) for x in zip(*[iter(s)]*2)]
>>> bytes
['02', '00', 'A8', 'C0']
>>> bytes = [int(x, 16) for x in bytes]
>>> bytes
[2, 0, 168, 192]
>>> print ".".join(str(x) for x in reversed(bytes))
192.168.0.2

Это коротко и ясно; оберните его в функцию с проверкой ошибок, чтобы удовлетворить ваши потребности.


Удобные функции группировки:

def group(iterable, n=2, missing=None, longest=True):
  """Group from a single iterable into groups of n.

  Derived from http://bugs.python.org/issue1643
  """
  if n < 1:
    raise ValueError("invalid n")
  args = (iter(iterable),) * n
  if longest:
    return itertools.izip_longest(*args, fillvalue=missing)
  else:
    return itertools.izip(*args)

def group_some(iterable, n=2):
  """Group from a single iterable into groups of at most n."""
  if n < 1:
    raise ValueError("invalid n")
  iterable = iter(iterable)
  while True:
    L = list(itertools.islice(iterable, n))
    if L:
      yield L
    else:
      break
3 голосов
/ 04 февраля 2010

Вы могли бы сделать что-то вроде этого:

>>> s = '0200A8C0'
>>> octets = [s[i:i+2] for i in range(0, len(s), 2)]
>>> ip = [int(i, 16) for i in reversed(octets)]
>>> ip_formatted = '.'.join(str(i) for i in ip)
>>> print ip_formatted
192.168.0.2

Разделение октетов, вероятно, могло бы быть сделано более элегантно, но я не могу придумать более простой способ решения моей проблемы.

РЕДАКТИРОВАТЬ: Или в одной строке:

>>> s = '0200A8C0'
>>> print '.'.join(str(int(i, 16)) for i in reversed([s[i:i+2] for i in range(0, len(s), 2)]))
192.168.0.2
0 голосов
/ 04 февраля 2010

Моя попытка:

a = '0200A8C0'
indices = range(0, 8, 2)
data = [str(int(a[x:x+2], 16)) for x in indices]
'.'.join(reversed(data))
...