首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 开发语言 > 编程 >

[转载][新闻资讯] [Flash/Flex] Adobe荐文:Array种编程

2012-12-20 
[转载][新闻资讯] [Flash/Flex] Adobe荐文:Array类编程http://bbs.9ria.com/viewthread.php?tid70786&ext

[转载][新闻资讯] [Flash/Flex] Adobe荐文:Array类编程
http://bbs.9ria.com/viewthread.php?tid=70786&extra=page%3D1%26amp;orderby%3Ddateline%26amp;filter%3D86400

资讯类型: 翻译
来源页面: http://www.adobe.com/devnet/flash/quickstart/programming_arrays_as3.html
资讯原标题: Programming with the Array class
资讯原作者: Buck DeFore
我的评论:
对这篇文你有啥看法,跟贴说说吧!欢迎口水和板砖,哈哈。欢迎大家和我们一同分享更多资讯。

本文阅读要求:
具备知识:
建议具备ActionScript 3编程经验.
用户水平:
初级

需要产品:
Flash Professional
样例:
ArrayMapMethod.zip
ArraySearchMethods.zip

数组是一种很重要的数据结构。数组最基础的用途是存放对象的容器。数组使维护和修改相关对象变得非常简单。当你处理一系列拥有相同特性的变量时,数组的方便性就更显而易见了。比如,你有一个列表的联系人,你可以把它们放入一个数组,这时数组就成为了一个通讯录,你可以很方便的遍历,或者提取想要的条目来进行修改。

下面的章节详细介绍了数组编程的细节。

数组类型

最常见的数组是索引数组。这个类型中,数组中的每一个元素都有一个索引值。AS3中的数组是以0为基数的;数组中第一个元素的索引是0,第二个的索引是1,以此类推。你可以通过中括号操作符 ( [ ] )来获取指定索引的元素: // Create a new array, adding three elements to it

var arr:Array = new Array("a", "b", "c");

trace(arr); // output: a,b,c



// Overwrite the third item (at index 2) with a new string

arr[2] = "aa";

// Set the third index to a new string

arr[3] = "bb";



trace(arr); // output: a,b,aa,cc
复制代码数组中可以包含其他数组。这称作多维数组。数组可以被看作是一行,其中每个元素可以看作是一列。如果一个数组中每个元素包含一个数组,这就是一个二维数组,可以被放入一个很多行和很多列组成的网格中。一个位图可以用一个二维数组来描述,其每个元素都是一行包含每列元素的数组。每列元素包含每个像素的色彩信息: // Create a new array

var arr:Array = new Array();

arr[0] = new Array("a", "b", "c");

arr[1] = new Array("d", "e", "f");

arr[2] = new Array("g", "h", "i");



trace(arr); // output: a,b,c,d,e,f,g,h,i

trace(arr[0]); // output: a,b,c

trace(arr[0][0]); // output: a

trace(arr[0][1]); // output: b

trace(arr[0][2]); // output: c

trace(arr[2][2]); // output: i        
复制代码最后一种类型的数组是关联数组,常被称作hash或map。这种数组不含索引,但相应的用键来指引元素。AS3中,关联数组的功能和一个Object类是相同的。关联数组的键值可以被看作是对象的属性。这样,你就可以用对象或数组的语法来创建关联数组: // Create a new array

var indexedArray:Array = new Array();

// Assign index 0 to a new string

indexedArray[0] = "bob";

trace(indexedArray[0]); // output: bob

trace(indexedArray.length); // output: 1



// Create a new array

var associativeArr:Array = new Array();

// Assign a new string to key "name", making the array associative

associativeArr["name"] = "bob";

trace(associativeArr["name"]) // output: bob

trace(associativeArr.length); // output: 0 (length method unavailable to associative arrays)



var obj:Object = new Object();

obj.name = "bob";

trace(obj.name); // output: bob
复制代码注意,不管你是如何创建的关联数组,它始终是一个未排序的键值对,正因如此,它不具备索引数组的特性。一旦一个数组被赋予了键值对,这个数组就无法使用length() 方法来确定里面包含的元素个数。

修改数组内容

你可以使用中括号操作符来对指定索引赋值,你也可以用Array类的方法在数组的开始和结尾加入或删除元素。在数组末尾添加一个元素,使用push()方法。在数组开始增加一个元素使用unshift()方法。你也可以使用 shift()和pop()方法在数组的开头和末尾删除元素。以下的例子展示了这些方法的区别: var arr:Array = new Array();

arr.push("green");

arr.push("blue");

arr.push("violet");

trace(arr); // output: green,blue,violet



arr.unshift("yellow");

arr.unshift("orange");

arr.unshift("red");

trace(arr); // output: red,orange,yellow,green,blue,violet



