用as3 serverSocket 做服务端,实现web客户端

2024-01-05 19:08

本文主要是介绍用as3 serverSocket 做服务端,实现web客户端,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

听说,AIR2.0 有serverSocket可以做服务端,一时兴起,于是来试试,在官方看了一个示例,挺简单的(ps简单就意味着功能太少),就想来做一个聊天服务器(之前学完用FMS做聊天室,但毕竟是第三方服务器,受限制的,始终希望能自己做服务器,改天学C++);

       最初都是用的air应用做的客户端连接不成问题。后来做web的客户端,连接不成功,服务端收到消息<policy-file-request/> 客户端提示安全性错误,Erorr:#2048  对于服务端的消息最初并不懂啥意思,于是摆渡。资料到是不少,意思是说,web客户端需要客户端提供安全策略文件,也就是crossdomain.xml,我检查http服务器根目录也有这个文件,并且内容如下:

<?xml version="1.0" ?>
 <cross-domain-policy>
     <site-control permitted-cross-domain-policies="all"/>

    <allow-access-from domain="*" to-ports="*" />

 </cross-domain-policy>

想想不对,这是客户端找服务端要文件,又以为是服务端应用程序的相同目录也要放一个,于是也放一个进去(是不是挺傻的);测试,还是不行;再摆渡,搜到了flashPlayer的socket读取安全策略的原理,原来flashPlayer 最初会去服务端的843端口找策略文件,找不到然后才去指定的端口找,如果都没有crossdomain,就会抛出安全错误。 在网上找资料,是说要单独建一个843端口服务端,专门侦听843端,都是用的java服务端(Java俺也不怎么懂); 不过我只看了一个关键性代码如下:

var xml:XML =<cross-domain-policy>  
       <site-control permitted-cross-domain-policies="all"/>  
       <allow-access-from domain=\"*\" to-ports=\"*\"/>
       </cross-domain-policy>

那些 \ 我不懂是干啥的,我放到FB 里不正常,要把四个 \ 都去掉,反正大致原理就是:843服务端收到客户端请求时,就把这个xml发送给客户端,然后再把socket关掉,由于843端口把客户端关了,客户端会重新连接,因为flashplayer之前收到了策略文件,可以访问服务端,所以就不会再去找843端口,直接连接到正常的服务端口去。

我不懂JAVA ,所以就用air单独又建一个服务端,端口为843;FB源代码如下:

<?xml version="1.0" encoding="utf-8"?>
<s:WindowedApplication xmlns:fx="http://ns.adobe.com/mxml/2009"
        xmlns:s="library://ns.adobe.com/flex/spark"
        xmlns:mx="library://ns.adobe.com/flex/mx"
        width="300" height="300" creationComplete="init()">
 <fx:Declarations>
  <!-- Place non-visual elements (e.g., services, value objects) here -->
 </fx:Declarations>
 
 <fx:Script>
 <![CDATA[
 import flash.events.ServerSocketConnectEvent;
 import flash.net.ServerSocket;
 import flash.net.Socket;
   
 private var serverSocket:ServerSocket;
 private var localIP:String="192.168.1.253";//我的电脑IP
 private var localPort:int=843;     //一定要是843端口
 protected function init():void{
 serverSocket = new ServerSocket();
 serverSocket.bind(localPort,localIP  );
 serverSocket.addEventListener(ServerSocketConnectEvent.CONNECT, onConnect );
 serverSocket.listen();
 }
  //客户端连接事件;
 protected function onConnect(e:ServerSocketConnectEvent):void{
   var tmpsocket:Socket=e.socket;
   var xml:XML =<cross-domain-policy>  
       <site-control permitted-cross-domain-policies="all"/>  
       <allow-access-from domain="*" to-ports="*"/>
       </cross-domain-policy>
   tmpsocket.writeUTFBytes(xml.toString());
   tmpsocket.flush();
   log("发送策略文件到:"+xml.toString()+tmpsocket.remoteAddress+":"+tmpsocket.remotePort)

  //关掉
   tmpsocket.close();
 }
  //显示消息;
 protected function log(str:String):void{
   tmsg.appendText(str+"\n");
 }
 ]]>
   </fx:Script>
 <s:TextArea id="tmsg" x="48" y="11" editable="false" enabled="true"/>
</s:WindowedApplication>

 以下是web客户端源代码:

