>PHP 프레임워크 >Workerman >Workerman 기반 실시간 푸시(Ajax 폴링 포기)

Workerman 기반 실시간 푸시(Ajax 폴링 포기)

藏色散人
藏色散人앞으로
2020-01-20 13:43:593366검색

다음 workerman 튜토리얼 칼럼에서는 Workerman을 기반으로 실시간 Push를 구현하고 Ajax Polling을 포기하는 방법을 소개하겠습니다. 도움이 필요한 친구들에게 도움이 되길 바랍니다!

Workerman 기반 실시간 푸시(Ajax 폴링 포기)

먼저 다음 사항에 대해 이야기해 보겠습니다.

TCP/IP

TCP/IP는 네트워크 계층, 전송 계층 및 애플리케이션 계층의 세 가지 계층으로 나눌 수 있는 프로토콜 그룹입니다.

네트워크 계층에는 IP 프로토콜, ICMP 프로토콜, ARP 프로토콜, RARP 프로토콜 및 BOOTP 프로토콜이 있습니다.

전송 계층에는 TCP 프로토콜과 UDP 프로토콜이 있습니다.

응용 프로그램 계층에는 다음이 있습니다.

TCP에는 FTP, HTTP, TELNET, SMTP 및 기타 프로토콜이 포함됩니다.

UDP에는 DNS, TFTP 및 기타 프로토콜이 포함됩니다.

짧은 연결

연결->데이터 전송- >연결 닫기

HTTP는 브라우저와 서버가 HTTP 작업을 수행할 때마다 연결이 설정되지만 작업이 완료되면 연결이 중단됩니다.

다음과 같이도 말할 수 있습니다. 짧은 연결은 SOCKET 연결이 전송되고 데이터가 수신된 후 즉시 연결이 끊어지는 것을 의미합니다.

긴 연결

연결->데이터 전송->연결 유지->데이터 전송->. . . ->연결을 닫습니다.

긴 연결이란 SOCKET 연결을 설정한 후 사용 여부에 관계없이 연결 상태를 유지하지만 보안이 취약하다는 의미입니다.

Http 긴 연결

HTTP도 긴 연결을 설정할 수 있습니다. Connection:keep-alive를 사용하세요. HTTP 1.1은 기본적으로 영구 연결을 사용합니다. HTTP1.1과 HTTP1.0을 비교하면 가장 큰 차이점은 지속적인 연결 지원이 추가되었다는 점이지만(최신 http1.0에서는 명시적으로 연결 유지를 지정할 수 있는 것 같습니다) 여전히 상태가 없거나 신뢰할 수 없습니다.

긴 연결과 짧은 연결은 언제 사용하나요?

긴 연결은 주로 빈번한 작업, 지점 간 통신에 사용되며 연결 수가 너무 많아서는 안됩니다. 각 TCP 연결에는 3단계 핸드셰이크가 필요하며, 각 작업을 먼저 연결한 후 실행하면 처리 속도가 많이 떨어지므로 각 작업 후에 연결이 끊어지지 않고 데이터 패킷이 직접 전송됩니다. 첫 번째 처리는 괜찮습니다. TCP 연결을 설정할 필요가 없습니다. 예를 들어 데이터베이스 연결에는 긴 연결이 사용됩니다. 짧은 연결로 자주 통신하면 소켓 오류가 발생하고 소켓을 자주 생성하는 것도 리소스 낭비입니다.

그리고 WEB 웹 사이트와 같은 http 서비스는 일반적으로 짧은 링크를 사용합니다. 왜냐하면 긴 연결은 서버의 특정 리소스를 소비하기 때문이며, 수천 또는 심지어 수억 개의 클라이언트 연결이 빈번한 WEB 웹 사이트와 마찬가지로 짧은 링크가 사용됩니다. 연결을 사용하면 일부 리소스가 절약되며 동시에 수천 명의 사용자가 있는 경우 각 사용자가 연결을 점유하는 것이 가능합니다. 따라서 동시성 양은 크지만, 빈번한 작업이 필요하지 않은 경우 각 사용자는 짧은 연결을 사용해야 합니다.

workerman뭔데? Workerman은 순수하게 PHP로만 개발된 오픈 소스 고성능 PHP 소켓 서버 프레임워크입니다. 모바일 앱, 모바일 통신, WeChat 애플릿, 모바일 게임 서버, 온라인 게임, PHP 채팅방, 하드웨어 통신, 스마트 홈, 차량 인터넷, 사물 인터넷 및 기타 분야의 개발에 널리 사용됩니다. TCP 긴 연결을 지원하고 Websocket, HTTP 및 기타 프로토콜을 지원하며 사용자 정의 프로토콜을 지원합니다. 비동기 Mysql, 비동기 Redis, 비동기 Http, 비동기 메시지 대기열 등과 같은 많은 고성능 구성 요소가 있습니다.

본점으로 들어가겠습니다. 실시간 통신을 달성하기 위해 우리는 그림과 같이 종종 ajax 폴링 메커니즘을 사용합니다.

Workerman 기반 실시간 푸시(Ajax 폴링 포기)

workerman 방법을 사용하여 나중에 구현할 수 있습니다. 또한 공식 매뉴얼에 나와 있듯이 TP에 의해 작성되었습니다

위 그림과 같이 다른 mvc 프레임워크와 결합하는 것이 좋습니다(예: ThinkPHP):

