如何在Linux和OS X上获取本地计算机的主IP地址?

时间:2012-11-10 13:32:11

标签: bash unix ip ifconfig

我正在寻找一个命令行解决方案,它将返回localhost的主要(第一个)IP地址,而不是127.0.0.1

该解决方案至少适用于Linux(Debian和RedHat)和OS X 10.7 +

我知道ifconfig两者都可用,但它们的输出在这些平台之间并不一致。

31 个答案:

答案 0 :(得分:425)

使用grepifconfig过滤IP地址:

ifconfig | grep -Eo 'inet (addr:)?([0-9]*\.){3}[0-9]*' | grep -Eo '([0-9]*\.){3}[0-9]*' | grep -v '127.0.0.1'

sed

ifconfig | sed -En 's/127.0.0.1//;s/.*inet (addr:)?(([0-9]*\.){3}[0-9]*).*/\2/p'

如果您只对某些接口,wlan0,eth0等感兴趣,那么:

ifconfig wlan0 | ...

您可以在.bashrc中将命令别名为创建您自己的名为myip的命令。

alias myip="ifconfig | sed -En 's/127.0.0.1//;s/.*inet (addr:)?(([0-9]*\.){3}[0-9]*).*/\2/p'"

更简单的方法是hostname -Ihostname -i旧版本hostname,但请参阅评论)。但是,这仅适用于Linux。

答案 1 :(得分:201)

对于linux机器(不是OS X):

hostname --ip-address

答案 2 :(得分:198)

以下内容适用于Linux,但不适用于OSX。

这完全不依赖于DNS,即使未正确设置/etc/hosts11.0.0.0的简写),它仍然有效:

ip route get 1 | awk '{print $NF;exit}'

或避免awk并在8.8.8.8使用Google的公共DNS显而易见:

ip route get 8.8.8.8 | head -1 | cut -d' ' -f8

不太可靠的方式:(见下面的评论)

hostname -I | cut -d' ' -f1

答案 3 :(得分:32)

已编辑( 2014-06-01 2018-01-09)

对于更强的配置,在每个接口上配置了许多接口和许多IP,我编写了一个 pure bash 脚本(不是基于 127.0.0.1 )来查找正确的界面 ip,基于 default route 。我在这个答案的最底部发布了这个脚本。

简介

由于默认情况下两个Os都安装了,因此Mac和Linux都有一个bash提示:

使用LANG=C

