欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  php教程

《PHP程序设计》 第五章 中场一:数据库连接

程序员文章站 2022-03-27 15:47:49
...

第五章 中场一:数据库连接

前两章中集中介绍了PHP语言,现在停下来开始创建一个应用程序。在本章中将创建一个连接MySQL数据库的应用程序。

在学习过前面两章之后,你肯定已经学会了怎样处理PHP内部数据,以及怎样编写语句和函数。下一步从逻辑上讲,应该到了学习如何用SQL(结构化查询语句)处理PHP的外部数据的时候了。但是,在开始学习这部分内容前,让我们暂时中断基本原理的学习,放松一下。
让我带你一道探究PHP应用程序的开发历程。从字面意义上讲,每一个应用程序都应该是独特的,但是每一个应用程序也都能在此前的工作基础上,即一系列通用功能的基础上构建。我建议将这两项技术混合在一起。盲目地使用前人已编写好的功能,会剥夺在程序中加入新特色,还会阻止为提高函数的效率而去修改旧函数。从另一方面讲,使用已有函数意味着能更快的开发出应用程序。所以必须在这两个极端中把握好自己,才能成为一个优秀的程序设计员。

注意:如果现在还对HTML不熟悉,那么现在就是开始学习的时间了。本书中假定你已经熟悉HTML了。如果还不熟悉HTML表格和表单的话,很快就会被搞糊涂的。

5.1
开端
每当开始一个新项目时,我喜欢从一个新的空目录开始。在这里,让我们把这个目录称为phpbook/ch05。当然,这个目录必须在Web服务器的根目录下。如果你是按照第一章的指令安装PHP的话,那么,Web服务器的根目录就应该为/usr/local/apache/htdocs。接着,我们将创建一个名为menu.php3的文件,其中包含有一个后台管理任务菜单,如清单5.1所示。

清单5.1 menu.php3
php require('common.inc'); ?>
php affy_header('Administrative Menu') ?>

Administrative Menu


ol>
li>href="connect.php3">Creat Database
Connection
li>
ol>
php affy_footer() ?>

在文件common.inc中包含对函数affy_headeraffy_footer的定义。这些函数在本章中稍后还会出现。
5.2
创建连接
当点击Create Database Connect(创建数据库连接)联接时,将会执行connect.php3文件,该文件将尝试连接在第二章学习中安装的MySQL数据库服务器。
清单5.2 显示了connect.php3文件使用用户名codebits和密码codebits尝试进行数据库连接,因为在安装MySQL时还没有创建该用户名,所以连接肯定失败。然而失败 -- 最起码在本例中 -- 却是一件好事,因为我们可以看看应该如何处理这个问题。图5.1给出了连接失败后将会显示的错误信息和表单。

清单5.2 connect.php3

Page 107 -108
清单 5.2


Page 108 Figure 5.1

5.1 连接失败时的错误信息显示

当数据库连接失败时,程序会给出一个错误信息提示和一个表单,用户可以在里面输入root用户的密码。就像在本章稍后讲的那样,有了root的密码,就可以创建名为codebits的用户。现在先跳过有关$arr_request数组的部分。
当函数mysql_connect被调用且连接失败时,该函数通常会显示如下信息:

Warnint: MySQL connection Failed: Access denied
for user: 'codebits@localhost' (Using password: YES)

绝大多数应用程序需要精确的控制显示的内容,尤其是高度图形化的应用程序。在函数mysql_connect前加上(@)符号将会抑制错误信息的显示。
注意表单语句的action属性指定点击submit按钮时,将会执行connect.php3文件。这是一个递归程序的例子,也就是说允许PHP文件调用它自己。
应用递归编程技术,可以将有关同一个主题的所有代码编制在同一个文件中。至于什么时候应该将函数组合成一个文件,或将程序分解成几个文件,这得凭经验。我的首要原则是:当实现一个特定功能所编的程序代码超过100行以上时,就要创建一个独立的文件。

5.3
获取HTML表单信息
即使输入一个密码并点击连接数据库,连接仍然会失败,原因是connect.php3还没有使用表单中的输入值去建立数据库连接。
PHP
引擎将每一个表单域放到一个叫做$HTTP_POST_VARS的数组中。在上述给出的例子中,数组有两个元素:usernamepassword。在此程序中可以通过$HTTP_POST_VARS['username']$HTTP_POST_VARS['password']访问表单信息。
使用$HTTP_POST_VARS['password']获得表单中的信息看起来比较简单。但是仍有一些隐藏的问题。首先,要检查表单域的名字(本例中的password)是大写、小写、还是大小写都有。
第二件问题包含的内容与本例关系不大。除了表单方法以外,还可以使用URL来运行PHP脚本,例如:

