linux下的socket编程(3)
server端的简单示例:
经过前面的client端的学习,我们已经知道了如何创建socket,所以接下来就是去绑定他到具体的一个端口上去。
绑定socket到一个端口上去
bind()函数可以将socket绑定一个端口上,client可以通过这个端口发起请求,端口对应的socket便会与client端的socket连接。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
|
#
include
<stdio.h>
#
include
<stdllib.h>
#
include
<sys/socket.h>
#
include
<sys/types.h>
#Include<arpa/inet.h>
int
main()
{
int
socket_desc;
struct sockaddr_in server;
socket_desc = socket(AF_INET,SOCK_STREAM,
0
);
if
(-
1
==socket_desc)
{
perror(
"socket create error\n"
);
exit(
1
);
}
//监听服务器自身
server.sin_family=AF_INET;
server.sin_port = htons(
8888
);
server.sin_addr.s_addr = INADDR_ANY;
//绑定到端口
if
(bind(socket_desc,(struct sockaddr* )&server,sizeof(server))<
0
)
{
perror(
"bind failed\n"
);
exit(
1
);
}
printf(
"bind success\n"
);
close(socket_desc);
return
0
;
}
|
对于socket绑定到一个明确的端口上,我们接下来要做的就是接受这个端口下面的所有数据。。
通过上面的实现,我们可以看出一个端口只能被一个socket使用。
监听端口:
在绑定完成socket与端口之后,我们还需要去监听端口。为此,我们需要将socket设置在被监听的状态。listen()将被用来将socket设置为被监听的模式下。
listen( socket_desc, 3);
listen(int sockfd,int backlog);可以将socket处于监听的状态下
接收请求建立连接:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
|
#
include
<stdio.h>
#
include
<stdlib.h>
#
include
<sys/socket.h>
#
include
<arpa/inet.h>
int
main()
{
int
sock_desc,new_socket,sockaddr_size;
struct sockaddr_in server,client;
//创建socket
sock_desc = socket(AF_INET,SOCK_STREAM,
0
);
if
(-
1
==sock_desc)
{
perror(
"cannot create socket\n"
);
exit(
1
);
}
server.sin_family = AF_INET;
server.sin_port = htons(
8888
);
server.sin_addr.s_addr = INADDR_ANY;
//绑定
if
(bind(sock_desc,(struct sockaddr*)&server,sizeof(server))<))
{
perror(
"bind error\n"
);
exit(
1
);
}
//监听
listen(sock_desc,
5
);
puts(
"waiting for incoming connecting \n"
);
//接受链接
sockadddr_size = sizeof(sruct sockaddr_in );
new_socket = accept(sock_desc,(struct sockaddr *)&client,(socklen_t * )&sockaddr_size);
if
(new_socket<
0
)
{
perror(
"accept failed"
);
exit(
1
);
}
puts(
"connecting accept "
);
//先关闭socket_desc 产生的new_socket
close(new_socket);
//z在关闭sock_desc
close(sock_desc);
return
0
;
}
|
运行上述代码:输出:waiting for incoming connecions.
现在代码已经正常跑起来了,并且等待请求连接。在另外一个终端内,我们发起一个请求:
teltnet 127.0.0.1 8888
在当前这个终端内将会输出:
trying 127.0.0.1
connected to loaclhost。
Escape character is ;;
connection closed by foreign host
同时在之前的终端中,server会输出:
waiting for incoming connecions.
connection accepted
便可以看到,server已经正确接收了client的连接请求并建立了连接,只是没有了后续操作,主机紧接着关闭了这个链接。
连接建立之后便可以顺利地进行双方的通信,这部分的send与recv操作完全一样。
另外, 服务端获取客户端的ip地址:
由前面能够知道accept()返回的是结构体sockaddr_in ,由此很容易得知client的ip和端口信息。
1
2
|
char * client_ip = inet_ntoa(client.sin_addr);
int
client_port = ntohs(client.sin_port);
|