阻止了区域设置问题
myip=
while IFS=$': \t' read -a line ;do
    [ -z "${line%inet}" ] && ip=${line[${#line[1]}>4?1:2]} &&
        [ "${ip#127.0.0.1}" ] && myip=$ip
  done< <(LANG=C /sbin/ifconfig)
echo $myip

将此功能纳入一个功能:

最小

getMyIP() {
    local _ip _line
    while IFS=$': \t' read -a _line ;do
        [ -z "${_line%inet}" ] &&
           _ip=${_line[${#_line[1]}>4?1:2]} &&
           [ "${_ip#127.0.0.1}" ] && echo $_ip && return 0
      done< <(LANG=C /sbin/ifconfig)
}

简单使用:

getMyIP
192.168.1.37

花哨整洁:

getMyIP() {
    local _ip _myip _line _nl=$'\n'
    while IFS=$': \t' read -a _line ;do
        [ -z "${_line%inet}" ] &&
           _ip=${_line[${#_line[1]}>4?1:2]} &&
           [ "${_ip#127.0.0.1}" ] && _myip=$_ip
      done< <(LANG=C /sbin/ifconfig)
    printf ${1+-v} $1 "%s${_nl:0:$[${#1}>0?0:1]}" $_myip
}

用法:

getMyIP
192.168.1.37

或者,运行相同的函数,但带有参数:

getMyIP varHostIP
echo $varHostIP
192.168.1.37
set | grep ^varHostIP
varHostIP=192.168.1.37

Nota:没有参数,此函数在STDOUT上输出,IP和换行符,带参数,没有打印任何内容,但创建了一个名为参数的变量并包含不带换行符的IP

Nota2:这是在Debian上测试的,LaCie攻击了nas和MaxOs。如果这在你的环境下不起作用,我会对反馈非常感兴趣!

此答案的旧版本

(未删除,因为基于sed,而不是bash。)

警告:区域设置存在问题!

快速和小:

myIP=$(ip a s|sed -ne '/127.0.0.1/!{s/^[ \t]*inet[ \t]*\([0-9.]\+\)\/.*$/\1/p}')

爆炸(也是工作;)

myIP=$(
    ip a s |
    sed -ne '
        /127.0.0.1/!{
            s/^[ \t]*inet[ \t]*\([0-9.]\+\)\/.*$/\1/p
        }
    '
)

修改

如何!这似乎不适用于 Mac OS ...

好的,这似乎在 Mac OS 上与在我的 Linux 上完全相同:

myIP=$(LANG=C /sbin/ifconfig  | sed -ne $'/127.0.0.1/ ! { s/^[ \t]*inet[ \t]\\{1,99\\}\\(addr:\\)\\{0,1\\}\\([0-9.]*\\)[ \t\/].*$/\\2/p; }')

分裂:

myIP=$(
    LANG=C /sbin/ifconfig  |
        sed -ne $'/127.0.0.1/ ! {
            s/^[ \t]*inet[ \t]\\{1,99\\}\\(addr:\\)\\{0,1\\}\\([0-9.]*\\)[ \t\/].*$/\\2/p;
        }')

我的剧本(jan 2018):

此脚本将首先找到用于的默认路由接口,然后搜索 gateway 的本地ip匹配网络并填充变量。最后两行只是打印,如:

Interface   : en0
Local Ip    : 10.2.5.3
Gateway     : 10.2.4.204
Net mask    : 255.255.252.0
Run on mac  : true

Interface   : eth2
Local Ip    : 192.168.1.31
Gateway     : 192.168.1.1
Net mask    : 255.255.255.0
Run on mac  : false

嗯,就是:

#!/bin/bash
runOnMac=false
int2ip() { printf ${2+-v} $2 "%d.%d.%d.%d" \
        $(($1>>24)) $(($1>>16&255)) $(($1>>8&255)) $(($1&255)) ;}
ip2int() { local _a=(${1//./ }) ; printf ${2+-v} $2 "%u" $(( _a<<24 |
                  ${_a[1]} << 16 | ${_a[2]} << 8 | ${_a[3]} )) ;}
while IFS=$' :\t\r\n' read a b c d; do
    [ "$a" = "usage" ] && [ "$b" = "route" ] && runOnMac=true
    if $runOnMac ;then
        case $a in 
            gateway )    gWay=$b  ;;
            interface )  iFace=$b ;;
        esac
    else
        [ "$a" = "0.0.0.0" ] && [ "$c" = "$a" ] && iFace=${d##* } gWay=$b
    fi
done < <(/sbin/route -n 2>&1 || /sbin/route -n get 0.0.0.0/0)
ip2int $gWay gw
while read lhs rhs; do
    [ "$lhs" ] && { 
        [ -z "${lhs#*:}" ] && iface=${lhs%:}
        [ "$lhs" = "inet" ] && [ "$iface" = "$iFace" ] && {
            mask=${rhs#*netmask }
            mask=${mask%% *}
            [ "$mask" ] && [ -z "${mask%0x*}" ] &&
                printf -v mask %u $mask ||
                ip2int $mask mask
            ip2int ${rhs%% *} ip
            (( ( ip & mask ) == ( gw & mask ) )) &&
                int2ip $ip myIp && int2ip $mask netMask
        }
    }
done < <(/sbin/ifconfig)
printf "%-12s: %s\n" Interface $iFace Local\ Ip $myIp \
       Gateway $gWay Net\ mask $netMask Run\ on\ mac $runOnMac

答案 4 :(得分:26)

在linux上

hostname -I

在macOS上

ipconfig getifaddr en0

请注意,hostname -I可以以不可靠的顺序返回多个地址(请参阅man hostname)。但对我来说,它只返回192.168.1.X这就是你想要的。

答案 5 :(得分:24)

解决方案

$ ip -o route get to 8.8.8.8 | sed -n 's/.*src \([0-9.]\+\).*/\1/p'
192.168.8.16

说明

正确的查询网络信息的方式是使用ip

  • -o单行输出
  • route get to获取到目的地的实际内核路由
  • 8.8.8.8 Google IP,但可以使用您想要覆盖的真实IP

e.g。 ip输出:

8.8.8.8 via 192.168.8.254 dev enp0s25 src 192.168.8.16 uid 1000 \   cache

要提取src ip,sed是最强大的,并且与正则表达式支持最兼容:

  • -n默认情况下无输出
  • 's/pattern/replacement/p'匹配模式并仅打印替换
  • .*src \([0-9.]\+\).*与内核使用的src IP匹配,以达到8.8.8.8

e.g。最终输出:

192.168.8.16

其他答案

我认为上述答案对我来说都不够好,因为它们在最近的机器中不起作用(Gentoo 2018)。

我在前面的答案中找到的问题:

  • 在命令输出中使用位置列;
  • 使用不推荐使用的ifconfig - 例如 - 不列出多个IP;
  • 使用awk进行一项可以更好地处理的简单任务;
  • ip route get 1目前尚不清楚,实际上是ip route get to 1.0.0.0
  • 的别名
  • 使用hostname命令,该命令在所有设备中没有-I选项,并且在我的情况下返回127.0.0.1

答案 6 :(得分:22)

你也可以试试这个,虽然它可能会告诉你127.0.0.1

hostname  -i

hostname -I

答案 7 :(得分:22)

您还可以在linux

中使用此命令获取eth0的IP版本4地址
/sbin/ip -4 -o addr show dev eth0| awk '{split($4,a,"/");print a[1]}'

输出将是这样的

[root@localhost Sathish]# /sbin/ip -4 -o addr show dev eth0| awk '{split($4,a,"/");print a[1]}'
192.168.1.22

答案 8 :(得分:14)

这适用于Linux和OSX

这将获得与默认路由相关联的接口

NET_IF=`netstat -rn | awk '/^0.0.0.0/ {thif=substr($0,74,10); print thif;} /^default.*UG/ {thif=substr($0,65,10); print thif;}'`

使用上面发现的界面,获取IP地址。

NET_IP=`ifconfig ${NET_IF} | grep -Eo 'inet (addr:)?([0-9]*\.){3}[0-9]*' | grep -Eo '([0-9]*\.){3}[0-9]*' | grep -v '127.0.0.1'`

OSX

uname -a

Darwin笔记本电脑14.4.0 Darwin内核版本14。4。0:2015年5月28日星期四11:35:04 PDT 2015; root:xnu-2782.30.5~1 / RELEASE_X86_64 x86_64

echo $NET_IF

en5

echo $NET_IP

192.168.0.130

CentOS Linux

uname -a

Linux dev-cil.medfx.local 2.6.18-164.el5xen 1 SMP Thu Sep 3 04:03:03 EDT 2009 x86_64 x86_64 x86_64 GNU / Linux

echo $NET_IF

的eth0

echo $NET_IP

192.168.46.10

答案 9 :(得分:11)

使用其他一些方法您可能会输入在系统上定义了多个IP地址的冲突。 默认情况下,此行始终获取IP地址。

ip route get 8.8.8.8 | head -1 | awk '{print $7}'

答案 10 :(得分:6)

我正在提取对此答案的评论:

ip route get 1 | sed -n 's/^.*src \([0-9.]*\) .*$/\1/p'

它基于@CollinAnderson的答案,在我的情况下不起作用。

答案 11 :(得分:6)

我必须向Collin Andersons回答说,如果你有两个接口并且它们都显示为up,那么这个方法也会考虑到。

ip route get 1 | awk '{print $NF;exit}'

我一直在使用Raspberry Pi的应用程序,并且需要实际使用的IP地址,而不仅仅是它是否启动。大多数其他答案将返回两个IP地址,这对我的情况无论如何都不一定有用。

答案 12 :(得分:5)

主网络接口IP

ifconfig `ip route | grep default | head -1 | sed 's/\(.*dev \)\([a-z0-9]*\)\(.*\)/\2/g'` | grep -oE "\b([0-9]{1,3}\.){3}[0-9]{1,3}\b" | head -1

答案 13 :(得分:4)

在作为默认网关的网络(例如,排除所有虚拟网络,泊坞桥)中查找此计算机的IP地址。互联网网关,WiFi网关,以太网

ip route| grep $(ip route |grep default | awk '{ print $5 }') | grep -v "default" | awk '/scope/ { print $9 }'

在Linux上工作。

测试:

➜  ~ ip route| grep $(ip route |grep default | awk '{ print $5 }') | grep -v "default" | awk '/scope/ { print $9 }'
192.168.0.114

➜  reverse-networking git:(feature/type-local) ✗ ifconfig wlp2s0
wlp2s0: flags=4163<UP,BROADCAST,RUNNING,MULTICAST>  mtu 1500
        inet 192.168.0.114  netmask 255.255.255.0  broadcast 192.168.0.255
        inet6 fe80::d3b9:8e6e:caee:444  prefixlen 64  scopeid 0x20<link>
        ether ac:x:y:z  txqueuelen 1000  (Ethernet)
        RX packets 25883684  bytes 27620415278 (25.7 GiB)
        RX errors 0  dropped 27  overruns 0  frame 0
        TX packets 7511319  bytes 1077539831 (1.0 GiB)
        TX errors 0  dropped 0 overruns 0  carrier 0  collisions 0

答案 14 :(得分:4)

ifconfig | grep 'inet ' | grep -v '127.0.0.1' | awk '{print $2}'

答案 15 :(得分:4)

这里有一个节点包。它是跨平台的,易于使用。

$ npm install --global internal-ip-cli

$ internal-ip
fe80::1

$ internal-ip --ipv4
192.168.0.3

这是一种有争议的方法,但使用npm进行工具变得越来越受欢迎,无论喜欢与否。

答案 16 :(得分:4)

在Linux系统上获取本地ipv4-address的最短路径:

hostname -I | awk '{print $1}'

答案 17 :(得分:3)

我经历了很多链接(StackExchange,AskUbuntu,StackOverflow等),并决定将所有最佳解决方案合并到一个shell脚本中。

在我看来,这两个QA是最好看的:

如何在shell脚本中获取外部IP地址? https://unix.stackexchange.com/q/22615

如何找到我的内部IP地址? https://askubuntu.com/a/604691

这是我的解决方案,基于rsp在其存储库中共享的一些想法(https://github.com/rsp/scripts/)。

有些人可能会说这个脚本对于如此简单的任务来说非常庞大,但我希望尽可能简化和灵活使用它。它支持简单的配置文件,允许重新定义默认值。

它已在Cygwin,MINGW和Linux(Red Hat)下成功测试。

显示内部IP地址

myip -i

显示外部IP地址

myip -e

源代码,也可通过以下链接获取:https://github.com/ildar-shaimordanov/tea-set/blob/master/home/bin/myip。主脚本旁边有配置文件示例。

#!/bin/bash

# =========================================================================
#
# Getting both internal and external IP addresses used for outgoing 
# Internet connections.
#
# Internal IP address is the IP address of your computer network interface 
# that would be used to connect to Internet.
#
# External IP address is the IP address that is visible by external 
# servers that you connect to over Internet.
#
# Copyright (C) 2016 Ildar Shaimordanov
#
# =========================================================================

# Details of the actual implementation are based on the following QA:
#
# How can I get my external IP address in a shell script?
# https://unix.stackexchange.com/q/22615
#
# How do I find my internal ip address?
# https://askubuntu.com/a/604691

# =========================================================================

for f in \
    "$( dirname "$0" )/myip.conf" \
    ~/.myip.conf \
    /etc/myip.conf
do
    [ -f "$f" ] && {
        . "$f"
        break
    }
done

# =========================================================================

show_usage() {
    cat - <<HELP
USAGE
  $( basename "$0" ) [OPTIONS]

DESCRIPTION
  Display the internal and external IP addresses

OPTIONS
  -i  Display the internal IP address
  -e  Display the external IP address
  -v  Turn on verbosity
  -h  Print this help and exit
HELP
    exit
}

die() {
    echo "$( basename "$0" ): $@" >&2
    exit 2
}

# =========================================================================

show_internal=""
show_external=""
show_verbose=""

while getopts ":ievh" opt
do
    case "$opt" in
    i )
        show_internal=1
        ;;
    e )
        show_external=1
        ;;
    v )
        show_verbose=1
        ;;
    h )
        show_usage
        ;;
    \? )
        die "Illegal option: $OPTARG"
        ;;
    esac
done

if [ -z "$show_internal" -a -z "$show_external" ]
then
    show_internal=1
    show_external=1
fi

# =========================================================================

# Use Google's public DNS to resolve the internal IP address
[ -n "$TARGETADDR" ] || TARGETADDR="8.8.8.8"

# Query the specific URL to resolve the external IP address
[ -n "$IPURL" ] || IPURL="ipecho.net/plain"

# Define explicitly $IPCMD to gather $IPURL using another tool
[ -n "$IPCMD" ] || {
    if   which curl >/dev/null 2>&1
    then
        IPCMD="curl -s"
    elif which wget >/dev/null 2>&1
    then
        IPCMD="wget -qO -"
    else
        die "Neither curl nor wget installed"
    fi
}

# =========================================================================

resolveip() {
    {
        gethostip -d "$1" && return
        getent ahostsv4 "$1" \
        | grep RAW \
        | awk '{ print $1; exit }' 
    } 2>/dev/null
}

internalip() {
    [ -n "$show_verbose" ] && printf "Internal: "

    case "$( uname | tr '[:upper:]' '[:lower:]' )" in
    cygwin* | mingw* | msys* )
        netstat -rn \
        | grep -w '0.0.0.0' \
        | awk '{ print $4 }'
        return
        ;;
    esac

    local t="$( resolveip "$TARGETADDR" )"
    [ -n "$t" ] || die "Cannot resolve $TARGETADDR"
    ip route get "$t" \
    | awk '{ print $NF; exit }'
}

externalip() {
    [ -n "$show_verbose" ] && printf "External: "

    eval $IPCMD "$IPURL" $IPOPEN
}

# =========================================================================

[ -n "$show_internal" ] && internalip
[ -n "$show_external" ] && externalip

# =========================================================================

# EOF

答案 18 :(得分:3)

我只使用Network Interface Names,我的自定义命令是

[[ $(ip addr | grep enp0s25) != '' ]] && ip addr show dev enp0s25 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p' || ip addr show dev eth0 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p' 

在我自己的笔记本中

[flying@lempstacker ~]$ cat /etc/redhat-release 
CentOS Linux release 7.2.1511 (Core) 
[flying@lempstacker ~]$ [[ $(ip addr | grep enp0s25) != '' ]] && ip addr show dev enp0s25 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p' || ip addr show dev eth0 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p'
192.168.2.221
[flying@lempstacker ~]$

但是如果网络接口拥有至少一个ip,那么它将显示所有ip属于它

例如

Ubuntu 16.10

root@yakkety:~# sed -r -n 's@"@@g;s@^VERSION=(.*)@\1@p' /etc/os-release
16.04.1 LTS (Xenial Xerus)
root@yakkety:~# [[ $(ip addr | grep enp0s25) != '' ]] && ip addr show dev enp0s25 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p' || ip addr show dev eth0 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p'
178.62.236.250
root@yakkety:~#

Debian Jessie

root@jessie:~# sed -r -n 's@"@@g;s@^PRETTY_NAME=(.*)@\1@p' /etc/os-release
Debian GNU/Linux 8 (jessie)
root@jessie:~# [[ $(ip addr | grep enp0s25) != '' ]] && ip addr show dev enp0s25 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p' || ip addr show dev eth0 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p'
192.81.222.54
root@jessie:~# 

CentOS 6.8

[root@centos68 ~]# cat /etc/redhat-release 
CentOS release 6.8 (Final)
[root@centos68 ~]# [[ $(ip addr | grep enp0s25) != '' ]] && ip addr show dev enp0s25 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p' || ip addr show dev eth0 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p'
162.243.17.224
10.13.0.5
[root@centos68 ~]# ip route get 1 | awk '{print $NF;exit}'
162.243.17.224
[root@centos68 ~]#

Fedora 24

[root@fedora24 ~]# cat /etc/redhat-release 
Fedora release 24 (Twenty Four)
[root@fedora24 ~]# [[ $(ip addr | grep enp0s25) != '' ]] && ip addr show dev enp0s25 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p' || ip addr show dev eth0 | sed -n -r 's@.*inet (.*)/.*brd.*@\1@p'
104.131.54.185
10.17.0.5
[root@fedora24 ~]# ip route get 1 | awk '{print $NF;exit}'
104.131.54.185
[root@fedora24 ~]#

似乎link提供的命令ip route get 1 | awk '{print $NF;exit}'更准确,更短,更短。

答案 19 :(得分:3)

假设您需要从世界其他地方看到的主要公共 IP,请尝试以下任何一种:

wget http://ipecho.net/plain -O - -q
curl http://icanhazip.com
curl http://ifconfig.me/ip

答案 20 :(得分:3)

另一个适用于Linux和OSX的ifconfig变量:

ifconfig | grep "inet " | cut -f2 -d' '

答案 21 :(得分:3)

ip addr show | grep -E '^\s*inet' | grep -m1 global | awk '{ print $2 }' | sed 's|/.*||'

答案 22 :(得分:2)

不确定这是否适用于所有操作系统,请尝试一下。

ifconfig | awk -F"[ :]+" '/inet addr/ && !/127.0/ {print $4}'

答案 23 :(得分:1)

ifconfig $(netstat -rn | grep -E "^default|^0.0.0.0" | head -1 | awk '{print $NF}') | grep 'inet ' | awk '{print $2}' | grep -Eo '([0-9]*\.){3}[0-9]*' 

答案 24 :(得分:1)

适用于Mac,Linux和Docker容器内部:

$ hostname --ip-address 2> /dev/null || (ifconfig | sed -En 's/127.0.0.1//;s/.*inet (addr:)?(([0-9]*\.){3}[0-9]*).*/\2/p' | awk '{print $ 1;出口}')

也适用于Makefile

LOCAL_HOST := ${shell hostname --ip-address 2> /dev/null || (ifconfig | sed -En 's/127.0.0.1//;s/.*inet (addr:)?(([0-9]*\.){3}[0-9]*).*/\2/p' | awk '{print $1; exit}')}

答案 25 :(得分:1)

对于linux,你需要的是这个命令:

ifconfig $1|sed -n 2p|awk '{ print $2 }'|awk -F : '{ print $2 }'

在你的shell中键入它,你就会知道你的ip。

答案 26 :(得分:1)

这更容易阅读: ifconfig | grep 'inet addr:' |/usr/bin/awk '{print $2}' | tr -d addr:

答案 27 :(得分:1)

如果您安装了npmnodenpm install -g ip && node -e "const ip = require('ip'); console.log(ip.address())"

答案 28 :(得分:1)

如果你知道网络接口(eth0,wlan,tun0等):

ifconfig eth0 | grep addr: | awk '{ print $2 }' | cut -d: -f2

答案 29 :(得分:1)

ifconfig | grep "inet addr:" | grep -v "127.0.0.1" | grep -Eo '[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}'  | head -1

答案 30 :(得分:0)

在Mac上,请考虑以下事项:

scutil --nwi | grep -Eo '[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}'