http://.../connect.php3?username=root&password=password

可以看到,用户名和密码通过URL进行传递,问号标志着域信息的开始,“&”则是域的定界符。幸运的是,PHP引擎也自动分析URL行,并将结果存入$HTTP_GET_VARS数组中。
问题(如果你认为它是的话)在于,程序可以从不止一个地方获得信息 -- 数组$HTTP_GET_VARS和数组 $HTTP_POST_VARS
对待这些(或其它一些的)问题,我的解决办法是创建一个名为$arr_request的数组,它从两个$HTTP数组中获得初始化的信息。在common.inc中可以使用如下编码行对数组$arr_repuest进行数值初始化。

// declare the request array which holds both
// url-based (get) and form-based (post) parameters.
$arr_request = array();

// move the url and form parameters into the
// request array. Form parameters supercede url
// parameters. Additionally, all keys are vonverted
// to lower-case.
If (count($HTTP_GET-VARS)) {
While (list($key, $value) = each ($HTTP_GET_VARS)) {
$arr_request[strtolower($key)] = $value;
}
}
if (count($HTTP_POST_VARS)) {
while (list($key, $value) = each ($HTTP_POST_VARS)) {
$arr_request[strtolower($key)] = $value;
}
}

如果在所有的PHP脚本中都包含有common.inc文件的话,那么不用担心脚本是怎么运行的。所有传过去的信息都以小写形式保存在数组$arr_request中,这就意味着,可以使用$arr_request['username']得到用户名信息。
PHP
提供了数组$HTTP_GET_VARS和数组$HTTP_POST_VARS的替代方式,HTML表单和基于URL的信息都可以直接做为PHP变量进行访问。例如,在PHP 脚本中,一个定义为的域信息可以直接在PHP程序中用$last_name访问,同样的基于URL的信息,比方说,http://www.site.com?last_name=join,能由$last_name获得。不过,我还是比较喜欢使用数组$arr_request,因为对于要循环使用传递给程序的所有信息来讲,这是非常有用的。如果该信息是一个标量,那么它就不适合被循环使用。例如:将所有参数名改为大写,以保证不致于因为使用换档键而破坏程序;或者在错误检测时,会需要显示所有的输入参数。

注意:本节只对CGI(通用网关接口)协议作很简单的介绍,更详细的内容请参阅本书附录A因特网资源中所列内容。

5.4
使用HTML表单信息
既然可以很容易地从PHP脚本程序中存取表单信息,现在是利用这些信息连接数据库的时间了。第一步是检查连接数据库的代码:

$id_link = @mysql_connect('localhost', 'affy', 'affy');

在这行代码中,用户名和密码都是字符串数值。为了能利用表单中的信息,这行代码需要加以变动,用变量代替数值:

$id_link = @mysql_connect(
'localhost',
$username,
$password);

既然用到了变量,那么必须对变量进行初始化。以下代码将执行这种初始化:

if ( count($arr_request) ) {
$username = $arr_request['username'];
$password = $arr_request['password'];
}
else {
$username = 'phpuser';
$password = 'phpuser';
}

当表单信息可用时,函数count的结果将大于1,使得if语句执行真条件的子句,此子句依次从$arr_request数组中取出用户名和密码信息。
当没有表单信息存在时,用户名和密码仍可以用字符串数值进行初始化。
第三种可能性是一个表单有这两个域但没有表单信息。如果调用connect.php3的表单没有usernamepassword域,会发生什么情况呢?如果这样,以上代码将失败。通过直接检查表单字段,而不是只依赖于$arr_request数组的元素数目可以让这段代码的更强壮(即,能处理这种环境下的失败)。例如:

$username = $arr_request['username'];
$password = $arr_request['password'];

if (empty($username)) $username = 'phpuser';
if (empty($password)) $password = 'phpuser';

因为PHP对没有初始化的数组元素将返回空字符串,以上代码的适应性将更强。使用标量比使用数组可以使代码更容易理解,并且在某些方面更有效率。如果这两个变量有一个为空时,这意味着表单没有提供任何值,将使用缺省值。
清单5.3显示了有以上更改的connect.php3文件,可以在上下文看到这两个变化的描述。

清单5.3 connect.php3修订版

Page 112 - 113
微信
  • 《PHP程序设计》 第五章 中场一:数据库连接

    声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn核实处理。