在AS3中使用自定义类来处理数据库查询


Using custom class to handle database inquiries in AS3

我最近开始深入研究AS3中的自定义类(以磨练我的最佳实践编码习惯),并希望创建一个数据库类,允许用户首先实例化一个类,该类包含该类中的方法在MySQL表中添加、删除、修改(等)行所需的所有信息(通过PHP)。当然,这涉及到使用URLRequest、URLLoader等等。我的问题是,是否有人知道如何从一个专门包含var数据的方法返回数据,而不依赖于该方法本质上调度事件(然后必须创建一个监听器,而不是将其内置到类中)。例如,

var myDB:dataBase = new dataBase("dbase","table","username","pword");
//this creates an instance of a database class with methods like:
trace(myDB.fetch(1)); //gets first row of table as a delimited string

if (myDB.delete(1)) {} 
//returns Boolean true if delete of row 1 was successful

我在下面找到了答案,其中包含一种创建返回事件的类的方法:

在Actionscript 3.0中结合URLRequest、URLLoader和Complete Event Listener?

但我希望该方法返回一个包含数据库数据的字符串或布尔确认,而不是调度事件侦听器。下面是我制作的类的一个例子:

package com.customClasses {
    import flash.net.URLLoader;
    import flash.net.URLRequest;
    import flash.net.URLVariables;
    import flash.net.URLLoaderDataFormat;
    import flash.net.URLRequestMethod;
    import fl.events.DataChangeEvent;
    import flash.events.Event
    public class dataBase {
        public var dbs:String = "";
        public var usr:String = "";
        public var pwd:String = "";
        public var tab:String = "";
        var returnData:String = "";
    // Constructor
    public function dataBase(dbs:String, usr:String, pwd:String, tab:String) {
        this.dbs = dbs;
        this.usr = usr;
        this.pwd = pwd;
        this.tab = tab;
    }
    public function fetch(idn:uint, par:String):String { 
        var returnData:String = "blank";
        var vUrlReq:URLRequest = new URLRequest ("dBase.php");
        var vUrlVars:URLVariables = new URLVariables(); 
        function onLoadVarsComplete(event:Event): void {
            //retrieve success variable from our PHP script:
            if(event.target.data.msg == "success") {
                var rawData:URLVariables = new URLVariables( event.target.data );
                returnData = rawData.fromPHP;
            } else {
                returnData = "failed!";
            }
        }
        vUrlReq.method = URLRequestMethod.POST;    
        vUrlVars.dir=dbs; // name of table affected
        vUrlVars.alpha=usr; // username
        vUrlVars.beta=pwd;  // password
        vUrlVars.dbase=tab; // name of table affected
        vUrlVars.func="fetch"; // function for php script to use
        vUrlVars.idnum=idn; //if >0 search for record with that id
        vUrlReq.data = vUrlVars;
        var vLoader:URLLoader = new URLLoader (vUrlReq);
        vLoader.addEventListener("complete", onLoadVarsComplete);
        vLoader.dataFormat = URLLoaderDataFormat.VARIABLES;
        vLoader.load(vUrlReq);
        return (returnData);
    }

returnData返回"空白"。。。所以我意识到我的方法没有按预期工作。我还意识到returnData字符串存在一些作用域问题,并且我使用的是嵌套函数(可能是no)。否则,有什么想法吗?

要执行您想要的操作,您可以使用回调函数或DataEvent侦听器,如下所示:

DB.as:

package  {
    import flash.net.*;
    import flash.events.*;
    public class DB extends EventDispatcher {       
        public static const DATA_LOADED = 'data_loaded';
        public function DB() {
        }
        public function getDataUsingDataEvent(file_path:String):void {
            var url_loader:URLLoader = new URLLoader();
                url_loader.addEventListener(
                    Event.COMPLETE, 
                    function(e:Event):void
                    {
                        var event:DataEvent = new DataEvent(DATA_LOADED, true, false, url_loader.data);
                        dispatchEvent(event);
                    }
                )
                url_loader.load(new URLRequest(file_path));     
        }
        public function getDataUsingCallback(file_path:String, callback:Function):void {
            var url_loader:URLLoader = new URLLoader();
                url_loader.addEventListener(
                    Event.COMPLETE, 
                    function(e:Event):void 
                    {
                        callback(url_loader.data);
                    }
                )
                url_loader.load(new URLRequest(file_path));     
        }
    }
}

然后:

var db:DB = new DB();
    db.addEventListener(
        DB.DATA_LOADED, 
        function(event:DataEvent):void {
            trace(event.data);
        }
    )   
    db.getDataUsingDataEvent('file_path');
    db.getDataUsingCallback(
        'file_path', 
        function(data:String):void {
            trace(data);
        }
    )

希望这能有所帮助。

正如您所说,这在AS3中是无法实现的。在返回函数并继续执行脚本之前,不能等待异步操作(如URLLoader/load())。

如果您不太喜欢使用addEventListener(),那么您可以做的是传递回调,或者实现promise的方法链接。这些模式并不一定比使用事件更好,也有其自身的问题,但可以让您编写作为序列更可读的代码。这些模式在Javascript(具有与ActionScript相同的异步行为)中很常见,例如jQuery。小心"回调地狱"answers"火车残骸"。这些技术的目的是使事情写起来更简单,但有时会使事情更容易出错。