• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            chaosuper85

            C++博客 首頁 新隨筆 聯(lián)系 聚合 管理
              118 Posts :: 0 Stories :: 3 Comments :: 0 Trackbacks
            域名服務(wù)(DNS)   如果你不知道 DNS 的意思,那么我告訴你,它代表域名服務(wù)(Domain Name Service)。它主要的功能是:你給它一個容易記憶的某站點的地址, 它給你 IP 地址(然后你就可以使用 bind(), connect(), sendto() 或者其它 函數(shù)) 。當(dāng)一個人輸入:    $ telnet whitehouse.gov telnet 能知道它將連接 (connect()) 到 "198.137.240.100"。 但是這是如何工作的呢? 你可以調(diào)用函數(shù) gethostbyname(): #include   struct hostent *gethostbyname(const char *name); 很明白的是,它返回一個指向 struct hostent 的指針。這個數(shù)據(jù)結(jié)構(gòu) 是這樣的:    struct hostent {    char *h_name;    char **h_aliases;    int h_addrtype;    int h_length;    char **h_addr_list;    };    #define h_addr h_addr_list[0] 這里是這個數(shù)據(jù)結(jié)構(gòu)的詳細(xì)資料: struct hostent:   h_name – 地址的正式名稱。   h_aliases – 空字節(jié)-地址的預(yù)備名稱的指針。   h_addrtype –地址類型; 通常是AF_INET。   h_length – 地址的比特長度。   h_addr_list – 零字節(jié)-主機(jī)網(wǎng)絡(luò)地址指針。網(wǎng)絡(luò)字節(jié)順序。   h_addr - h_addr_list中的第一地址。 gethostbyname() 成功時返回一個指向結(jié)構(gòu)體 hostent 的指針,或者 是個空 (NULL) 指針。(但是和以前不同,不設(shè)置errno,h_errno 設(shè)置錯 誤信息。請看下面的 herror()。) 但是如何使用呢? 有時候(我們可以從電腦手冊中發(fā)現(xiàn)),向讀者灌輸 信息是不夠的。這個函數(shù)可不象它看上去那么難用。 這里是個例子: #include   #include   #include   #include   #include   #include int main(int argc, char *argv[])    {    struct hostent *h; if (argc != 2) { /* 檢查命令行 */    fprintf(stderr,"usage: getip address\n");    exit(1);    } if ((h=gethostbyname(argv[1])) == NULL) { /* 取得地址信息 */    herror("gethostbyname");    exit(1);    } printf("Host name : %s\n", h->h_name);   printf("IP Address : %s\n",inet_ntoa(*((struct in_addr *)h->h_addr))); return 0;    } 在使用 gethostbyname() 的時候,你不能用 perror() 打印錯誤信息 (因為 errno 沒有使用),你應(yīng)該調(diào)用 herror()。 相當(dāng)簡單,你只是傳遞一個保存機(jī)器名的字符串(例如 "whitehouse.gov") 給 gethostbyname(),然后從返回的數(shù)據(jù)結(jié)構(gòu) struct hostent 中獲取信息。 唯一也許讓人不解的是輸出 IP 地址信息。h->h_addr 是一個 char *, 但是 inet_ntoa() 需要的是 struct in_addr。因此,我轉(zhuǎn)換 h->h_addr 成 struct in_addr *,然后得到數(shù)據(jù)。 -------------------------------------------------------------------------------- 客戶-服務(wù)器背景知識   這里是個客戶--服務(wù)器的世界。在網(wǎng)絡(luò)上的所有東西都是在處理客戶進(jìn) 程和服務(wù)器進(jìn)程的交談。舉個telnet 的例子。當(dāng)你用 telnet (客戶)通過23 號端口登陸到主機(jī),主機(jī)上運(yùn)行的一個程序(一般叫 telnetd,服務(wù)器)激活。 它處理這個連接,顯示登陸界面,等等。 圖2:客戶機(jī)和服務(wù)器的關(guān)系 圖 2 說明了客戶和服務(wù)器之間的信息交換。 注意,客戶--服務(wù)器之間可以使用SOCK_STREAM、SOCK_DGRAM 或者其它(只要它們采用相同的)。一些很好的客戶--服務(wù)器的例子有 telnet/telnetd、 ftp/ftpd 和 bootp/bootpd。每次你使用 ftp 的時候,在遠(yuǎn) 端都有一個 ftpd 為你服務(wù)。 一般,在服務(wù)端只有一個服務(wù)器,它采用 fork() 來處理多個客戶的連 接。基本的程序是:服務(wù)器等待一個連接,接受 (accept()) 連接,然后 fork() 一個子進(jìn)程處理它。這是下一章我們的例子中會講到的。 -------------------------------------------------------------------------------- 簡單的服務(wù)器   這個服務(wù)器所做的全部工作是在流式連接上發(fā)送字符串 "Hello, World!\n"。你要測試這個程序的話,可以在一臺機(jī)器上運(yùn)行該程序,然后 在另外一機(jī)器上登陸:    $ telnet remotehostname 3490 remotehostname 是該程序運(yùn)行的機(jī)器的名字。 服務(wù)器代碼: #include   #include   #include   #include   #include   #include   #include   #include #define MYPORT 3490 /*定義用戶連接端口*/ #define BACKLOG 10 /*多少等待連接控制*/ main()    {    int sockfd, new_fd; /* listen on sock_fd, new connection on new_fd */    struct sockaddr_in my_addr; /* my address information */    struct sockaddr_in their_addr; /* connector's address information */    int sin_size; if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {    perror("socket");    exit(1);    } my_addr.sin_family = AF_INET; /* host byte order */    my_addr.sin_port = htons(MYPORT); /* short, network byte order */    my_addr.sin_addr.s_addr = INADDR_ANY; /* auto-fill with my IP */    bzero(&(my_addr.sin_zero),; /* zero the rest of the struct */ if (bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr))== -1) {    perror("bind");    exit(1);    } if (listen(sockfd, BACKLOG) == -1) {    perror("listen");    exit(1);    } while(1) { /* main accept() loop */    sin_size = sizeof(struct sockaddr_in);    if ((new_fd = accept(sockfd, (struct sockaddr *)&their_addr, \    &sin_size)) == -1) {    perror("accept");    continue;    }    printf("server: got connection from %s\n", \    inet_ntoa(their_addr.sin_addr));    if (!fork()) { /* this is the child process */    if (send(new_fd, "Hello, world!\n", 14, 0) == -1)    perror("send");    close(new_fd);    exit(0);    }    close(new_fd); /* parent doesn't need this */ while(waitpid(-1,NULL,WNOHANG) > 0); /* clean up child processes */    }    } 如果你很挑剔的話,一定不滿意我所有的代碼都在一個很大的main() 函數(shù)中。如果你不喜歡,可以劃分得更細(xì)點。 你也可以用我們下一章中的程序得到服務(wù)器端發(fā)送的字符串。 -------------------------------------------------------------------------------- 簡單的客戶程序   這個程序比服務(wù)器還簡單。這個程序的所有工作是通過 3490 端口連接到命令行中指定的主機(jī),然后得到服務(wù)器發(fā)送的字符串。 客戶代碼: #include   #include   #include   #include   #include   #include   #include   #include #define PORT 3490 /* 客戶機(jī)連接遠(yuǎn)程主機(jī)的端口 */ #define MAXDATASIZE 100 /* 每次可以接收的最大字節(jié) */ int main(int argc, char *argv[])    {    int sockfd, numbytes;    char buf[MAXDATASIZE];    struct hostent *he;    struct sockaddr_in their_addr; /* connector's address information */ if (argc != 2) {    fprintf(stderr,"usage: client hostname\n");    exit(1);    } if ((he=gethostbyname(argv[1])) == NULL) { /* get the host info */    herror("gethostbyname");    exit(1);    } if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {    perror("socket");    exit(1);    } their_addr.sin_family = AF_INET; /* host byte order */   their_addr.sin_port = htons(PORT); /* short, network byte order */   their_addr.sin_addr = *((struct in_addr *)he->h_addr);   bzero(&(their_addr.sin_zero),; /* zero the rest of the struct */ if (connect(sockfd, (struct sockaddr *)&their_addr,sizeof(struct sockaddr)) == -1) {    perror("connect");    exit(1);    } if ((numbytes=recv(sockfd, buf, MAXDATASIZE, 0)) == -1) {    perror("recv");    exit(1);    } buf[numbytes] = '\0'; printf("Received: %s",buf); close(sockfd); return 0;    } 注意,如果你在運(yùn)行服務(wù)器之前運(yùn)行客戶程序,connect() 將返回 "Connection refused" 信息,這非常有用。 --------------------------------------------------------------------------------
            posted on 2010-01-16 21:46 chaosuper 閱讀(148) 評論(0)  編輯 收藏 引用

            只有注冊用戶登錄后才能發(fā)表評論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            久久AV高潮AV无码AV| 久久91精品国产91| 国产成人久久精品激情| 久久精品www| 一本久久综合亚洲鲁鲁五月天亚洲欧美一区二区 | 中文字幕亚洲综合久久菠萝蜜 | 久久精品一区二区三区AV| 青草国产精品久久久久久| 久久精品成人国产午夜| 欧美精品九九99久久在观看| 久久精品麻豆日日躁夜夜躁| 欧美一级久久久久久久大片| 午夜欧美精品久久久久久久| 伊人久久大香线焦综合四虎| 国产aⅴ激情无码久久| 国产精品女同一区二区久久| 国产一区二区久久久| 一本大道久久a久久精品综合| 久久天天躁狠狠躁夜夜avapp| 激情伊人五月天久久综合| 亚洲欧洲中文日韩久久AV乱码| 99久久婷婷国产综合亚洲| 久久久久亚洲国产| 国产99久久久国产精品~~牛| 新狼窝色AV性久久久久久| 久久青青草原精品国产不卡| 久久久久久a亚洲欧洲aⅴ| 亚洲精品无码久久千人斩| 亚洲精品99久久久久中文字幕| 99久久99久久精品国产片果冻| 久久久无码精品亚洲日韩蜜臀浪潮| 亚洲欧洲精品成人久久奇米网| 亚洲综合婷婷久久| 一本伊大人香蕉久久网手机| 久久精品国产亚洲av影院| 国内精品综合久久久40p| 伊人久久成人成综合网222| AA级片免费看视频久久| 97久久天天综合色天天综合色hd | 97久久久精品综合88久久| 欧美大香线蕉线伊人久久|