1 ThinkPHP와 Workerman은 독립적으로 배포할 수 있는 두 개의 독립적인 시스템입니다. (다른 서버에 배포 가능) 서로 간섭하지 않고.

2. ThinkPHP는 HTTP 프로토콜을 사용하여 브라우저에 렌더링하고 표시할 웹 페이지를 제공합니다.

3. ThinkPHP에서 제공하는 페이지의 js는 websocket 연결을 시작하고 워커맨에 연결합니다.

4. 연결 후 데이터 패킷(사용자 이름, 비밀번호 또는 일종의 토큰 문자열 포함)이 워커맨에게 전송됩니다. 웹소켓 연결이 속한 사용자를 확인합니다.

5. ThinkPHP는 데이터를 브라우저에 푸시해야 하는 경우에만 작업자의 소켓 인터페이스를 호출하여 데이터를 푸시합니다.

6. 나머지 요청은 여전히 ​​ThinkPHP의 원래 HTTP 방식에 따라 처리됩니다.

요약:

Workerman을 브라우저에 푸시할 수 있는 채널로 사용하고, 데이터를 브라우저에 푸시해야 하는 경우에만 Workerman 인터페이스를 호출하여 푸시를 완료하세요. 비즈니스 로직은 ThinkPHP에서 모두 완성됩니다.

자, 여기서는 Workerman 컨테이너를 실행합니다. CLI 모드에서 실행 중이라는 점에 유의하세요

Workerman 기반 실시간 푸시(Ajax 폴링 포기)

그러면 프로젝트 정보 수신에서 이렇게 플레이하고 코드를 첨부하겠습니다

<script>
    // 连接服务端
    var socket = io(&#39;http://127.0.0.1:2120&#39;);
    // uid可以是自己网站的用户id,以便针对uid推送
    uid = 123;
    // socket连接后以uid登录
    socket.on(&#39;connect&#39;, function(){
    socket.emit(&#39;login&#39;, uid);
    });
    // 后端推送来消息时
    socket.on(&#39;new_msg&#39;, function(msg){
        console.log("收到消息:"+msg);  //自己业务逻辑处理
    });
    
</script>

다음으로 사용자가 사용자에게 정보를 보낼 때

// 指明给谁推送,为空表示向所有在线用户推送
        $to_uid = "123";
        // 推送的url地址
        $push_api_url = "http://127.0.0.1:2121/";
        $post_data = array(
           "type" => "publish",
           "content" => "数据",
           "to" => $to_uid, 
        );
        $ch = curl_init ();
        curl_setopt ( $ch, CURLOPT_URL, $push_api_url );
        curl_setopt ( $ch, CURLOPT_POST, 1 );
        curl_setopt ( $ch, CURLOPT_HEADER, 0 );
        curl_setopt ( $ch, CURLOPT_RETURNTRANSFER, 1 );
        curl_setopt ( $ch, CURLOPT_POSTFIELDS, $post_data );
        curl_setopt ($ch, CURLOPT_HTTPHEADER, array("Expect:"));
        $return = curl_exec ( $ch );
        curl_close ( $ch );
        var_export($return);

를 추가합니다. 그 중 Workererman의 푸시 핵심 코드가 구현되어 있습니다

// 全局数组保存uid在线数据
$uidConnectionMap = array();
// 记录最后一次广播的在线用户数
$last_online_count = 0;
 
 
// PHPSocketIO服务
$sender_io = new SocketIO(2120);
// 客户端发起连接事件时,设置连接socket的各种事件回调
 
// 当$sender_io启动后监听一个http端口,通过这个端口可以给任意uid或者所有uid推送数据
$sender_io->on(&#39;workerStart&#39;, function(){
    // 监听一个http端口
    $inner_http_worker = new Worker(&#39;http://0.0.0.0:2121&#39;);
    // 当http客户端发来数据时触发
    $inner_http_worker->onMessage = function($http_connection, $data){
        global $uidConnectionMap;
        $_POST = $_POST ? $_POST : $_GET;
        // 推送数据的url格式 type=publish&to=uid&content=xxxx
        switch(@$_POST[&#39;type&#39;]){
            case &#39;publish&#39;:
                global $sender_io;
                $to = @$_POST[&#39;to&#39;];
                $_POST[&#39;content&#39;] = htmlspecialchars(@$_POST[&#39;content&#39;]);
                // 有指定uid则向uid所在socket组发送数据
                if($to){
                    $sender_io->to($to)->emit(&#39;new_msg&#39;, $_POST[&#39;content&#39;]);
                // 否则向所有uid推送数据
                }else{
                    $sender_io->emit(&#39;new_msg&#39;, @$_POST[&#39;content&#39;]);
                }
                // http接口返回,如果用户离线socket返回fail
                if($to && !isset($uidConnectionMap[$to])){
                    return $http_connection->send(&#39;offline&#39;);
                }else{
                    return $http_connection->send(&#39;ok&#39;);
                }
        }
        return $http_connection->send(&#39;fail&#39;);
    };
    
});
 
if(!defined(&#39;GLOBAL_START&#39;))
{
    Worker::runAll();
}

ok 그러면 끝입니다!

위 내용은 Workerman 기반 실시간 푸시(Ajax 폴링 포기)의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
이 기사는 csdn.net에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제