Как сделать подстановочный поиск для IP-адресов, используя INET_ATON в MySQL?

Я нашел этот метод для хранения IP-адресов в базе данных MySQL как целое число, используя INET_ATON: /questions/43032611/kakoj-tip-dannyih-mysql-ispolzovat-dlya-ip-adresa/43032621#43032621

Поскольку адреса IPv4 имеют длину 4 байта, вы можете использовать INT ( UNSIGNED ), который имеет ровно 4 байта:

`ipv4` INT UNSIGNED

А также INET_ATON а также INET_NTOA конвертировать их:

INSERT INTO `table` (`ipv4`) VALUES (INET_ATON("127.0.0.1"));
SELECT INET_NTOA(`ipv4`) FROM `table`;

Для адресов IPv6 вы можете использовать BINARY вместо:

`ipv6` BINARY(16)

И использовать PHP inet_pton а также inet_ntop для конвертации:

'INSERT INTO `table` (`ipv6`) VALUES ("'.mysqli_real_escape_string(inet_pton('2001:4860:a005::68')).'")'
'SELECT `ipv6` FROM `table`'
$ipv6 = inet_pton($row['ipv6']);

Но как я могу выполнить поиск по шаблону, например, 192.168.%, Используя функцию INET_ATON и PHP ip2long?

2 ответа

Один из хитрых трюков, предлагаемых MySQL, - сдвиг битов. Вы можете использовать его, чтобы увидеть, содержится ли ip в блоке адреса, записанном в нотации cidr. Вы можете использовать этот метод, рассматривая ваши адреса как блок XXXX/16 cidr.

set @cidr_block:='10.20.30.40/16';

select inet_ntoa(inet_aton(substring_index(@cidr_block,'/',1))>>(32-substring_index(@cidr_block,'/',-1))<<(32-substring_index(@cidr_block,'/',-1))) as first_ip,
                 inet_aton(substring_index(@cidr_block,'/',1))>>(32-substring_index(@cidr_block,'/',-1))<<(32-substring_index(@cidr_block,'/',-1))  as first_ip_num,
        inet_ntoa((((inet_aton(substring_index(@cidr_block,'/',1))>>(32-substring_index(@cidr_block,'/',-1)))+1)<<(32-substring_index(@cidr_block,'/',-1)))-1) as last_ip,
                 (((inet_aton(substring_index(@cidr_block,'/',1))>>(32-substring_index(@cidr_block,'/',-1)))+1)<<(32-substring_index(@cidr_block,'/',-1)))-1  as last_ip_num
;
+-----------+--------------+---------------+-------------+
| first_ip  | first_ip_num | last_ip       | last_ip_num |
+-----------+--------------+---------------+-------------+
| 10.20.0.0 |    169082880 | 10.20.255.255 |   169148415 |
+-----------+--------------+---------------+-------------+
1 row in set (0.00 sec)

Ярлык, чтобы увидеть, находится ли ip в блоке адреса - просто просейте и адрес cidr, и ip, чтобы увидеть, совпадают ли они. Конечно, это будет сканирование таблицы, если применять к сохраненным значениям.

select inet_aton('127.0.0.1')>>16 = inet_aton('127.0.10.20')>>16 as `1 = true`;
+----------+
| 1 = true |
+----------+
|        1 |
+----------+
1 row in set (0.00 sec)

select inet_aton('127.0.0.1')>>16 = inet_aton('127.10.10.20')>>16 as `0 =  false`;
 +-----------+
 | 0 = false |
 +-----------+
 |         0 |
 +-----------+
 1 row in set (0.00 sec)

Поиск с подстановочными знаками работает со строками и, поскольку он не может извлекать выгоду из индексов, он имеет тенденцию быть чрезвычайно медленным.

Если вы храните IP-адреса в нормализованном представлении, предназначенном для машин (по сравнению с удобочитаемым точечным обозначением), вы можете рассматривать их, как если бы они были числами, использовать множество стандартных операторов и эффективно использовать индексы. Пример:

SELECT *
FROM foo
WHERE dot_notation LIKE '192.168.%';

... можно переписать как:

SELECT *
FROM foo
WHERE as_integer BETWEEN INET_ATON('192.168.0.0') AND INET_ATON('192.168.255.255');

Даже эти INET_ATON() экземпляры для простоты чтения, вы можете просто ввести полученное целое число. Если вы используете PHP, это тривиально, потому что вы можете перенести его на PHP:

$sql = 'SELECT *
    FROM foo
    WHERE as_integer BETWEEN ? AND ?';
$params = [
   // Not sure whether you still need the sprintf('%u') trick in 64-bit PHP
   ip2long('192.168.0.0'), ip2long('192.168.255.255')
];

Я не могу проверить это прямо сейчас, но я понимаю, что это должно работать и с IPv6.

Другие вопросы по тегам