首先,先打开官网手册   http://doc.workerman.net/

根据手册里安装里的提示,完成环境检测,和安装对应的扩展,并把对应的WorkerMan代码包下载解压至根目录

在根目录下创建一个index.php

index.php代码如下:

<?php
use Workerman\Worker;
require_once __DIR__ . '/Workerman/Autoloader.php';
use Workerman\Lib\Timer; Worker::$daemonize = true;
// 所有的打印输出全部保存在/stdout.log文件中
Worker::$stdoutFile = '/stdout.log'; // 注意:这里与上个例子不同,使用的是websocket协议
$ws_worker = new Worker("websocket://0.0.0.0:2999"); // 启动1个进程对外提供服务
$ws_worker->count = 1; // 设置实例的名称
$ws_worker->name = 'Worker1'; //定义全局变量
$user_ip = []; // 心跳间隔120秒
define('HEARTBEAT_TIME', 120); // 当收到客户端发来的数据后返回hello $data给客户端
//闭包使用use传递参数进行使用
$ws_worker->onMessage = function($connection, $data)use($ws_worker)
{
// 给connection临时设置一个lastMessageTime属性,用来记录上次收到消息的时间
$connection->lastMessageTime = time(); global $user_ip;
file_put_contents('666.txt',$data, FILE_APPEND | LOCK_EX); //用户发送的信息内容
$content = substr($data,mb_strpos($data,':') + 1,mb_strlen($data));
//用户发送信息的状态请求,比如,登录,发送消息等等
$sort = strstr($data,':',true);
//获取用户的ip
$ip = $connection->getRemoteIp();
if($sort == 'login'){
//判断用户昵称是否重复
$array_search = array_search($content,$user_ip, false);
if( $array_search != $ip && !empty($array_search)){
//触发回调,关闭服务器和客户端链接
$connection->close('sort:'.'401');
}
if(!array_key_exists($ip,$user_ip)){
//记录登录时间
echo '登录ip:'.$ip.' 登录时间:'.date('Y-m-d H:i:s', time()).'\n';
}
// 录入ip和昵称
$user_ip[$ip] = $content;
//登录成功,返回200
$connection->send('sort:'.'200'); }else if($sort == 'text'){
//获取用户发送的消息,并返回
// $connection->send('text:'.$user_ip[$ip].':'.$content); //广播,通过$ws_worker->connections获取所有连接服务端的对象,循环给所有在线用户推送信息
foreach($ws_worker->connections as $connection)
{
$connection->send('text:'.$user_ip[$ip].'发送消息:'.$content);
}
}else{
//未知错误,返回500
$connection->send('sort:'.'500');
} }; // 进程启动后设置一个每秒运行一次的定时器
$ws_worker->onWorkerStart = function($ws_worker) {
Timer::add(1, function()use($ws_worker){
$time_now = time();
foreach($ws_worker->connections as $connection) {
// 有可能该connection还没收到过消息,则lastMessageTime设置为当前时间
if (empty($connection->lastMessageTime)) {
$connection->lastMessageTime = $time_now;
continue;
}
// 上次通讯时间间隔大于心跳间隔,则认为客户端已经下线,关闭连接,返回402
if ($time_now - $connection->lastMessageTime > HEARTBEAT_TIME) {
$connection->close('sort:'.'402');
}
}
});
}; //用户断开连接时触发函数
$ws_worker->onClose = function($connection)
{
global $user_ip;
//清除用户的ip和用户名
$ip = $connection->getRemoteIp();
unset($user_ip[$ip]);
echo 'ip:'. $ip . ' 下线时间:'.date('Y-m-d H:i:s', time()).'\n';
}; // 运行worker
Worker::runAll();

然后在根目录中的index.html写入以下代码

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<script src="https://libs.baidu.com/jquery/1.8.3/jquery.min.js"></script>
</head>
<body>
<div>你的昵称:<input type="text" id="username" value=""/></div>
<div>
回复内容:
<textarea style="width: 500px;height: 100px" id="content" value=''></textarea>
</div>
<div>
<input type="button" onclick="login()" value="连接服务器">
<input type="button" onclick="send()" value="发送消息">
<input type="button" onclick="quit()" value="退出">
</div> <script> var socket = null; //将socket实例保存到变量中
var isLogin = false; //登录标识符 var ws; function login(){
var name = document.getElementById('username').value;
if(name.length == 0 || name == ''){
alert('请输入昵称');
return false;
}
ws = new WebSocket("ws://www.xxx.com:2999"); ws.onopen = function() {
ws.send('login:'+name);
ws.onmessage = function(e) {
console.log(e);
callback(e);
}
}
} function send(){
if(!isLogin){
alert('请先连接服务器');
return false;
} var content =document.getElementById('content').value; ws.send('text:' + content);
ws.onmessage = function(e) {
callback(e);
}
} function quit(){
if(!isLogin){
alert('请先连接服务器');
}
ws.send('quit');
ws.onmessage = function(e){
callback(e);
}
} //返回值处理
function callback(e){
//返回的状态
var sort = e.data.substring(0,e.data.indexOf(':'));
//返回的内容
var content = e.data.substring(e.data.indexOf(':')+1); if(sort == 'text'){
alert(content);
}else if(sort == 'sort'){ if(content == 401){
alert('用户名重复,连接失败');
}else if(content == 200){
isLogin = true;
alert('连接成功');
}else if(content == 402){
isLogin = false;
alert('长时间未发消息,已断开连接,请重新连接服务器');
}else if(content == 202){
isLogin = false;
alert('成功断开连接');
}else if(content == 500){
alert('未知错误');
}else{
alert('未知错误');
} }else{
alert('连接失败');
}
}
</script>
</body>
</html>

