网站地图    收藏   

主页 > php专栏 > php应用 >

深入分析mysql与mysqli的不同之处 - php高级应用

来源:自学PHP网    时间:2014-11-27 22:16 作者: 阅读:

[导读] php中mysql与mysqli对于数据库的连接操作是有所区别的,从最基于的区别来看是ysqli 连接是永久连接,而mysql是非永久连接了,下面我们来深入的分析它们两的区别.首先,两个函数都是用来处理...

深入分析mysql与mysqli的不同之处

php中mysql与mysqli对于数据库的连接操作是有所区别的,从最基于的区别来看是ysqli 连接是永久连接,而mysql是非永久连接了,下面我们来深入的分析它们两的区别.

首先,两个函数都是用来处理数据库操作的,mysqli 连接是永久连接,而mysql是非永久连接.

mysql连接每当第二次使用时,都会重新打开一个新的进程,而mysqli则只使用同一个进程,可以很大程度的减轻服务器端压力.

其次,mysqli封装了诸如事务等一些高级操作,同时封装了数据库操作过程中的很多可用的方法.

具体查看 http://cn.php.net/mysqli

在实际使用中,较多使用到mysqli的事务.

PHP代码示例如下:

  1. $mysqli = new mysqli('localhost','root','','DB_Lib2Test'); 
  2. $mysqli->autocommit(false);//开始事物 
  3. $mysqli->query($sql1); 
  4. $mysqli->query($sql2); 
  5. if(!$mysqli->errno){ 
  6. $mysqli->commit(); 
  7.   echo 'ok'
  8. }else
  9. echo 'err'
  10.  $mysqli->rollback(); 

第二部分,php中关于mysqli和mysql区别

php-mysql 是 php 操作 mysql 资料库最原始的 extension ,php-mysqli 的 i 代表 improvement ,提更了相对进阶的功能,就 extension 而言,本身也增加了安全性。而 pdo (php data object) 则是提供了一个 abstraction layer 来操作资料库。

