琴韵古风 發表於 2019-8-3 22:44:00

uni-app结合PHP实现单用户登陆

<p>单用户登陆,即在一个应用中,同一个用户只能在线登陆一个,一个用户登陆,在其他设备上会被即时挤下线,确认后清空登陆该设备上的登陆装填并退回到登陆界面。</p>
<p>&nbsp; &nbsp; uni-app是目前能通过使用vue.js框架只需要编写一套代码同时打包Android,IOS,微信小程序,头条支付宝小程序和H5,通过使用HBuilder工具方便调试与云打包,关于苹果证书,推荐CW.PUB,https://cw.pub/index/document/index。使用HBuilder打越狱包通过那个网站签名就可以在正常苹果手机安装,不过网上还有其他些方法这里就不列举了。</p>
<p>&nbsp; &nbsp; &nbsp;一般APP做单用户登陆会使用第三方消息推送平台,虽然uni-app虽然也可以对接友盟,极光等推送平台。但还是因为时间,对接平台审核等流程时间不允许。之前使用gatewayworkman和websocket做了即时聊天,所以单用户登陆也使用websocket实现。</p>
<h1 id="h1_1">&nbsp;</h1>
<h1 id="h1_2">uni-app socket单用户登陆例</h1>
<p>1. uni-app前端在初始化socke时发送当前设备的唯一标识,然后实时接收一个“强制退出”类型的消息,一下只是简单示例。</p>
<pre><code class="hljs coffeescript"><span class="hljs-regexp">//初始化
socket.<span class="hljs-literal">on(<span class="hljs-string">'init', <span class="hljs-function"><span class="hljs-params">() =&gt; { <span class="hljs-regexp">//连接初始化
        socket.send({
                type: <span class="hljs-string">'login',
                token: uni.getStorageSync(<span class="hljs-string">'access_token'),
                device_no: plus.device.uuid,                        <span class="hljs-regexp">//手机设备唯一编号
        });
}).<span class="hljs-literal">on(<span class="hljs-string">'quit_push',<span class="hljs-function"><span class="hljs-params">(res)=&gt; {
        <span class="hljs-keyword">if(res) {
                uni.showModal({
                        title: <span class="hljs-string">'退出通知',
                        content: <span class="hljs-string">'你的账号在其他设备上登录!',
                        showCancel: <span class="hljs-literal">true,
                        cancelText: <span class="hljs-string">'取消',
                        confirmText: <span class="hljs-string">'确定',
                        success: res =&gt; {
                                <span class="hljs-keyword">if(res.confirm) {
                                        uni.clearStorageSync()
                                        store.commit(<span class="hljs-string">'chat/clear')
                                        uni.reLaunch({
                                                url:<span class="hljs-string">"../../pages/login/index"
                                        })
                                }<span class="hljs-keyword">else <span class="hljs-keyword">if(res.cancel) {
                                        uni.clearStorageSync()
                                        store.commit(<span class="hljs-string">'chat/clear')
                                        uni.reLaunch({
                                                url:<span class="hljs-string">"../../pages/login/index"
                                        })
                                }
                        }
                });
        }
});</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>&nbsp;</p>
<p>2. 后端接收“设备唯一标识”参数,先查找缓存是否存在,不存在记录设备标识和socket的clientid。</p>
<p>&nbsp;</p>
<p>3. 登陆接口接收设备标识,缓存或库里取出标识记录与当前接收的设备标识判断是否一致,不一致则根据缓存中的clientid发送消息。</p>
<pre><code class="hljs php">$is_online = Db::name(<span class="hljs-string">'UserLoginClient')-&gt;where(<span class="hljs-string">'user_id',$user[<span class="hljs-string">'id'])-&gt;order(<span class="hljs-string">'id desc')-&gt;find();
<span class="hljs-keyword">if(<span class="hljs-keyword">isset($device_no) &amp;&amp; $device_no &amp;&amp; $is_online[<span class="hljs-string">'device_no'] != $device_no &amp;&amp; !<span class="hljs-keyword">empty($is_online[<span class="hljs-string">'device_no'])) {
                Tools::sendToClient($is_online[<span class="hljs-string">'client_id'],json_encode([
                                                                                                                                                                                                <span class="hljs-string">'type' =&gt; <span class="hljs-string">'quit_push',
                                                                                                                                                                                                <span class="hljs-string">'data' =&gt; <span class="hljs-string">'ip',
                                                                                                                                                                                                <span class="hljs-string">'message' =&gt; <span class="hljs-string">'强制下线'
                                                                                                                                                                                        ]));
                        }</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>&nbsp;</p>
<p>4. 工具类sendToClient方法部分</p>
<pre><code class="hljs php"><span class="hljs-keyword">public <span class="hljs-keyword">static <span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">sendToClient<span class="hljs-params">($client_id, $message)
    {
      Gateway::sendToClient($client_id, $message);
    }</span></span></span></span></span></span></code></pre>