弄好了之后,打开终端,cd 至网站根目录,执行index.php脚本,监听端口,并加入守护进程

php index.php start -d

执行成功,那么你的一个简单的即时通讯功能就完成了

测试:两个不同的用户登录网页,输入昵称,连接服务器,其中一个发送消息,另一个在页面中有收到

那么恭喜你,一个简单的即时通讯功能就这样做出来了

参考文章:https://blog.csdn.net/qq_33862644/article/details/79554321

workerman搭建聊天室的更多相关文章

  1. 使用nodejs+express+socketio+mysql搭建聊天室

    使用nodejs+express+socketio+mysql搭建聊天室 nodejs相关的资料已经很多了,我也是学习中吧,于是把socket的教程看了下,学着做了个聊天室,然后加入简单的操作mysq ...

  2. 使用Angular和Nodejs搭建聊天室

    一,利用Node搭建静态服务器 这个是这个项目的底层支撑部分.用来支持静态资源文件像html, css, gif, jpg, png, javascript, json, plain text等等静态 ...

  3. laravel整合workerman做聊天室

    测试工具  http://www.blue-zero.com/WebSocket/ 2018年8月6日17:28:24 <?php namespace App\Console\Commands; ...

  4. 使用socket.io搭建聊天室

    最近在学习nodejs,需要找一些项目练练手.找来找去发现了一个聊天室的教程,足够简单,也能从中学到一些东西.下面记录我练习过程中待一些笔记. nodeJS模块 共用到了2个模块,express和so ...

  5. SpringBoot--使用socket搭建聊天室

    1.添加依赖 <dependency> <groupId>org.springframework.boot</groupId> <artifactId> ...

  6. Workerman创建聊天室实例

    // 标记是全局启动 define('GLOBAL_START', 1); require_once __DIR__ . '/Workerman/Connection.php'; require_on ...

  7. 基于swoole搭建聊天室程序

    1. 创建websocket服务器 swoole从1.7.9版本开始, 内置了websocket服务器功能,我们只需几行简单的PHP代码,就可以创建出一个异步非阻塞多进程的WebSocket服务器. ...

  8. 使用 NIO 搭建一个聊天室

    使用 NIO 搭建一个聊天室 前面刚讲了使用 Socket 搭建了一个 Http Server,在最后我们使用了 NIO 对 Server 进行了优化,然后有小伙伴问到怎么使用 Socket 搭建聊天 ...

  9. 使用WebRTC搭建前端视频聊天室——点对点通信篇

    WebRTC给我们带来了浏览器中的视频.音频聊天体验.但个人认为,它最实用的特性莫过于DataChannel——在浏览器之间建立一个点对点的数据通道.在DataChannel之前,浏览器到浏览器的数据 ...

随机推荐

  1. day73:drf:drf视图相关类&路由Routers&创建虚拟环境

    目录 1.APIView 2.GenericAPIView:通用视图类 3.5个视图扩展类:ListModelMixin,CreateModelMixin,RetrieveModelMixin,Upd ...

  2. 谈谈Android项目框架的前世今生

    嗨,大家好,今天出了大太阳,真是美好的开始. 这篇文章和大家说说Android届流行的三大框架,了解下架构的前世今生,以及我对于这些框架的一些认识和看法. 三大框架区别 MVC 架构介绍 Model: ...

  3. array_walk_recursive 地址引用报错的问题

    今天看十八哥的视频,学习array_walk_recursive的用法,发现一直报错: PHP版本:5.6.19 代码界面: 报错界面: 查了很长时间,不知道什么问题,后来在网上终于找到原因所在: + ...

  4. LOJ 6068「2017 山东一轮集训 Day4」棋盘

    题意 一个 \(n\times n\) 的棋盘上面有若干障碍物. 定义两个棋子可以互相攻击当且仅当这两个棋子的横坐标或纵坐标相等而且中间不能隔着障碍物.(可以隔棋子) 有 \(q\) 次询问,每次询问 ...

  5. Learn day9 粘包\struct用法\hashlib校验\socketserver并发\模块引入\进程\join\守护进程

    1.粘包现象 总结 : 导致黏包现象的两种情况 hello,worl d (1) 在发送端,发送数据太快,频繁发送 (2) 在接收端,接收数据太慢,延迟截取 # ### 服务端 import sock ...

  6. JS中使用for-each遍历数组

    1 let array = [1, 3, 6, 8, 9, 0, 5]; 2 /* 3 index是数组索引 4 value代表数组的值 5 arr是指整个数组 6 */ 7 array.forEac ...

  7. Json Master masters JSON!

    对于一个软件开发人员, JSON 是最熟悉的东西之一了, 每一个开发人员基本上每一天都会跟 JSON 打交道. 作为一个大前端开发人员, 当看到从服务器返回的 JSON 数据时, 尤其是大数据量或者复 ...

  8. 1_Two Sum

    1.Two Sum Given an array of integers, return indices of the two numbers such that they add up to a s ...

  9. Loadrunner11简单压测接口教程

    一.需求 使用Loadrunner压测目标接口,要求支持1000并发数. 目标接口:https://www.xxx.com/digitaldata/api/signer/1.0/signerRegis ...

  10. 当年使用dpdk干的事

    mark一下  晚点上传 先不上传 ....0727