<?xml version="1.0" encoding="utf-8"?>
<s:Application xmlns:fx="http://ns.adobe.com/mxml/2009"
      xmlns:s="library://ns.adobe.com/flex/spark"
      xmlns:mx="library://ns.adobe.com/flex/mx"
      width="500" height="400" minWidth="500" minHeight="500">
 
 <fx:Script>
  <![CDATA[
   import flash.events.*;
   import flash.net.Socket;
   import flash.utils.ByteArray;
   
   
   private var localIP:String="192.168.1.253";
   private var localPort:int=8888;
   private var socket:Socket;
   
   
   protected function bt_conn_clickHandler(event:MouseEvent):void
   {   socket=new Socket();
    socket.connect(localIP,localPort);
    socket.addEventListener(Event.CONNECT,onconnect);
    socket.addEventListener(IOErrorEvent.IO_ERROR,onError);
    socket.addEventListener(SecurityErrorEvent.SECURITY_ERROR,onseError);
    socket.addEventListener(ProgressEvent.SOCKET_DATA,onData);
   }
   private function onconnect(e:Event):void{
    log("连接成功");
    
   }
   
   private function onError(e:IOErrorEvent):void{
    log("连接失败,服务器没打开!")
   }
   private function onseError(e:SecurityErrorEvent):void{
    log("连接失败,安全错误"+e.text)
   }
   //按钮发送
   protected function bt_send_clickHandler(event:MouseEvent):void
   {   
    sendmsg()
   }
   //接收到消息
   protected function onData(e:ProgressEvent):void{
    var ba:ByteArray=new ByteArray()
    var tmpsocket:Socket=e.target as Socket;
    tmpsocket.readBytes(ba,0,tmpsocket.bytesAvailable);
    log(ba.toString());
    
    
   }
   //发送消息;
   private function sendmsg():void{
    if(text_input.text!=""){
     socket.writeUTFBytes(text_input.text)
     socket.flush();
    
     text_input.text="";
    }else{
     log("不能发送空消息");
    } 
   }
   //显示消息
   private function log(str:String):void{
    text_msg.appendText(str+"\n")
   }
   
  ]]>
 </fx:Script>
 
 <fx:Declarations>
  <!-- Place non-visual elements (e.g., services, value objects) here -->
 </fx:Declarations>
 <s:Button id="bt_conn" x="10" y="348" label="连接" click="bt_conn_clickHandler(event)"/>
 <s:Button id="bt_send" x="381" y="352" label="发送" click="bt_send_clickHandler(event)"/>
 <s:TextInput id="text_input" x="122" y="319" width="251" height="54" enabled="true"
     enter="sendmsg()" prompt="输入消息"/>
 <s:TextArea id="text_msg" x="121" y="93" width="209" height="192" editable="false"
    enabled="true"/>
</s:Application>

AIR serverSocket服务端源代码:

<?xml version="1.0" encoding="utf-8"?>
<s:WindowedApplication xmlns:fx="http://ns.adobe.com/mxml/2009"
        xmlns:s="library://ns.adobe.com/flex/spark"
        xmlns:mx="library://ns.adobe.com/flex/mx"
        width="500" height="500">
 
 <fx:Script>
  <![CDATA[
   import flash.events.ServerSocketConnectEvent;
   import flash.net.ServerSocket;
   import flash.net.Socket;
   import flash.utils.ByteArray;
   
   
   
   private var serverSocket:ServerSocket;
   private var localIP:String="192.168.1.253";
   private var localPort:int=8888;
   private var clientList:Array=[];
  
   
   protected function bt_bind_clickHandler(event:MouseEvent):void
   {  
    serverSocket = new ServerSocket();
    serverSocket.bind(localPort,localIP);
    serverSocket.addEventListener(ServerSocketConnectEvent.CONNECT, onConnect );
    serverSocket.listen();
    log("绑定到:"+serverSocket.localAddress+":"+serverSocket.localPort);
   }
      
   protected function onConnect(e:ServerSocketConnectEvent):void{
    var tmpsocket:Socket=e.socket
    
    clientList.push(tmpsocket);
    tmpsocket.addEventListener(Event.CLOSE,onClose);
    tmpsocket.addEventListener(ProgressEvent.SOCKET_DATA,onSocketData);
    tmpsocket.writeUTFBytes("服务器消息:你已连接到服务器") 
    tmpsocket.flush();
    log("有新连接"+e.socket.remoteAddress+":"+e.socket.remotePort);
    }
   
   
   protected function onClose(e:Event):void{
    for (var clt:int=0;clt<clientList.length;clt++){
     if(clientList[clt].remoteAddress==e.target.remoteAddress){
      clientList.splice(clt,1); 
     }
    }
         
     log(e.target.remoteAddress+"断开") 
        
   }
   
   
   protected function onSocketData(e:ProgressEvent):void{
    var tmpsocket:Socket=e.target as Socket;
       var ba:ByteArray=new ByteArray()
         tmpsocket.readBytes(ba,0,e.target.bytesAvailable)
    var msg:String=ba.toString();
    if(msg.indexOf("<policy-file-request/>")>-1){
     log("不安全的客户端");
     
    }else{
          for each(var clt:Socket in clientList){
           clt.writeUTFBytes(tmpsocket.remoteAddress+":"+tmpsocket.remotePort+"消息:"+msg);
           clt.flush();
      
          }
        
       log("收到"+tmpsocket.remoteAddress+"消息:"+msg);   
    }
       
   }
   
   protected function bt_send_clickHandler(event:Event):void
   {
    sendmsg()
   }
   private function sendmsg():void{
       if(clientList.length>0){
         for each(var clt:Socket in clientList){
     
      clt.writeUTFBytes("服务器消息:"+text_send.text);
      clt.flush();
      log("消息已发送:"+clt.remoteAddress+":"+text_send.text);
     
         } 
    
       }else{
            log("当前无连接!")
         }
    text_send.text="";
   }
   protected function log(str:String):void{
    tr.appendText(str+"\n");
   }
  ]]>
 </fx:Script>
 
 <fx:Declarations>
  <!-- Place non-visual elements (e.g., services, value objects) here -->
 </fx:Declarations>
 <s:Button id="bt_bind" x="36" y="78" label="创建服务器" click="bt_bind_clickHandler(event)"/>
 <s:TextArea id="tr" x="41" y="124" width="243" height="196" contentBackgroundAlpha="0.73"
    editable="false" enabled="true"/>
 <s:Button id="bt_send" x="319" y="385" width="52" label="发送" click="bt_send_clickHandler(event)"/>
 <s:TextInput id="text_send" x="64" y="352" width="222" height="54" enabled="true" enter="sendmsg()"/>
 
 
