Python中本地网络的IP地址/主机名列表

Posted

技术标签:

【中文标题】Python中本地网络的IP地址/主机名列表【英文标题】:List of IP addresses/hostnames from local network in Python 【发布时间】:2010-09-17 11:01:41 【问题描述】:

如何在 Python 中轻松获取本地网络的 IP 地址或主机名列表?

最好是多平台的,但需要先在 Mac OS X 上运行,然后其他的。

编辑:本地是指本地网络中的所有活动地址,例如192.168.xxx.xxx

因此,如果我的计算机(在本地网络中)的 IP 地址是 192.168.1.1,并且我连接了另外三台计算机,我希望它返回 IP 地址 192.168.1.2192.168.1.3、@987654325 @,可能还有他们的主机名。

【问题讨论】:

【参考方案1】:

如果“本地”是指在同一网段上,则必须执行以下步骤:

    确定您自己的 IP 地址 确定您自己的网络掩码 确定网络范围 扫描所有地址(除了最低的,即您的网络地址,最高的,即您的广播地址)。 使用 DNS 的反向查找来确定响应扫描的 IP 地址的主机名。

或者您可以让 Python 在外部执行 nmap 并将结果通过管道传回您的程序。

【讨论】:

arp -a??? # 额外的评论空间被识别【参考方案2】:

更新:脚本现在位于github。

我写了一个small python script,它利用了scapy 的arping()

【讨论】:

【参考方案3】:

如果您知道可以使用的计算机名称:

import socket
IP1 = socket.gethostbyname(socket.gethostname()) # local IP adress of your computer
IP2 = socket.gethostbyname('name_of_your_computer') # IP adress of remote computer

否则,您将不得不扫描与本地计算机 (IP1) 具有相同掩码的所有 IP 地址,如另一个答案中所述。

【讨论】:

我如何获取我的计算机名称 socket.gethostname() 返回计算机名【参考方案4】:

我从其他一些线程中收集了以下功能,它适用于我在 Ubuntu 中。

import os
import socket    
import multiprocessing
import subprocess
import os


def pinger(job_q, results_q):
    """
    Do Ping
    :param job_q:
    :param results_q:
    :return:
    """
    DEVNULL = open(os.devnull, 'w')
    while True:

        ip = job_q.get()

        if ip is None:
            break

        try:
            subprocess.check_call(['ping', '-c1', ip],
                                  stdout=DEVNULL)
            results_q.put(ip)
        except:
            pass


def get_my_ip():
    """
    Find my IP address
    :return:
    """
    s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    s.connect(("8.8.8.8", 80))
    ip = s.getsockname()[0]
    s.close()
    return ip


def map_network(pool_size=255):
    """
    Maps the network
    :param pool_size: amount of parallel ping processes
    :return: list of valid ip addresses
    """

    ip_list = list()

    # get my IP and compose a base like 192.168.1.xxx
    ip_parts = get_my_ip().split('.')
    base_ip = ip_parts[0] + '.' + ip_parts[1] + '.' + ip_parts[2] + '.'

    # prepare the jobs queue
    jobs = multiprocessing.Queue()
    results = multiprocessing.Queue()

    pool = [multiprocessing.Process(target=pinger, args=(jobs, results)) for i in range(pool_size)]

    for p in pool:
        p.start()

    # cue hte ping processes
    for i in range(1, 255):
        jobs.put(base_ip + '0'.format(i))

    for p in pool:
        jobs.put(None)

    for p in pool:
        p.join()

    # collect he results
    while not results.empty():
        ip = results.get()
        ip_list.append(ip)

    return ip_list


if __name__ == '__main__':

    print('Mapping...')
    lst = map_network()
    print(lst)

【讨论】:

pinger() 中写着DEVNULL = open(os.devnull, 'w') 的那一行是什么?你不应该在函数结束时关闭os.devnull 以防止内存泄漏吗? devnull 是重定向标准输出并像文件一样打印到它的控制台 Mapping... [] #这是我得到的输出。我只得到一个盒子。这有什么原因吗? 你在linux上吗?上次我检查这在 Windows 上没有按预期工作【参考方案5】:

对于 OSX(和 Linux),一个简单的解决方案是使用 os.popen 或 os.system 并运行 arp -a 命令。

例如:

devices = []
for device in os.popen('arp -a'): devices.append(device)

这将为您提供本地网络上的设备列表。

【讨论】:

【参考方案6】:

我找到了这个network scanner in python article 并写了这个短代码。它做你想做的!但是,您确实需要知道设备的可访问端口。端口 22 是 ssh 标准和我正在使用的。我想你可以遍历所有端口。一些默认值是:

linux: [20, 21, 22, 23, 25, 80, 111, 443, 445, 631, 993, 995]
windows: [135, 137, 138, 139, 445]
mac: [22, 445, 548, 631]
import socket

def connect(hostname, port):
    sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    socket.setdefaulttimeout(1)
    result = sock.connect_ex((hostname, port))
    sock.close()
    return result == 0

for i in range(0,255):
    res = connect("192.168.1."+str(i), 22)
    if res:
        print("Device found at: ", "192.168.1."+str(i) + ":"+str(22))

编辑 by TheLizzard:

使用上面的代码并添加线程:

from threading import Thread, Lock
from time import perf_counter
from sys import stderr
from time import sleep
import socket


# I changed this from "192.168.1.%i" to "192.168.0.%i"
BASE_IP = "192.168.0.%i"
PORT = 80


