五、丰富的数组函数 
PHP 4.0中新添加了30个与组数有关的函数,其中一些常见的函数可以判断一个数组中是否包含某个元素,对一个数组中的元素进行计数,添加或删除数组中的元素或者对数组中的元素进行排序。
如果有一个很大的数组,而你需要找出其中是否包含一个特定的元素,就可以使用in_array()。下面的例子将显示“Not found in this array”,因为在一个名字为$namesArray的数组中查找Albert,而在$namesArray数组中不存在这样一个元素。 
 $namesArray = array("Joe", "Jane", "Bob", "Mary", "Paul", "Eddie", "John"); 
$lookingFor = "Albert"; 
if (in_array($lookingFor, $namesArray)) { 
echo "You've found it!"; 
} else { 
echo "Not found in this array!"; 
} 
?> 
如果把$lookingFor的值改为Mary,就会得到“You've found it!”的信息,因为Mary是$namesArray数组中的一个元素。 
如果要对一个数组中的元素个数进行计数,只要简单地使用count()函数即可: 
 $namesArray = array("Joe", "Jane", "Bob", "Mary", "Paul", "Eddie", "John"); 
$count = count($namesArray); ?> 
返回的$count的值为7。 
可以在一个数组的开头或结尾处添加元素,还可以使用array_merge()来建立一个包含二个或更多数组中元素的新数组,合并时,元素的顺序会按指定的顺序排列,如果原来的数组是被排过序的,在合并后需要对它重新排序。 
我们可以首先利用array_push()在数组的结尾处添加一个元素: 
 /* 建立一个数组 */ 
$fruitArray = array("apple", "orange", "banana", "kiwi", "pear"); 
/* 向数组中添加元素 */ 
array_push($fruitArray, "grape", "pineapple", "tomato"); 
/*显示每个元素及其序号*/ 
while (list($key,$value) = each($fruitArray)) { 
echo "$key : $value
"; 
} 
?> 
运行上面的程序将得到下面的结果: 
0 : apple 
1 : orange 
2 : banana 
3 : kiwi 
4 : pear 
5 : grape 
6 : pineapple 
7 : tomato 
如果需要在数组的开头添加元素,其代码与上面的代码差不多,唯一的不同之处是需要用array_unshift()代替array_push()。 
 
/* 建立一个数组*/ 
$fruitArray = array("apple", "orange", "banana", "kiwi", "pear"); 
/* 向数组中添加元素*/ 
array_unshift($fruitArray, "grape", "pineapple", "tomato"); 
/* 显示每个元素及其序号*/ 
while (list($key,$value) = each($fruitArray)) { 
echo "$key : $value
"; 
} 
?> 
运行上面的程序将得到下面的结果: 
0 : grape 
1 : pineapple 
2 : tomato 
3 : apple 
4 : orange 
5 : banana 
6 : kiwi 
7 : pear 
array_merge()函数可以把二个或更多的数组合并为一个数组。 
 /*建立第一个数组*/ 
$fruitArray = array("apple", "orange", "banana", "kiwi", "pear"); 
/*/建立第二个数组*/ 
$vegArray = array("carrot", "green beans", "ASParagus", "artichoke", "corn"); 
/*把这二个数组合并为一个数组*/ 
$goodfoodArray = array_merge($fruitArray, $vegArray); 
/* 显示每个元素及其序号*/ 
while (list($key,$value) = each($goodfoodArray)) { 
echo "$key : $value
"; 
} 
?> 
运行上面的脚本将得到下面的结果: 
0 : apple 
1 : orange 
2 : banana 
3 : kiwi 
4 : pear 
5 : carrot 
6 : green beans 
7 : asparagus 
8 : artichoke 
9 : corn 
现在我们已经掌握了如何添加元素和合并数组,我们再来看看如何从一个数组中删除元素。从一个数组的末尾删除一个元素可以使用array_pop()函数,使用array_shift()函数可以从一个数组的开头删除一个元素。尽管使用array_pop()或 array_shift()从数组中删除了一个元素,你还可以把这个元素当作一个变量来使用。 
使用array_pop()从一个数组的末尾删除一个元素: 
 
