#include <string.h>
#include <sys/socket.h>
#include <sys/ioctl.h>
#include <net/if.h>
#include <stdio.h>
#include <netinet/in.h>
#include <arpa/inet.h>
int main()
{
int inet_sock;
struct ifreq ifr;
inet_sock = socket(AF_INET, SOCK_DGRAM, 0);
//eth0為接口到名稱
strcpy(ifr.ifr_name, "eth1");
//SIOCGIFADDR標志代表獲取接口地址
if (ioctl(inet_sock, SIOCGIFADDR, &ifr) == 0)
perror("ioctl");
printf("%s\n", inet_ntoa(((struct sockaddr_in*)&(ifr.ifr_addr))->sin_addr));
return 0;
}
============================================================
ifreq結構定義在/usr/include/net/if.h,用來配置ip地址,激活接口,配置MTU等接口信息的。
其中包含了一個接口的名字和具體內容——(是個共用體,有可能是IP地址,廣播地址,子網掩碼,MAC號,MTU或其他內容)。
ifreq包含在ifconf結構中。而ifconf結構通常是用來保存所有接口的信息的。
=================================================
用ioctl獲得本地ip地址時要用到兩個結構體ifconf和ifreq,它們對於大多數人
來說都是比較陌生的,這裡給大家一種比較簡單的理解方法,當然只一種幫助
理解的方法,在描述中可能會有一些地方與真實定義有所出入,僅供參考.
首先先認識一下ifconf和ifreq:
//ifconf通常是用來保存所有接口信息的//if.hstruct ifconf { int ifc_len; /* size of buffer */ union { char *ifcu_buf; /* input from user->kernel*/ struct ifreq *ifcu_req; /* return from kernel->user*/ } ifc_ifcu;};#define ifc_buf ifc_ifcu.ifcu_buf /* buffer address */#define ifc_req ifc_ifcu.ifcu_req /* array of structures */ //ifreq用來保存某個接口的信息//if.hstruct ifreq { char ifr_name[IFNAMSIZ]; union { struct sockaddr ifru_addr; struct sockaddr ifru_dstaddr; struct sockaddr ifru_broadaddr; short ifru_flags; int ifru_metric; caddr_t ifru_data; } ifr_ifru;};#define ifr_addr ifr_ifru.ifru_addr#define ifr_dstaddr ifr_ifru.ifru_dstaddr#define ifr_broadaddr ifr_ifru.ifru_broadaddr上邊這兩個結構看起來比較復雜,我們現在把它們簡單化一些:比如說現在我們向實現獲得本地IP的功能。我們的做法是:1. 先通過ioctl獲得本地所有接口的信息,並保存在ifconf中2. 再從ifconf中取出每一個ifreq中表示ip地址的信息具體使用時我們可以認為ifconf就有兩個成員:ifc_len 和 ifc_buf,如圖一所示: ifc_len:表示用來存放所有接口信息的緩沖區長度ifc_buf:表示存放接口信息的緩沖區所以我們需要在程序開始時對ifconf的ifc_led和ifc_buf進行初始化接下來使用ioctl獲取所有接口信息,完成後ifc_len內存放實際獲得的借口信息總長度並且信息被存放在ifc_buf中。如下圖示:(假設讀到兩個接口信息)接下來我們只需要從一個一個的接口信息獲取ip地址信息即可。下面有一個簡單的參考:#include #include #include #include #include in.h>#include <string.h>#include if.h>#include int main(){ int i=0; int sockfd; struct ifconf ifconf; unsigned char buf[512]; struct ifreq *ifreq; //初始化ifconf ifconf.ifc_len = 512; ifconf.ifc_buf = buf; if((sockfd = socket(AF_INET, SOCK_DGRAM, 0))<0) { perror("socket"); exit(1); } ioctl(sockfd, SIOCGIFCONF, &ifconf); //獲取所有接口信息 //接下來一個一個的獲取IP地址 ifreq = (struct ifreq*)buf; for(i=(ifconf.ifc_len/sizeof(struct ifreq)); i>0; i--) {// if(ifreq->ifr_flags == AF_INET){ //for ipv4 printf("name = [%s]\n", ifreq->ifr_name); printf("local addr = [%s]\n", inet_ntoa(((struct sockaddr_in*)&(ifreq->ifr_addr))->sin_addr)); ifreq++;// } } return 0;} 此方法僅供參考,也適用於獲取其他信息。簡述:創建一個套接口。#include <winsock.h>SOCKET PASCAL FAR socket( int af, int type, int protocol);af:一個地址描述。目前僅支持AF_INET格式,也就是說ARPA Internet地址格式。type:新套接口的類型描述。protocol:套接口所用的協議。如調用者不想指定,可用0。注釋 socket()函數用於根據指定的地址族、數據類型和協議來分配一個套接口的描述字及其所用的資源。如果協議protocol未指定(等於0),則使用缺省的連接方式。對於使用一給定地址族的某一特定套接口,只支持一種協議。但地址族可設為AF_UNSPEC(未指定),這樣的話協議參數就要指定了。協議號特定於進行通訊的“通訊域”。支持下述類型描述:類型 解釋SOCK_STREAM 提供有序的、可靠的、雙向的和基於連接的字節流,使用帶外數據傳送機制,為Internet地址族使用TCP。SOCK_DGRAM 支持無連接的、不可靠的和使用固定大小(通常很小)緩沖區的數據報服務,為Internet地址族使用UDP。SOCK_STREAM類型的套接口為全雙向的字節流。對於流類套接口,在接收或發送數據前必需處於已連接狀態。用connect()調用建立與另一套接口的連接,連接成功後,即可用send()和recv()傳送數據。當會話結束後,調用closesocket()。帶外數據根據規定用send()和recv()來接收。實現SOCK_STREAM類型套接口的通訊協議保證數據不會丟失也不會重復。如果終端協議有緩沖區空間,且數據不能在一定時間成功發送,則認為連接中斷,其後續的調用也將以WSAETIMEOUT錯誤返回。SOCK_DGRAM類型套接口允許使用sendto()和recvfrom()從任意端口發送或接收數據報。如果這樣一個套接口用connect()與一個指定端口連接,則可用send()和recv()與該端口進行數據報的發送與接收返回值 若無錯誤發生,socket()返回引用新套接口的描述字。否則的話,返回INVAID_SOCKET錯誤,應用程序可通過WSAGetLastError()獲取相應錯誤代碼。錯誤代碼:WSANOTINITIALISED:在使用此API之前應首先成功地調用WSAStartup()。WSAENETDOWN:WINDOWS套接口實現檢測到網絡子系統失效。WSAEAFNOSUPPORT:不支持指定的地址族。WSAEINPROGRESS:一個阻塞的WINDOWS套接口調用正在運行中。WSAEMFILE:無可用文件描述字。WSAENOBUFS:無可用緩沖區,無法創建套接口。WSAEPROTONOSUPPORT:不支持指定的協議。WSAEPROTOTYPE:指定的協議不適用於本套接口。WSAESOCKTNOSUPPORT:本地址族中不支持該類型套接口。參見 accept(), bind(), connect(), getsockname(), getsockopt(), setsockopt(), listen(), recv(), recvfrom(), select(), send(), sendto(), shutdown(), ioctlsocket().4.2 數據庫例程socket()函數原型:SOCKET WSAAPI socket(int af,int type,int protocol);該函數及參數定義包含在winsock2.h頭文件中,在MSDN中查不到具體參數。參數選項及定義:地址族af:(常用AF_INET實現TCP/UDP協議)#define AF_UNSPEC 0 /* unspecified */#define AF_UNIX 1 /* local to host (pipes, portals) */#define AF_INET 2 /* internetwork: UDP, TCP, etc. */#define AF_IMPLINK 3 /* arpanet imp addresses */#define AF_PUP 4 /* pup protocols: e.g. BSP */#define AF_CHAOS 5 /* mit CHAOS protocols */#define AF_NS 6 /* XEROX NS protocols */#define AF_IPX AF_NS /* IPX protocols: IPX, SPX, etc. */#define AF_ISO 7 /* ISO protocols */#define AF_OSI AF_ISO /* OSI is ISO */#define AF_ECMA 8 /* european computer manufacturers */ #define AF_DATAKIT 9 /* datakit protocols */#define AF_CCITT 10 /* CCITT protocols, X.25 etc */#define AF_SNA 11 /* IBM SNA */ #define AF_DECnet 12 /* DECnet */#define AF_DLI 13 /* Direct data link interface */#define AF_LAT 14 /* LAT */#define AF_HYLINK 15 /* NSC Hyperchannel */#define AF_APPLETALK 16 /* AppleTalk */#define AF_NETBIOS 17 /* NetBios-style addresses */#define AF_VOICEVIEW 18 /* VoiceView */#define AF_FIREFOX 19 /* Protocols from Firefox */#define AF_UNKNOWN1 20 /* Somebody is using this! */#define AF_BAN 21 /* Banyan */#define AF_ATM 22 /* Native ATM Services */#define AF_INET6 23 /* Internetwork Version 6 */#define AF_CLUSTER 24 /* Microsoft Wolfpack */#define AF_12844 25 /* IEEE 1284.4 WG AF */套接字類型type:#define SOCK_STREAM 1 /* stream socket */#define SOCK_DGRAM 2 /* datagram socket */#define SOCK_RAW 3 /* raw-protocol interface */#define SOCK_RDM 4 /* reliably-delivered message */#define SOCK_SEQPACKET 5 /* sequenced packet stream */協議類型protocol:#define IPPROTO_IP 0 /* dummy for IP */#define IPPROTO_ICMP 1 /* control message protocol */#define IPPROTO_IGMP 2 /* internet group management protocol */#define IPPROTO_GGP 3 /* gateway^2 (deprecated) */#define IPPROTO_TCP 6 /* tcp */#define IPPROTO_PUP 12 /* pup */#define IPPROTO_UDP 17 /* user datagram protocol */#define IPPROTO_IDP 22 /* xns idp */#define IPPROTO_ND 77 /* UNOFFICIAL net disk proto */