学习思路
以下是我对学习网络编程的一个简单的学习思路,之后我将会按照这个计划去逐步学习网络编程相关的知识。
- step 1. 原生php实现TCP Server -> 原生php实现http协议 -> 掌握tcpdump的使用 -> 深刻理解tcp连接过程
- step 2. 原生php实现多进程webserver 2.1 引入I/O多路复用 2.2 引入php协程(yield) 2.3 对比 I/O多路复用版本 和 协程版本的性能差异
- step 3. 实现简单的go web框架
- step 4. php c扩展实现简单的webserver
什么我会选择用php去学习网络编程?因为对于我来说,php算是最熟悉的,其次php相对来说简单些,同时php自身也有相应的函数支持。
我们今天先开始第一部分的学习。
正文
我们先简单回顾下php作为后端语言的常见的交互方式过程:
client –(protocol:http)–> nginx –(protocol:fastcgi)–> php-fpm –(interface:sapi)–> php
在这里nginx充当的web server和反向代理server的角色,把http协议转换成了fastcgi协议。看到这里有些小伙伴可能会说了:“如果php自己直接处理http请求,不就可以不用nginx&php-fpm了么?”遗憾的是原生php木有实现http协议(是吧,欢迎纠错)。
然后可能又有小伙伴说:“原生php不是支持tcp协议么?nginx把http请求代理成tcp协议不就可以不用php-fpm了吗。”,嗯,是的,没错。这位小伙伴的描述的交互过程如下
client –(protocol:http)–> nginx –(protocol:tcp)–> php
这样看起来是没啥问题,很不错的想法,但是理论来说还是没有实现http协议,接收到的内容应该还是一坨字符串。我们马上来试一下:
STEP 1: 起一个NGINX服务
STEP 2: PHP简单实现一个TCP SERVER,简单的代码如下
<?php
$server = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
socket_bind($server, '127.0.0.1', '8889');
socket_listen($server);
while (true) {
$client = socket_accept($server);
if (! $client) {
continue;
}
$request = socket_read($client, 1024);
// 查看接收到的内容
var_dump($request);
socket_close($client);
}
访问结果
所以我们就需要实现http协议,既然都实现了http协议,那就可以直接使用http作为web server了。
client –(protocol:http)–> php
接着我们看看如何用php创建一个简单的TCP Server过程如下:
主要涉及的PHP函数如下
socket_create
socket_listen
socket_accept
socket_recv || socket_read
socket_write
socket_close
代码如下
<?php
$server = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
socket_bind($server, '127.0.0.1', '8889');
socket_listen($server);
while (true) {
// accept
$client = socket_accept($server);
if (! $client) {
continue;
}
$request = socket_read($client, 1024);
socket_close($client);
echo socket_strerror(socket_last_error($server)) . "\n";
}
没毛病,TCP Server起来了。
原生PHP实现HTTP协议
上面简单的TCP Server基本出来了,我们需要让php直接成为一个Web Server,想一想Web Server是基于HTTP协议的,HTTP协议又是基于TCP协议实现的。也就是说我们在上面的TCP Server基础上实现下HTTP协议即可。我们改进下流程图加入HTTP部分(橙黄色),如下
实现HTTP协议的过程其实就是:
- 能读懂发来请求的信息
- 能返回给浏览器等客户端它们能懂的信息
协议无非就是双方协定好的规范,一样在HTTP/1.1中 请求&响应的格式基本如下
请求:
<HTTP Method> <url> <HTTP Version>
<KEY>:<VALUE>\r\n
...
\r\n
响应:
<HTTP Version> <HTTP Status> <HTTP Status Description>
<KEY>:<VALUE>\r\n
...
\r\n
所以简单来说,我们的php代码只要按照上面的规范解析和返回出对应的内容即可,简单的代码例子如下:
<?php
$server = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
socket_bind($server, '127.0.0.1', '8889');
socket_listen($server);
$http = new HttpProtocol();
while(true){
$client = socket_accept($server);
if (!$client){
continue;
}
$request = socket_read($client, 1024);
$http->response("hello word");
socket_write($client, $http->resposeData);
socket_close($client);
}
class HttpProtocol {
private $header=[
"http"=>"HTTP/1.1 200 OK",
"content-type" => "Content-Type:text/html",
"server" => "Server: php/0.0.1",
];
public $resposeData="";
public function response($msg){
$count = count($this->header);
$finalData="";
foreach($this->header as $key => $value){
$finalData.=$value."\r\n";
}
$this->resposeData = $finalData."\r\n".$msg;
}
}
?>
运行代码:
结果展示:
参考网址[http://tigerb.cn/2018/11/24/php-network-programming/]
标签:http,入门,编程,TCP,server,client,php,socket From: https://blog.51cto.com/u_11635800/5877050