var
WSAData: TWSAData;
SockAddrIn: TSockAddrIn;
Host: PHostEnt;
// Эти переменные объявлены в Winsock.pasbeginif WSAStartup($101, WSAData) = 0 thenbegin
Host := GetHostByName(@Localname[1]);
if Host <> nilthenbegin
SockAddrIn.sin_addr.S_addr := longint(plongint(Host^.h_addr_list^)^);
LocalIP := inet_ntoa(SockAddrIn.sin_addr);
end;
WSACleanUp;
end;
end;
Here is the translation of the text into Russian:
Код на Delphi, который пытается найти локальный IP-адрес машины, на которой он запущен. Вот разбивка каждого элемента:
Переменные
WSAData: переменная типа TWSAData, объявленная в модуле Winsock.pas.
SockAddrIn: переменная типа TSockAddrIn, также объявленная в модуле Winsock.pas.
Host: указатель на структуру PHostEnt, которая представляет информацию о хосте и адресе.
Localname: массив символов, который содержит имя локальной машины.
Код
Код начинает с вызова функции WSAStartup с значением $101 (версия Winsock 2.0) для инициализации библиотеки Winsock. Если успешно, он устанавливает переменную WSAData.
Затем код вызывает функцию GetHostByName, чтобы разрешить имя локальной машины (Localname) до IP-адреса. Результат хранится в переменной Host.
Если разрешение было успешным (т.е., Host не равен nil), код извлекает IP-адрес из структуры Host с помощью поля h_addr_list, которое содержит связанный список IP-адресов, связанных с именем хоста. Первый элемент в этом списке хранится в поле SockAddrIn.sin_addr.S_addr.
Наконец, код вызывает функцию inet_ntoa, чтобы конвертировать бинарное представление IP-адреса (хранящееся в поле SockAddrIn.sin_addr) в строку, читаемую человеком (LocalIP).
Альтернативное решение
Пока что код должен работать для поиска локального IP-адреса, но есть некоторые потенциальные проблемы:
Использование функции GetHostByName не является необходимым, если вам нужно только найти локальный IP-адрес. Вы можете использовать функцию gethostbyname или getaddrinfo вместо нее.
Код предполагает, что первый элемент в поле h_addr_list - это правильный IP-адрес, что может не всегда быть так (например, если машина имеет несколько сетевых интерфейсов).
Код не обрабатывает ошибки должным образом. Хорошая практика - проверять коды ошибок, возвращаемые функциями Winsock, и обрабатывать их соответствующим образом.
Вот обновленная версия кода, которая решает эти проблемы:
var
SockAddrIn: TSockAddrIn;
LocalIP: string;
begin
if WSAStartup($101, WSAData) = 0 then
begin
SockAddrIn.sin_family := PF_INET;
SockAddrIn.sin_port := 0;
SockAddrIn.sin_addr.S_addr := inet_addr('127.0.0.1'); // или используйте gethostbyname('localhost')
LocalIP := inet_ntoa(SockAddrIn.sin_addr);
end
else
begin
// обработка ошибки
end;
WSACleanUp;
end;
В этом коде используется функция inet_addr для установки IP-адреса в конкретное значение (в этом случае, адрес циклического обратного соединения 127.0.0.1) вместо использования GetHostByName. Код также устанавливает поле sin_family в PF_INET, чтобы указать, что мы работаем с IPv4-адресами. Наконец, код использует функцию inet_ntoa для конвертации бинарного IP-адреса в строку, читаемую человеком.
В статье описывается способ нахождения собственного IP-адреса с помощью программирования в языке Delphi и использованием функции GetHostByName из библиотеки Winsock.
Комментарии и вопросы
Материалы статей собраны из открытых источников, владелец сайта не претендует на авторство. Там где авторство установить не удалось, материал подаётся без имени автора. В случае если Вы считаете, что Ваши права нарушены, пожалуйста, свяжитесь с владельцем сайта.