如果存储方式选错了,未来不仅浪费空间,还可能导致查询性能下降。今天我就带你一步一步搞懂如何在 MySQL 里优雅地存储 IP 地址。
当我们在使用 MySQL 设计表结构的时候都会遇到一个经典问题:“如果要存 IP 地址,到底该用什么数据类型?”
别小看这个问题,存储方式一旦选错,不仅会浪费宝贵的存储空间,未来进行数据查询时也可能拖慢整个系统的速度。接下来,我们就深入探讨一下在 MySQL 中高效存储 IP 地址的正确方法。

为啥不能直接用字符串?
很多小伙伴的第一反应是:IP 地址不就是“192.168.0.1”这种字符串吗?那我直接用VARCHAR(15) 或者 CHAR(15) 存不就行了吗?
表面上来看没啥大的问题,但这样做有几个明显的弊端:
- 空间浪费:字符串比数字类型占用更多字节,还要额外存储长度信息。
- 查询效率低:字符串比较比数字比较要慢,索引效果也差。
- 不方便计算:比如要判断 IP 是否落在某个网段里,字符串处理起来就很麻烦。
所以,直接用字符串存 IP 地址,只能算是最简单但不优雅的做法。
那么,到底该怎么做才比较合适呢?
IPv4:用INT UNSIGNED存整数更高效
IPv4 地址其实就是一个32位的整数,比如:
192.168.0.1 → 3232235521在 MySQL 里我们完全可以用INT UNSIGNED类型来存储,既节省空间(4字节),又利于索引和范围查询。
插入时,用INET_ATON()转换:
INSERT INTO user_logs(ip) VALUES (INET_ATON('192.168.0.1'));查询时,用INET_NTOA()再转回字符串:
SELECT INET_NTOA(ip) FROM user_logs;这样就能同时兼顾空间效率和查询效率。
IPv6:用BINARY(16)存二进制
IPv6 地址更长,是128位,像这样:
2001:0db8:85a3:0000:0000:8a2e:0370:7334如果直接用字符串存,需要 CHAR(39),非常浪费空间。
更推荐的做法是用 BINARY(16) 或 VARBINARY(16) 存二进制形式。
MySQL 也有直接现成的函数:
插入时,用INET6_ATON():
INSERT INTO user_logs(ipv6) VALUES (INET6_ATON('2001:db8::1'));查询时,用INET6_NTOA() 转回:
SELECT INET6_NTOA(ipv6) FROM user_logs;占用空间16字节,支持索引,性能杠杠滴。
那么,如果既要支持 IPv4,又要支持 IPv6 怎么办呢?
这就更简单了,直接:
字段类型设为 VARBINARY(16) 或 BINARY(16),统一用 INET6_ATON()/INET6_NTOA() 存取
MySQL 会自动把 IPv4 地址映射到 IPv6 格式里,做到兼容存储,是不是非常 nice?
来个实战
以下面的建表语句为例:
CREATE TABLE user_logs (
id BIGINT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
ip BINARY(16) NOT NULL COMMENT '存储IPv4/IPv6地址',
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
);
-- 插入IPv4
INSERT INTO user_logs(ip) VALUES (INET6_ATON('192.168.0.1'));
-- 插入IPv6
INSERT INTO user_logs(ip) VALUES (INET6_ATON('2001:db8::1'));
-- 查询
SELECT INET6_NTOA(ip) AS ip_address FROM user_logs;一套方案,IPv4 + IPv6 通吃,相当 OK