<p>&nbsp;</p>
<h2 id="h2_3">推送单用户登陆例</h2>
<p>1. 首先对接了友盟,包括前端后端都加了SDK和使用上了他们的方法。</p>
<p>&nbsp;</p>
<p>2. 消息推送有一个唯一值"token",这里简称“pushtoken”,由客户端生成,可以标识一个唯一的设备。</p>
<p>&nbsp;</p>
<p>3. 后端登陆时,接收pushtoken,同样判断该pushtoken是否存在,不存在就以用户ID为键存储。</p>
<p>&nbsp;</p>
<p>4. 存在时再判断与缓存是否一致,一致则加长缓存时间,不一致则给旧的pushtoken(缓存中的)推送一条消息,并缓存新的pushtoken。</p>
<pre><code class="hljs php"><span class="hljs-keyword">if (<span class="hljs-keyword">self::$headToken &amp;&amp; Cache::has(<span class="hljs-keyword">self::$prefix . <span class="hljs-keyword">self::$userId)) {
            <span class="hljs-keyword">if (<span class="hljs-keyword">self::$headToken == Cache::get(<span class="hljs-keyword">self::$prefix . <span class="hljs-keyword">self::$userId)) {
                Cache::set(<span class="hljs-keyword">self::$prefix . <span class="hljs-keyword">self::$userId, <span class="hljs-keyword">self::$headToken, <span class="hljs-keyword">self::$timeOut);
            } <span class="hljs-keyword">else {      <span class="hljs-comment">// 换了手机,客户端重新发送pushtoken到服务端,服务端与缓存中的pushtoken比较,不同则给原来pushtoken手机推送一条并重新缓存新的token
                <span class="hljs-comment">// modify by wensen on 20180816
                <span class="hljs-comment">// $addr = getCity();
                $addr = getMobCity();
                $ip = request()-&gt;ip();
               
                <span class="hljs-keyword">if ($addr) {
                  $addr[<span class="hljs-string">'province'] = <span class="hljs-keyword">empty($addr[<span class="hljs-string">'province']) ? <span class="hljs-string">'' : $addr[<span class="hljs-string">'province'];
                  $addr[<span class="hljs-string">'city'] = <span class="hljs-keyword">empty($addr[<span class="hljs-string">'city']) ? <span class="hljs-string">'' : $addr[<span class="hljs-string">'city'];

                  <span class="hljs-comment">// $address = "\t" . $addr['country'] . "-" . $addr['region'] . "-" . $addr['city'] . " (IP:" . $ip . ")\t";
                  $address = <span class="hljs-string">"\t" . $addr[<span class="hljs-string">'country'] . <span class="hljs-string">"-" . $addr[<span class="hljs-string">'province'] . <span class="hljs-string">"-" . $addr[<span class="hljs-string">'city'] . <span class="hljs-string">" (IP:" . $ip . <span class="hljs-string">")\t";
                } <span class="hljs-keyword">else {
                  $address = <span class="hljs-string">"IP:" . $ip . <span class="hljs-string">"";
                }

                $OldToken = Cache::get(<span class="hljs-keyword">self::$prefix . <span class="hljs-keyword">self::$userId);

                <span class="hljs-keyword">if (strlen($OldToken) == <span class="hljs-number">64) {
                  $content = <span class="hljs-keyword">array(
                        <span class="hljs-string">'title' =&gt; <span class="hljs-string">'APP紧急通知',
                        <span class="hljs-string">'body' =&gt; <span class="hljs-string">'您的账号于:' . date(<span class="hljs-string">'Y-m-d H:i:s') . <span class="hljs-string">'在' . $address . <span class="hljs-string">'处登录,若不为您本人登录,请您立即修改密码!',
                        <span class="hljs-string">'pull_service' =&gt; <span class="hljs-string">'login'
                  );
                  \umeng\Push::send($OldToken, <span class="hljs-string">'unicast', $content, <span class="hljs-string">'message', <span class="hljs-keyword">true);
                } <span class="hljs-keyword">elseif (strlen($OldToken) == <span class="hljs-number">44) {
                  $content = <span class="hljs-keyword">array(
                        <span class="hljs-string">'pull_service' =&gt; <span class="hljs-string">'login',
                        <span class="hljs-string">'msg' =&gt; <span class="hljs-string">'您的账号于:' . date(<span class="hljs-string">'Y-m-d H:i:s') . <span class="hljs-string">'在' . $address . <span class="hljs-string">'处登录,若不为您本人登录,请您立即修改密码!'
                  );
                  \umeng\Push::send($OldToken, <span class="hljs-string">'unicast', $content, <span class="hljs-string">'message', <span class="hljs-keyword">true);
                }
                Cache::set(<span class="hljs-keyword">self::$prefix . <span class="hljs-keyword">self::$userId, <span class="hljs-keyword">self::$headToken, <span class="hljs-keyword">self::$timeOut);
            }
      } <span class="hljs-keyword">else {
            Cache::set(<span class="hljs-keyword">self::$prefix . <span class="hljs-keyword">self::$userId, <span class="hljs-keyword">self::$headToken, <span class="hljs-keyword">self::$timeOut);
      }</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>&nbsp;</p>
<p>5. APP客户端接收推送进行弹窗提示和退出处理。</p>
<p>&nbsp;</p>
<p>6. 以上是根据友盟的SDK封装的推送方法,其中包括单播,广播,跳应用activity,跳网页连接等等。</p>
<p>&nbsp;</p>
<p><img class="zoom-in-cursor lazyload" alt="" width="430" height="430" data-src="https://oscimg.oschina.net/oscnet/89e837d7d0749d6a0ff6cab4114f7320d24.jpg"></p>

</div>
<div id="MySignature" role="contentinfo">
    个人网站:www.zerofc.cn
公众号:ZEROFC_DEV
QQ交流群:515937120
QQ:2652364582
头条号:1637769351151619
B站:286666708
大鱼号:北桥苏<br><br>
来源:https://www.cnblogs.com/zerofc/p/11296634.html
頁: [1]
查看完整版本: uni-app结合PHP实现单用户登陆