参考资料:《Boost Asio C++网络编程》第五章
学会网络编程之后,理所当然会有这样的想法:在单位电脑上部署自己写的服务段软件,在家里电脑上部署客户端,从而远程办公。但是实际操作下来可以发现,两台电脑上的ip地址,根本不能建立socket直连,这是为什么呢。
对于普通网络用户,电脑入网都分配的是内网IP。尽管IPv6已经出来多年,目前互联网公网IP普遍还在IPv4阶段,可利用的IP地址相当有限。宽带商本身仅仅能拿到少量IPv4地址的使用权,不会给每个个人用户配置单独的公网IP,为了让每个用户上网,把用户都纳入局域网中分配局域网IP;建立网关设备作为门户,收集每个用户的socket请求然后变为门户IP的数据包,转发到公网上去,建立长期连接;用户连接单独开启一个端口用于标识数据转发地址;但是不允许外网主动申请连接,如图1
既然内网之间无法连接,内网的网络用户,需要其他方式才能进行互联,实现聊天、传文件、远程控制等功能。上述可知,内网能主动连接公网服务器建立长期连接,但不能反向,那么不同内网用户可以预先向同一公网服务器建立长期连接,然后公网服务器进行转发,间接实现内网电脑之间的互联,如图2:
一般情况下,一台公网服务器就会占用一个公网IP,如果服务量巨大,那这台服务器会崩溃,需要将压力分散到多个服务器中,一种可用策略如图3:
代理服务器很简单,需要配备两个端口,一个接收客户端sock连接,一个接收服务端socket连接。端口可以设置reuse属性,进行多进程监听,但是最好一个端口仅对应一个服务。
class proxy : public boost::enable_shared_from_this<proxy> {
proxy(ip::tcp::endpoint ep_client, ip::tcp::endpoint ep_serv
er) : ... {}
void on_start() {
do_read(client_, buff_client_);
do_read(server_, buff_server_);
}
这里是异步调用,直接启动client和server的监听即可,后续回调函数需要将sock作为参数,好识别究竟是客户端还是服务端发出了响应:
void do_read(ip::tcp::socket & sock, char* buff) {
async_read(sock, buffer(buff, max_msg), MEM_FN3(read_com
plete,ref(sock),_1,_2), MEM_FN3(on_read,ref(sock),_1,_2));
}
注意这里使用std/boost::ref(),显示调用std/boost::bind执行引用参数传递,因为bind默认参数列表都是值传递。
void on_read(ip::tcp::socket & sock, const error_code& err,
size_t bytes) {
char * buff = &sock == &client_ ? buff_client_ : buff_se
rver_;
do_write(&sock == &client_ ? server_ : client_, buff, by
tes);
}
实际操作很简单,如果是client的sock读取进来,那么就向server的sock写入同样的数据;反之亦然。代理的一大作用在这里,就是可以把client/server传入的数据进行修改,再用不同的sock传递出去。