// Remove the last item from the array

var lastItem:String = arr.pop();

trace(lastItem); // output: violet

// Remove the first item from the array

var firstItem:String = arr.shift();

trace(firstItem); // output: red



trace(arr); // output: orange,yellow,green,blue
复制代码你可以使用splice()方法在数组中间添加或删除元素。splice()方法的第一个参数是起始修改元素的索引: // Create an array with two strings

var arr:Array = new Array("black", "white");



// Add two to the beginning

arr.splice(0, 0, "red", "orange", "yellow");
复制代码如果你是增加元素,第二个参数设为0,后面跟要加的元素: // Add colors to the end

arr.splice(5, 0, "green", "blue", "violet");
复制代码如果你要删除元素,第二个参数设为要删除元素的个数: // Remove two in the middle, starting at the third index

arr.splice(3,2);
复制代码你可以用slice()或concat()方法,以现有的数组来创建新数组。slice()方法返回一个现有数组的子集。concat()方法返回所传给它的参数的合并集。传递给这两个方法的原有数组保持不变。 // Create a combined array without changing an initial array

var initialArray:Array = new Array("red", "orange", "yellow");

// Create an array that combines the initial array with three more strings

var combinedArray:Array = initialArray.concat("green", "blue", "violet");

// Create a subset of this array without changing the original

var slicedArray:Array = combinedArray.slice(3);



trace(initialArray); // output: red,orange,yellow

trace(combinedArray); // output: red,orange,yellow,green,blue,violet

trace(slicedArray); // output: green,blue,violet          
复制代码
传递数组中的元素给回调函数

数组类的方法使查询数组内元素和修改元素变得方便。这一节将详细介绍。

数组类的五个方法使用回调函数:every(), some(), forEach(), map(),和filter()。这些方法将函数当作一个参数。数组中的每个元素调用一次指定的函数。每个回调函数要有三个参数:第一个是数组中元素的指针,第二个是该元素的索引,第三个是数组本身。

这些回调函数可以方便的在数组内所有元素上执行代码。比如,你可以用forEach()方法得到数字内所有元素:var arr:Array = new Array(  { country:"Cameroon", continent:"Africa" },

                            { country:"Laos", continent:"Asia" },

                            { country:"Uruguay", continent:"South America" },

                            { country:"Romania", continent:"Europe" },

                            { country:"Canada", continent:"North America" } );



arr.forEach(traceContinent);





function traceContinent(element:*, index:int, arr:Array):void {



    trace(element.country + " is found in " + element.continent);

}
复制代码map()方法和forEach()相似,但它返回的是回调函数所返回的新数组。这个例子创建了一个DataProvider实例,用来格式化填充列表。 (注意你需要FLA的list组件来进行编译):import fl.controls.Button;

import fl.controls.DataGrid;

import fl.controls.List;

import fl.data.DataProvider;



var famousWorks:Array = new Array(  { author:"Homer", work:"The Iliad" },

                                    { author:"Jules Verne", work:"Journey to the Center of the Earth" },

                                    { author:"James Joyce", work:"Ulysses" },

                                    { author:"John Steinbeck", work:"Grapes of Wrath" },

                                    { author:"Allen Ginsberg", work:"Howl" } );



// create grid to show initial array

var grid:DataGrid = new DataGrid();

grid.width = 350;

grid.height = 125;

grid.move(10, 10);

grid.dataProvider = new DataProvider(famousWorks);

addChild(grid);



// create list for formatted output

var list:List = new List();

list.width = 350;

list.move(10, 180);

list.rowCount = famousWorks.length;

addChild(list);



// create button to trigger the map method

var mapBtn:Button = new Button();

mapBtn.x = 80;

mapBtn.y = 10 + grid.height + 10;

mapBtn.width = 200;

mapBtn.label = "Format using Array.map()";

addChild(mapBtn);



mapBtn.addEventListener(MouseEvent.CLICK, onMapButtonClick);



function onMapButtonClick(evt:MouseEvent):void

{

        // generate a formmatted list using the map() method

        var formattedWorks:Array = famousWorks.map(formatFamousWork);

        var dp:DataProvider = new DataProvider(formattedWorks);

        list.dataProvider = dp;

}



function formatFamousWork(element:*, index:int, arr:Array):String {

    return element.work + " (by " + element.author + ")";

}
复制代码
这个例子的源码在本文最上方ArrayMapMethod.zip中。Flash Professional CS5版本FLA文件是ArrayMapMethod_CS5.fla。

你可以使用filter()方法来创建另一个数组的子集。在filter()方法的回调函数中,如果数组元素需要在新数组中保留则必需返回true,反之返回false。下面的例子随机产生100个0-100的随机数组成数组,使用filter()方法来创建大于90的数组子集:var arr:Array = new Array();