class Threader:
    """
    This is a class that calls a list of functions in a limited number of
    threads. It uses locks to make sure the data is thread safe.
    Usage:
        from time import sleep

        def function(i):
            sleep(2)
            with threader.print_lock:
                print(i)

        threader = Threader(10) # The maximum number of threads = 10
        for i in range(20):
            threader.append(function, i)
        threader.start()
        threader.join()

    This class also provides a lock called: `<Threader>.print_lock`
    """
    def __init__(self, threads=30):
        self.thread_lock = Lock()
        self.functions_lock = Lock()
        self.functions = []
        self.threads = []
        self.nthreads = threads
        self.running = True
        self.print_lock = Lock()

    def stop(self) -> None:
        # Signal all worker threads to stop
        self.running = False

    def append(self, function, *args) -> None:
        # Add the function to a list of functions to be run
        self.functions.append((function, args))

    def start(self) -> None:
        # Create a limited number of threads
        for i in range(self.nthreads):
            thread = Thread(target=self.worker, daemon=True)
            # We need to pass in `thread` as a parameter so we
            # have to use `<threading.Thread>._args` like this:
            thread._args = (thread, )
            self.threads.append(thread)
            thread.start()

    def join(self) -> None:
        # Joins the threads one by one until all of them are done.
        for thread in self.threads:
            thread.join()

    def worker(self, thread:Thread) -> None:
        # While we are running and there are functions to call:
        while self.running and (len(self.functions) > 0):
            # Get a function
            with self.functions_lock:
                function, args = self.functions.pop(0)
            # Call that function
            function(*args)

        # Remove the thread from the list of threads.
        # This may cause issues if the user calls `<Threader>.join()`
        # But I haven't seen this problem while testing/using it.
        with self.thread_lock:
            self.threads.remove(thread)


start = perf_counter()
# I didn't need a timeout of 1 so I used 0.1
socket.setdefaulttimeout(0.1)

def connect(hostname, port):
    with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as sock:
        result = sock.connect_ex((hostname, port))
    with threader.print_lock:
        if result == 0:
            stderr.write(f"[perf_counter() - start:.5f] Found hostname\n")

threader = Threader(10)
for i in range(255):
    threader.append(connect, BASE_IP%i, PORT)
threader.start()
threader.join()
print(f"[perf_counter() - start:.5f] Done searching")
input("Press enter to exit.\n? ")

【讨论】:

超级慢,我不认为这是一个可行的解决方案 @dlammy 我知道我有点晚了,但我添加了线程。现在它可以在不到 3 秒的时间内扫描所有 256 个可能的 IP。【参考方案7】:

试试:

import socket

print ([ip for ip in socket.gethostbyname_ex(socket.gethostname())[2] if not ip.startswith("127.")][:1])

【讨论】:

你能解释一下socket.gethostbyname_ex(socket.gethostname())的索引1处的数组吗?我不明白为什么它是空的。【参考方案8】:

我已经完成了以下代码来获取 MAC 已知设备的 IP。这可以进行相应的修改,以通过一些字符串操作获得所有 IP。希望这会对你有所帮助。

#running windows cmd line  statement and put output into a string
cmd_out = os.popen("arp -a").read()
line_arr = cmd_out.split('\n')
line_count = len(line_arr)


#search in all lines for ip
for i in range(0, line_count):
    y = line_arr[i]
    z = y.find(mac_address)

    #if mac address is found then get the ip using regex matching
    if z > 0:
        ip_out= re.search('[0-9]+\.[0-9]+\.[0-9]+\.[0-9]+', y, re.M | re.I)

【讨论】:

【参考方案9】:

我刚刚遇到了问题。我是这样解决的:

import kthread #pip install kthread
from time import sleep
import subprocess

def getips():
    ipadressen = 
    def ping(ipadresse):
        try:
            outputcap = subprocess.run([f'ping', ipadresse, '-n', '1'], capture_output=True) #sends only one package, faster
            ipadressen[ipadresse] = outputcap
        except Exception as Fehler:
            print(Fehler)
    t = [kthread.KThread(target = ping, name = f"ipgetteripend", args=(f'192.168.0.ipend',)) for ipend in range(255)] #prepares threads
    [kk.start() for kk in t] #starts 255 threads
    while len(ipadressen) < 255:
        print('Searching network')
        sleep(.3)
    alldevices = []
    for key, item in ipadressen.items():
        if not 'unreachable' in item.stdout.decode('utf-8') and 'failure' not in item.stdout.decode('utf-8'): #checks if there wasn't neither general failure nor 'unrechable host'
            alldevices.append(key)
    return alldevices

allips = getips() #takes 1.5 seconds on my pc

【讨论】:

【参考方案10】:

this question 中的一个答案可能会对您有所帮助。 python 似乎有一个与平台无关的版本,但我还没有尝试过。

【讨论】:

不,我不想要我的 IP 地址,我想要其他人。史蒂夫莫耶所说的,但有代码:)【参考方案11】:

这里有一个小工具scanip,可以帮助你获取网络中所有的ip地址和对应的mac地址(适用于Linux)。

https://github.com/vivkv/scanip

【讨论】:

以上是关于Python中本地网络的IP地址/主机名列表的主要内容,如果未能解决你的问题,请参考以下文章

从IP地址获取主机名

如何获取本地 Wi-Fi 网络中主机名的 IP?安卓工作室(JAVA)

Python获取当前主机(本地计算机)IP地址

Python获取当前主机(本地计算机)IP地址

linux网络管理/etc/hosts文件实现主机名和ip地址映射

网络编程