/*建立一个数组*/ 
$fruitArray = array("apple", "orange", "banana", "kiwi", "pear"); 
/* 从数组的末尾删除一个元素*/ 
$popped = array_pop($fruitArray); 
/* 显示删除后数组的内容和你删除的元素*/ 
while (list($key,$value) = each($fruitArray)) { 
echo "$key : $value
"; 
} 
echo "
and finally, in $popped: $popped"; 
?> 
运行上面的脚本会得到下面的结果: 
0 : apple 
1 : orange 
2 : banana 
3 : kiwi 
and finally, in $popped: pear 
我们再来讨论一个从一个数组的末尾删除元素的例子: 
 
/* 建立一个数组*/ 
$fruitArray = array("apple", "orange", "banana", "kiwi", "pear"); 
/*从一个数组的开始删除一个元素*/ 
$shifted = array_shift($fruitArray); 
/* 显示删除后数组的内容和你删除的元素*/ 
while (list($key,$value) = each($fruitArray)) { 
echo "$key : $value
"; 
} 
echo "
and finally, in $shifted: $shifted"; 
?> 
运行上述脚本会得到如下的显示结果: 
0 : orange 
1 : banana 
2 : kiwi 
3 : pear 
and finally, in $shifted: apple 
另外还有几个函数可以对数组中的元素进行排序,但在这里我们将只简要介绍基本的排序函数,说明排序的过程: 
 /*建立一个数组*/ 
$fruitArray = array("apple", "orange", "banana", "kiwi", "pear"); 
/* 对数组进行排序*/ 
sort($fruitArray); 
/*显示每个元素及其序号*/ 
while (list($key,$value) = each($fruitArray)) { 
echo "$key : $value
"; 
} 
?> 
运行上述的脚本会得到如下的显示结果: 
0 : apple 
1 : banana 
2 : kiwi 
3 : orange 
4 : pear
 六、动态图像的创建 
只要安装一些第三方的库文件并具有一定的几何知识,就可以利用PHP来创建和处理图像了。事实上,这不需要太多的几何知识,因为我大学没有毕业,仍然可以利用PHP创建图像。 
在使用基本的图像创建函数之前,需要安装GD库文件。如果要使用与JPEG有关的图像创建函数,还需要安装jpeg-6b,如果要在图像中使用Type 1型字体,则必须安装t1lib。 
在建立图像创建环境之前,还需要作一些准备工作。首先,安装t1lib;其次安装jpeg-6b,然后再安装GD库文件。在安装时一定要按这里给定的顺序进行安装,因为在编译GD为库时会用到jpeg-6b,如果没有安装jpeg-6b,在编译时就会出错。 
在安装完这三个组件后,还需要重新配置PHP,这也是你对采用DSO方式安装PHP感到庆幸的地方之一。运行make clean,然后在当前的配置中添加下面的内容: 
--with-gd=[/path/to/gd] 
--with-jpeg-dir=[/path/to/jpeg-6b] 
--with-t1lib=[/path/to/t1lib] 
完成添加后执行make命令,然后再执行make install命令。重新启动Apache后运行phpinfo()来检查一下新的设置是否生效了。现在就可以开始图像创建工作了。 
根据所安装的GD库文件的版本不同,你也许能或者不能创建GIF或PNG格式的图形文件,如果安装的是gd-1.6或以前的版本,可以使用GIF格式的文件但不能创建PNG格式,如果安装的是gd-1.6以后的版本,可以创建PNG文件但不能创建GIF格式的文件。 
创建一幅简单的图像也需要用到许多的函数,我们将一步一步地进行说明。 
在这个例子中,我们将创建一个PNG格式的图像文件,下面的代码是一个包含所创建的图像的MIME类型的头部: 
 header ("Content-type: image/png"); 
使用ImageCreate()创建一个代表空白图像的变量,这个函数要求以像素为单位的图像大小的参数,其格式是ImageCreate(x_size, y_size)。如果要创建一个大小为250X250的图像,就可以使用下面的语句: 
$newImg = ImageCreate(250,250); 
由于图像还是空白的,因此你可能会希望用一些彩色来填充它。然而,你需要首先使用ImageColorAllocate()函数用其RGB值为这种颜色指定一个名字,这一函数的格式为ImageColorAllocate([image], [red], [green], [blue])。如果要定义天蓝色,可以使用如下的语句: 
$skyblue = ImageColorAllocate($newImg,136,193,255); 
接下来,需要使用ImageFill()函数用这种颜色填充这个图像,ImageFill()函数有几个版本,例如ImageFillRectangle()、ImageFillPolygon()等。为简单起见,我们通过如下的格式使用ImageFill()函数: 
ImageFill([image], [start x point], [start y point], [color]) 
ImageFill($newImg,0,0,$skyblue); 
最后,建立图像后释放图像句柄和所占用的内存: 
ImagePNG($newImg); 
ImageDestroy($newImg); ?> 
这样,创建图像的全部代码如下所示: 
 header ("Content-type: image/png"); 