PHP实例代码如下:

  1. <?php 
  2. mysql_connect($db_host$db_user$db_password); 
  3. mysql_select_db($dn_name); 
  4. $result = mysql_query("SELECT `name` FROM `users` WHERE `location` = '$location'"); 
  5. while ($row = mysql_fetch_array($result, MYSQL_ASSOC)) 
  6. {//开源代码phpfensi.com 
  7. echo $row['name']; 
  8. mysql_free_result($result); 
  9. ?> 

这种方式不能 Bind Column,以前例的 SQL 叙述来说,$location 的地方容易被 SQL Injection,后来于是发展出了 mysql_escape_string() (备注:5.3.0之后弃用) 以及 mysql_real_escape_string() 来解决这个问题,不过这麽一搞,整个叙述会变得?杂且丑陋,而且如果栏位多了,可以想见会是怎样的情形…

PHP实例代码如下:

  1. <?php 
  2. $query = sprintf("SELECT * FROM users WHERE user='%s' AND password='%s'"
  3. mysql_real_escape_string($user), 
  4. mysql_real_escape_string($password)); 
  5. mysql_query($query); 
  6. ?> 

在 php-mysqli 中有了不少进步,除了透过 bind column 来解决上述问题,而且也多援 transaction, multi query,并且同时提供了 object oriented style (下面这段 php-mysqli 范例的写法) 和 procedural style (上面 php-mysql 范例的写法)两种写法…等等.

PHP实例代码如下:

  1. <?php 
  2. $mysqli = new mysqli($db_host$db_user$db_password$db_name); 
  3. $sql = "INSERT INTO `users` (id, name, gender, location) VALUES (?, ?, ?, ?)"
  4. $stmt = $mysqli->prepare($sql); 
  5. $stmt->bind_param('dsss'$source_id$source_name$source_gender$source_location); 
  6. $stmt->execute(); 
  7. $stmt->bind_result($id$name$gender$location); 
  8. while ($stmt->fetch()) 
  9. echo $id . $name . $gender . $location
  10. $stmt->close(); 
  11. $mysqli->close(); 
  12. ?> 

但看到这边又发现了一些缺点,例如得 bind result,这个就有点多馀,不过这其实无关紧要,因为最大的问题还是在于这不是一个抽象(abstraction)的方法,所以当后端更换资料库的时候,就是痛苦的开始…

于是 pdo 就出现了(备注:目前 ubuntu 和 debian 来说,pdo 并没有直接的套件可以安装,而是必须透过 pecl 安装)。

PHP实例代码如下:

  1. roga@carlisten-lx:~$ pecl search pdo 
  2. ======================================= 
  3. Package Stable/(Latest) Local 
  4. PDO 1.0.3 (stable) PHP Data Objects Interface. 
  5. PDO_4D 0.3 (beta) PDO driver for 4D-SQL database 
  6. PDO_DBLIB 1.0 (stable) FreeTDS/Sybase/MSSQL driver for PDO 
  7. PDO_FIREBIRD 0.2 (beta) Firebird/InterBase 6 driver for PDO 
  8. PDO_IBM 1.3.2 (stable) PDO driver for IBM databases 
  9. PDO_INFORMIX 1.2.6 (stable) PDO driver for IBM Informix INFORMIX databases 
  10. PDO_MYSQL 1.0.2 (stable) MySQL driver for PDO 
  11. PDO_OCI 1.0 (stable) Oracle Call Interface driver for PDO 
  12. PDO_ODBC 1.0.1 (stable) ODBC v3 Interface driver for PDO 
  13. PDO_PGSQL 1.0.2 (stable) PostgreSQL driver for PDO 
  14. PDO_SQLITE 1.0.1 (stable) SQLite v3 Interface driver for PDO 
  15. pdo_user 0.3.0 (beta) Userspace driver for PDO 

当透过 pecl 安装装好后,就可以透过以下方式来操作资料库,代码示例:

  1. <?php 
  2. $dsn = "mysql:host=$db_host;dbname=$db_name"
  3. $dbh = new PDO($dsn$db_user$db_password); 
  4. $sql = "SELECT `name`, `location` FROM `users` WHERE `location` = ? , `name` = ?"
  5. $sth = $dbh->prepare($sql); 
  6. $sth->execute(array($location$name)); 
  7. $result = $sth->fetch(PDO::FETCH_OBJ); 
  8. echo $result->name . $result->location; 
  9. $dbh = NULL; 
  10. ?> 

乍看之下,pdo 的程式码好像也没有比较短,那到底好处是什麽呢?

1. pdo 连接资料库时透过 connection string 来决定连接何种资料库。

2. pdo 可以透过 pdo::setattribute 来决定连线时的设定,像是 persistent connection, 回传错误的方式(exception, e_warning, null)。甚至是回传栏位名称的大小写…等等。

3. pdo 支援 bind column 的功能,除了基本的 prepare, execute 以外,也可以 bind 单一栏位,并且指定栏位型态。

4. pdo 是 abstraction layer 所以就算更换储存媒介,需要花的功夫比起来是最少的。

可惜的是,这些东西都已经出现很久了,但还是不够大众化。我想或许是肇因于大家习惯看坊间的书籍学习,但那些书本往往只会介绍最简单最传统的方式。导致很多人还是在用 mysql 这种方直接连资料库。

最喜爱透过 dbi 来连接资料库,像是 activerecord 以及 propel orm(object-relational mapping)。

例如说以 activerecord 为例,如果要实现这样的 sql 叙述…

insert into `users` (id, name, gender, location) values(1, 'roga', 'male', 'tpe')

以 pdo 来写是,代码示例:

  1. <?php 
  2. $sql = "INSERT INTO `users` (id, name, gender, location) VALUES(?, ?, ?, ?)"
  3. $sth = $dbh->prepare($sql); 
  4. $sth->execute(array(1, 'roga''male''tpe')); 
  5. ?> 

但以 ActiveRecord 来说的话,则是,代码示例:

  1. <?php 
  2. $user = new User(); 
  3. $user->id = 1; 
  4. $user->name = 'roga'
  5. $user->gender = 'male'
  6. $user->location = 'tpe'
  7. $user->save(); 
  8. ?> 

后者在语法上是不是简洁很多呢,而且也大幅降低对 SQL 语言的依赖性!(不同资料库对 SQL 实作的问题可参考 Comparison of different SQL implementations)

以上是一些简单的介绍,如有疏漏谬误也欢迎大家补充。

mysql是非持继连接函数而mysqli是永远连接函数,也就是说mysql每次链接都会打开一个连接的进程而mysqli多次运行mysqli将使用同一连接进程,从而减少了服务器的开销

有些朋友在编程的时候,使用new mysqli('localhost', usenamer', 'password', 'databasename');总是报错,Fatal error: Class 'mysqli' not found in d:...

mysqli类不是php自带的吗?

不是默认开启的,win下要改php.ini,去掉php_mysqli.dll前的;,linux下要把mysqli编译进去。

一:Mysqli.dll是一个允许以对象的方式或者过程操作数据库的,它的使用方式也很容易。这里就几个常见的操作和 mysql.dll做一个对比。

1、mysql.dll(可以理解为函数式的方式):

  1. $conn = mysql_connect('localhost''user''password'); //连接mysql数据库 
  2. mysql_select_db('data_base'); //选择数据库 
  3.  
  4. $result = mysql_query('select * from data_base');//这里有第二个可选参数,指定打开的连接 
  5. $row = mysql_fetch_row( $result ) ) //为了简单,这里只取一行数据 
  6. echo $row[0]; //输出第一个字段的值 

mysqli也有过程式的方式,只不过开始贯以mysqli的前缀,其他都差不多,如果mysqli以过程式的方式操作的话,有些函数必须指定资源,比如说 mysqli_query(资源标识,SQL语句),并且资源标识的参数是放在前面的,而 mysql_query(SQL语句,'可选')的资源标识是放在后面的,并且可以不指定,它默认是上一个打开的连接或资源.

2、mysqli.dll(对象方式),代码示例:

  1. $conn = new mysqli('localhost''user''password','data_base'); 
  2. //这里的连接是new出来的,最后一个参数是直接指定数据库,不用mysql_select_db()了 
  3. //也可以构造时候不指定,然后 $conn -> select_db('data_base') 
  4. $result = $conn -> query( 'select * from data_base' ); 
  5. $row = $result -> fetch_row(); //取一行数据 
  6. echo row[0]; //输出第一个字段的值 

二、mysql_fetch_row(),mysql_fetch_array()

这两个函数,返回的都是一个数组,区别就是第一个函数返回的数组是只包含值,我们只能$row[0],$row[1],这样以数组下标来读取数据,而mysql_fetch_array()返回的数组既包含第一种,也包含键值对的形式,可以这样读取数据,假如数据库的字段是 username,passwd,代码示例:

$row['username'], $row['passwd']

而且,如果用($row as $kay => $value)来操作的话,还以直接取得数据库的字段名称.

更主要的是mysqli是php5提供的新函数库,(i)表示改进,其执行速度更快.

自学PHP网专注网站建设学习,PHP程序学习,平面设计学习,以及操作系统学习

京ICP备14009008号-1@版权所有www.zixuephp.com

网站声明:本站所有视频,教程都由网友上传,站长收集和分享给大家学习使用,如由牵扯版权问题请联系站长邮箱904561283@qq.com

添加评论