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



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

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

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

使用grep过滤ifconfig中的 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 -I(hostname -i旧版本的hostname,但请参阅注释)。但是,这仅在 Linux 上。

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

这根本不依赖于DNS,即使/etc/hosts设置不正确(11.0.0.0的简写),它也可以工作:

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

或者避免awk并使用谷歌的公共 DNS8.8.8.8以保持显而易见性:

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

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

hostname -I | cut -d' ' -f1

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

hostname --ip-address

解决方案

$ 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谷歌IP,但可以使用您想要访问的真实IP

例如ip输出:

8.8.8.8 via 192.168.8.254 dev enp0s25 src 192.168.8.16 uid 1000    cache

要提取srcip,sed是最原始且与正则表达式支持最兼容的:

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

例如,最终输出:

192.168.8.16

其他答案

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

我在前面的答案中发现的问题:

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

在 Linux 上

hostname -I

在 macOS 上

ipconfig getifaddr en0

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

已编辑 (2014-06-01 2018-01-092021-07-25)

从前段时间开始,我现在使用更新ip工具。但是在抨击下,我将简单地做:

read -r _{,} gateway _ iface _ ip _ < <(ip r g 1.0.0.0)

然后

printf '%-12s %sn'  gateway $gateway iface $iface ip $ip
gateway      192.168.1.1
iface        eth0
ip           192.168.1.37

从那里,面具是:

while IFS=$' /trn' read lne lip lmask _;do
[ "$lne" = "inet" ] && [ "$lip" = "$ip" ] && mask=$lmask
done < <(ip a s dev $iface)
echo Mask is $mask bits.
Mask is 24 bits.

然后,如果您想将掩码视为 IP:

printf -v msk '%*s' $mask ''
printf -v msk %-32s ${msk// /1}
echo $((msk=2#${msk// /0},msk>>24)).$((msk>>16&255)).$((msk>>8&255)).$((msk&255))
255.255.255.0

编辑 (2014-06-012018-01-09)

为了更强大的配置,每个接口上配置了许多接口和许多IP,我编写了一个纯bash脚本(不基于127.0.0.1),用于基于default route查找正确的接口ip。我把这个脚本贴在这个答案的最底部。

Intro

由于两个操作系统都默认安装了bash,因此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

fancy tidy:

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;
}')

My script (jan 2018):

此脚本将首先找到用于的默认路由接口,然后搜索网关的本地 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=$' :trn' 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: %sn" Interface $iFace Local Ip $myIp 
Gateway $gWay Net mask $netMask Run on mac $runOnMac

仅适用于某些版本的 Ubuntu。 虽然它可能只是告诉你127.0.0.1

hostname  -i

hostname -I

您还可以通过在 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

使用其他一些方法 您可能会在系统上定义多个 IP 地址时发生冲突。 此行始终获取默认使用的 IP 地址。

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

这适用于 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

达尔文笔记本电脑 14.4.0 达尔文内核版本 14.4.0:PDT 2015 年 5 月 28 日星期四 11:35:04; 根: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

Linuxdev-cil.medfx.local 2.6.18-164.el5xen 1 SMP 周四 9 月 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

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

我提取我对这个答案的评论:

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

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

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

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

在作为默认网关的网络中查找此计算机的 IP 地址(例如,排除所有虚拟网络、docker 桥),例如互联网网关、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

我必须补充一下 Collin Anderson 的回答,这种方法还考虑了您是否有两个接口并且它们都显示为 up。

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

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

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

在 Linux 系统上获取本地 ipv4 地址的最短方法:

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

主网络接口 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

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

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

另一个同时适用于Linux和OSX的ifconfig变体:

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

我只使用网络接口名称,我的自定义命令是

[[ $(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。

例如

乌班图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:~#

德比安·杰西

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 ~]#

费多拉 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 ~]#

似乎链接提供的命令ip route get 1 | awk '{print $NF;exit}'更准确,更重要的是,它更短。

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

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

如何在外壳脚本中获取我的外部 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

适用于 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}')}

所有内容都有一个节点包。它是跨平台的,易于使用。

$ npm install --global internal-ip-cli
$ internal-ip
fe80::1
$ internal-ip --ipv4
192.168.0.3

这是一种有争议的方法,但是使用npm进行工具变得越来越流行,不管你喜欢与否。

如果您知道网络接口(eth0、wlan、tun0 等):

ifconfig eth0 | grep addr: | awk '{ print $2 }' | cut -d: -f2
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
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]*' 

对于 linux,你需要的是以下命令:

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

在你的外壳中输入这个,你只会知道你的IP。

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

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

相关内容

  • 没有找到相关文章

最新更新