$newImg = ImageCreate(250,250); 
$skyblue = ImageColorAllocate($newImg,136,193,255); 
ImageFill($newImg,0,0,$skyblue); 
ImagePNG($newImg); 
ImageDestroy($newImg); 
?> 
如果把这个脚本文件保存为skyblue.php,并用浏览器访问它,就会看到一个天蓝色的250X250的PNG格式的图像。 
我们还可以使用图像创建函数对图像进行处理,例如把一个较大图像作成一个小图像: 
假设你有一幅图像,想从中裁剪出一个35X35大小的图像。你所需要作的是创建一个35X35大小的空白图像,创建一个包含原来图像的图像流,然后把一个经过调整大小的原来的图像放到新的空白图像中。 
要完成这一任务的关健函数是ImageCopyResized(),它要求的格式如下所示:ImageCopyResized([new image handle],[original image handle],[new image X], [new Image Y], [original image X], [original image Y], [new image X], [new image Y], [original image X], [original image Y])。 
 /* 发送一个头部,以便让浏览器知道该文件所包含的内容类型*/ 
header("Content-type: image/png"); 
/* 建立保存新图像高度和宽度的变量*/ 
$newWidth = 35; 
$newHeight = 35; 
/* 建立给定高度和宽度的新的空白图像*/ 
$newImg = ImageCreate($newWidth,$newHeight); 
/* 从原来较大的图像中得到数据*/ 
$origImg = ImageCreateFromPNG("test.png"); 
/*拷贝调整大小后的图像,使用ImageSX()、ImageSY()得到原来的图像在X、Y方面上的大小 */ 
ImageCopyResized($newImg,$origImg,0,0,0,0,$newWidth,$newHeight,ImageSX($origImg),ImageSY($origImg)); 
/*创建希望得到的图像,释放内存 */ 
ImagePNG($newImg); 
ImageDestroy($newImg); ?> 
如果把这一小段脚本保存为resized.php,然后用浏览器对它进行访问,就会看到一个35X35大小的PNG格式的图形。 
七、基于PHP的用户认证 
如果希望在每一段脚本上都进行口令保护,可以结合使用header()语句、$PHP_AUTH_USER和$PHP_AUTH_PW来建立基本的认证方案,通常的基于服务器的提问/响应顺序都如下所示: 
1、用户从服务器上请求一个文件。如果这个文件在服务器上是被保护的,则在响应的头部向用户返回一个401(示经授权的用户)字符串。 
2、浏览器收到这个响应后,弹出要求用户输入用户名/口令的对话框。 
3、用户在对话框中输入一个用户名和口令,点击OK按钮将信息返回服务器供认证使用。 
4、如果用户名和口令有效,被保护的文件将向用户开放,只要用户还在使用文件,认证会一直有效。 
一段简单的PHP脚本文件通过向用户发送一个适当的能够引起自动显示用户名/口令对话框的HTTP头部就可以模仿HTTP的提问/响应系统,PHP把用户在用户名/口令对话框中输入的信息存储在$PHP_AUTH_USER和$PHP_AUTH_PW中,使用这二个变量,就可以与存储在文本文件、数据库等文件中的用户名/口令进行比较。 
这个例子采用了二个硬编码的值进行认证,但无论用户名和口令放在什么地方,其原理都是相同的。
 
