這兩天盡忙着看winsock了,玩過它的人都知道正確的填寫IP地址是很重要的一個環節。所謂填寫IP地址就是指在SOCKADDR_IN結構體中填寫IP地址:
struct sockaddr_in {
short sin_family;
u_short sin_port;
struct in_addr sin_addr;
char sin_zero[8];
};
在這裏已經定義了 #typedef sockaddr_in SOCKADDR_IN 。該結構中的一個in_addr 結構體 sin_addr 就是要填寫的IP地址。in_addr 結構的定義如下(MSDN):
typedef struct in_addr {
union {
struct { u_char s_b1,s_b2,s_b3,s_b4; } S_un_b;
struct { u_short s_w1,s_w2; } S_un_w;
u_long S_addr;
} S_un;
} in_addr;
於是我如同往常填寫信息如下:
SOCKADDR_IN serverAdd;
unsigned long ipadd;
m_ServerIp.GetAddress( ipadd ); // m_ServerIp 是一個CIPAddressCtrl 控件,用來讀取IP地址
serverAdd.sin_family = AF_INET; // IP地址家族
serverAdd.sin_addr.s_addr = htonl(ipadd) ; // 填寫IP
serverAdd.sin_port = htons(m_ServerPort); // 填寫端口
這裏的serverAdd.sin_addr.s_addr = htonl(ipadd) ; 引起了我的困惑,本應該是serverAdd.sin_addr.S_un.S_addr = htonl(ipadd) ; 這樣纔對嘛,怎麼就直接訪問union裏面的成員呢S_addr(當時還沒有看出大小寫問題)? 看了半天,也在網上查了半天沒有結果。最後眼睛就直勾勾的看着這行的最後一個s_addr ,突然發現s原來是小寫的,呵呵,查看定義(winsock2.h):
struct in_addr {
union {
struct { u_char s_b1,s_b2,s_b3,s_b4; } S_un_b;
struct { u_short s_w1,s_w2; } S_un_w;
u_long S_addr;
} S_un;
#define s_addr S_un.S_addr
/* can be used for most tcp & ip code */
#define s_host S_un.S_un_b.s_b2
/* host on imp */
#define s_net S_un.S_un_b.s_b1
/* network */
#define s_imp S_un.S_un_w.s_w2
/* imp */
#define s_impno S_un.S_un_b.s_b4
/* imp # */
#define s_lh S_un.S_un_b.s_b3
/* logical host */
};
看到第7行的那句宏定義了嗎? 終於明白了,呵呵。此時猛然記起某本書上的心經:不要定義只有大小寫區分的變量如:student, Student, 或者是隻有s區別的如:student, students。當時還不覺得什麼,現在看起來真是太重要了,尤其是對像我這麼近視的人,呵呵。
最後說一句,MSDN也太不夠意思了也不把代碼寫完整點,不過索性還好可以在IDE中迅速找到定義的代碼。不過對於在使用記事本寫程序的兄弟姐妹們來說,可能就不容易了,呵呵。
來自:http://blog.csdn.net/chaostring/article/details/3102948