</s:WindowedApplication>


这篇关于用as3 serverSocket 做服务端,实现web客户端的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/573869

相关文章

使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)

《使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)》在现代软件开发中,处理JSON数据是一项非常常见的任务,无论是从API接口获取数据,还是将数据存储为JSON格式,解析... 目录1. 背景介绍1.1 jsON简介1.2 实际案例2. 准备工作2.1 环境搭建2.1.1 添加

Java实现任务管理器性能网络监控数据的方法详解

《Java实现任务管理器性能网络监控数据的方法详解》在现代操作系统中,任务管理器是一个非常重要的工具,用于监控和管理计算机的运行状态,包括CPU使用率、内存占用等,对于开发者和系统管理员来说,了解这些... 目录引言一、背景知识二、准备工作1. Maven依赖2. Gradle依赖三、代码实现四、代码详解五

Redis连接失败:客户端IP不在白名单中的问题分析与解决方案

《Redis连接失败:客户端IP不在白名单中的问题分析与解决方案》在现代分布式系统中,Redis作为一种高性能的内存数据库,被广泛应用于缓存、消息队列、会话存储等场景,然而,在实际使用过程中,我们可能... 目录一、问题背景二、错误分析1. 错误信息解读2. 根本原因三、解决方案1. 将客户端IP添加到Re

java如何分布式锁实现和选型

《java如何分布式锁实现和选型》文章介绍了分布式锁的重要性以及在分布式系统中常见的问题和需求,它详细阐述了如何使用分布式锁来确保数据的一致性和系统的高可用性,文章还提供了基于数据库、Redis和Zo... 目录引言:分布式锁的重要性与分布式系统中的常见问题和需求分布式锁的重要性分布式系统中常见的问题和需求

SpringBoot基于MyBatis-Plus实现Lambda Query查询的示例代码

《SpringBoot基于MyBatis-Plus实现LambdaQuery查询的示例代码》MyBatis-Plus是MyBatis的增强工具,简化了数据库操作,并提高了开发效率,它提供了多种查询方... 目录引言基础环境配置依赖配置(Maven)application.yml 配置表结构设计demo_st

python使用watchdog实现文件资源监控

《python使用watchdog实现文件资源监控》watchdog支持跨平台文件资源监控,可以检测指定文件夹下文件及文件夹变动,下面我们来看看Python如何使用watchdog实现文件资源监控吧... python文件监控库watchdogs简介随着Python在各种应用领域中的广泛使用,其生态环境也

el-select下拉选择缓存的实现

《el-select下拉选择缓存的实现》本文主要介绍了在使用el-select实现下拉选择缓存时遇到的问题及解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录项目场景:问题描述解决方案:项目场景:从左侧列表中选取字段填入右侧下拉多选框,用户可以对右侧

Python pyinstaller实现图形化打包工具

《Pythonpyinstaller实现图形化打包工具》:本文主要介绍一个使用PythonPYQT5制作的关于pyinstaller打包工具,代替传统的cmd黑窗口模式打包页面,实现更快捷方便的... 目录1.简介2.运行效果3.相关源码1.简介一个使用python PYQT5制作的关于pyinstall

使用Python实现大文件切片上传及断点续传的方法

《使用Python实现大文件切片上传及断点续传的方法》本文介绍了使用Python实现大文件切片上传及断点续传的方法,包括功能模块划分(获取上传文件接口状态、临时文件夹状态信息、切片上传、切片合并)、整... 目录概要整体架构流程技术细节获取上传文件状态接口获取临时文件夹状态信息接口切片上传功能文件合并功能小

python实现自动登录12306自动抢票功能

《python实现自动登录12306自动抢票功能》随着互联网技术的发展,越来越多的人选择通过网络平台购票,特别是在中国,12306作为官方火车票预订平台,承担了巨大的访问量,对于热门线路或者节假日出行... 目录一、遇到的问题?二、改进三、进阶–展望总结一、遇到的问题?1.url-正确的表头:就是首先ur