/* 检查$PHP_AUTH_USER和$PHP_AUTH_PW中的值*/ 
if ((!isset($PHP_AUTH_USER)) || (!isset($PHP_AUTH_PW))) { 
/* 如果没有值,则发送一个能够引发对话框出现的头部*/ 
header('WWW-Authenticate: Basic realm="My Private Stuff"'); 
header('HTTP/1.0 401 Unauthorized'); 
echo 'Authorization Required.'; 
exit; 
} else if ((isset($PHP_AUTH_USER)) && (isset($PHP_AUTH_PW))){ 
/* 变量中有值,检查它们是否正确*/ 
if (($PHP_AUTH_USER != "validname") || ($PHP_AUTH_PW != "goodpassword")) { 
/* 如果输入的用户名和口令中有一个不正确,则发送一个能够引发对话框出现的头部 */ 
header('WWW-Authenticate: Basic realm="My Private Stuff"'); 
header('HTTP/1.0 401 Unauthorized'); 
echo 'Authorization Required.'; 
exit; 
} else if (($PHP_AUTH_USER == "validname") || ($PHP_AUTH_PW == "goodpassword")) { 
/* 如果二个值都正确,显示成功的信息 */ 
echo "
You're authorized!
"; 
} 
} 
?> 
需要注意的是,如果你使用的是基于文件的保护机制,它并不能保证目录中所有的文件的安全。它可能保护大部分的文件,如果你认为它能够保护给定目录中的所有文件,你的这种认识就需要变变了。 
八、PHP和COM 
如果你喜欢冒险,并且在Windows上运行CGI、ISAPI或Apache模块版本的PHP,就可以访问COM的函数。好了,详细解释COM的工作就交给微软和许多大部头的书了,为了能简单地了解一下COM的功能,下面是一小段常见的脚本。 
这一段PHP脚本在后端启动微软的字处理Word,打开一个新的文档,输入一些文字,保存文档,并关闭Word。 
 
// 建立一个指向新COM组件的索引 
$word = new COM("word.application") or die("Can't start Word!"); 
// 显示目前正在使用的Word的版本号 
echo "Loading Word, v. {$word->Version}
"; 
// 把它的可见性设置为0(假),如果要使它在最前端打开,使用1(真) 
// to open the application in the forefront, use 1 (true) 
$word->Visible = 0; 
// 在Word中创建新的文档 
$word->Documents->Add(); 
// 在新文档中添加文字 
$word->Selection->TypeText("Testing 1-2-3..."); 
//把文档保存在Windows临时目录中 
$word->Documents[1]->SaveAs("/Windows/temp/comtest.doc"); 
// 关闭与COM组件之间的连接 
$word->Quit(); 
// 在屏幕上显示其他信息 
echo "Check for the file..."; 
?> 
如果你有一个intranet网站,数据存储在SQL Server中,用户需要这些数据的Excel格式,则可以让PHP运行必要的SQL查询并对输出进行格式化,然后利用COM打开Excel,把数据转化为Excel格式的数据,然后把数据保存在用户的台式机上。 
九、PHP和Java 
PHP另一个有趣的功能是它可以调用现有的Java 对象中的方法,使得你可以在基于Java的应用中集成PHP。如果要在工作中推广PHP的应用,这一功能就非常有用了,你得到的结果是,“这里的一切都是基于Java的。” 
要利用这一功能,你的服务器上必须安装有JVM(Java虚拟机器)。如果安装的是由Sun、Kaffe、IBM或Blackdown的JDK,就已经安装有了JVM。 
在配置PHP时,需要在配置文件中添加with-java小节,然后修改php.ini文件中的一部分,对php.ini文件的修改主要是需要添加下面的内容: 
[Java] 
java.library.path=/path/to/library 
java.class.path=/classpath/ 
extension_dir=/path/to/extensions 
extension=libphp_java.so 
需要注意的是,所作的修改与你的安装类型有关,你需要阅读PHP安装目录下ext/java目录中README文件,学习如何配置Java功能。 
下面是一段如何创建一个新的Java对象的PHP脚本的例子,这段脚本将访问并在显示器是显示一些Java属性。它与COM的例子同样有趣,应该会给我们一些启发。 
 
$system = new Java("java.lang.System"); 
echo "
Java version = " . $system->getProperty("java.version") . "
"; 
echo "Java vendor = " . $system->getProperty("java.vendor") . "
"; 
?> 
如果你有Java知识,将会对开发工作带来很大的帮助,这种集成的能力是未来PHP被接受和增长的关健。 
十、PHP和XML 
PHP中包含有一个可选的支持Expat解析的XML扩展,利用PHP中与XML相关的函数,可以创建一个分析程序来处理有效的XML文档。如果你使用的是1.3.7版或版本更高的Apache软件,就不需要再安装额外的库文件了,你所需要作的就只是配置PHP中的with-xml。 
象Java和COM一样,PHP中对XML的支持也很有趣,而且增长也很快,如果你了解Expat或LibXML,请加入这一方面的开发吧。