var totalElements:uint = 100;

for(var i:uint = 0; i<totalElements; i++) {



    arr[i] = Math.round(Math.random()*100);

}



function isLargerThan90(element:*, index:int, arr:Array):Boolean {

    return element > 90;



}

var highestNumbers:Array = arr.filter(isLargerThan90);

trace(highestNumbers);
复制代码every()和some()方法要求回调函数返回一个布尔值。every()方法对数组中每个元素执行回调函数,直到回调函数返回false。相似的,some()方法对每个元素执行回调函数直到返回true。这个动作可以用来确定数组是否达到每个标准。下面的例子确定一个数组是否都是由字符串组成:var arr:Array = new Array();

var totalElements:uint = 1000;



for(var i:uint = 0; i<totalElements; i++) {

    arr[i] = "name";

}



trace("Array is entirely composed of strings: " + arr.every(hasOnlyStrings));



function hasOnlyStrings(element:*, index:int, arr:Array):Boolean {

    return element is String;

}



arr[42] = 81; // change one element to a number



trace("Array is entirely composed of strings: " + arr.every(hasOnlyStrings));
复制代码  数组中搜索元素

你可以用indexOf()或lastIndexOf()方法寻找数组中的元素。这些方法使用优化的搜索方法,比自定义的搜索循环有更好的表现。下面的例子比较用indexOf()方法搜索、循环搜索和for each搜索:// assumes having two components on the screen:

// 1. A TextArea named outputTxt

// 2. A Button named startBtn



var loopCount:Number = Math.pow(10,6);

var loopingTally:uint = 0;

var forEachTally:uint = 0;

var indexOfTally:uint = 0;

var testsToMake:uint = 5;



// Create a large array composed entirely of zeroes

var a:Array = new Array();



for (var i:uint=0; i<loopCount; i++)

{

    a[i] = 0;

}



// Start a timer that will start a new test every half-second, up to the total tests needed

var t:Timer = new Timer(500,testsToMake);

// Run the makeTest function every time the timer ticks

t.addEventListener(TimerEvent.TIMER,makeTest);

// When the timer has completed, run the showResults function

t.addEventListener(TimerEvent.TIMER_COMPLETE,showResults);

t.start();



function findByLooping(toSearchFor:Number):uint {

    var start:int = getTimer();

    for (i=0; i<loopCount; i++)

    {

        if(a[i] == toSearchFor) {

            var end:int = getTimer();

            var elapsed:int = end - start;

            trace ("Found " + toSearchFor + " by looping at index: " + i + " time: " + String(elapsed));

            break;

        }

    }

    return elapsed;

}

function findByForEach(toSearchFor:Number):uint {

    var start:int = getTimer();

    for each(var entry:Number in a)

    {

        if(entry == toSearchFor) {

            var end:int = getTimer();

            var elapsed:int = end - start;

            trace ("Found " + toSearchFor + " in for each loop at time: " + String(elapsed));

            break;

        }

    }

    return elapsed;

}

function findByIndexOf(toSearchFor:Number):uint {

    var start:int = getTimer();

    var foundIndex:int = a.indexOf(toSearchFor);

    var end:int = getTimer();

    var elapsed:int = end - start;

    trace ("Found " + toSearchFor + " by indexOf() at index: " + foundIndex + " time: " + String(elapsed));

    return elapsed;

}

function makeTest(e:TimerEvent):void {

    // Get a random number

    var randomNumber:Number = Math.random();

    // Assign it to a random index in our dummy array

    a[Math.floor(Math.random()*loopCount)] = randomNumber;

    // Find it, timing various methods

    loopingTally += findByLooping(randomNumber);

    forEachTally += findByForEach(randomNumber);

    indexOfTally += findByIndexOf(randomNumber);   

}

function showResults(e:TimerEvent):void {

    trace("== Average search time by looping: " + loopingTally/testsToMake);

    trace("== Average search time by for each: " + forEachTally/testsToMake);

    trace("== Average search time by indexOf: " + indexOfTally/testsToMake);

}
复制代码这是一些样例输出:== Average search time by looping: 134.6

== Average search time by for each: 26

== Average search time by indexOf: 6.4
复制代码可以看出,使用indexOf()方法比另外两种方法都要快。

点击Start按钮开始测试: 这个例子的源码在本文最上方ArraySearchMethods.zip中。Flash Professional CS5版本FLA文件是ArraySearchMethods_CS5.fla。

深入学习

关于排序数组的更多信息,请参考ActionScript 3.0 Reference的Array类,包含使用sort()和sortOn()方法的详细介绍。 播放列表的例子在 ActionScript 3.0 Developer's Guide中的"Working with arrays"章节。

热点排行