PHP开发笔记系列(一)-PDO使用

之前一段时间,开始了php的研究,看了关于PDO的一些资料,发现不错,整理和总结一下,作为开发笔记,留待日后使用,《PHP开发笔记系列(一)-PDO使用》。

PDO是PHP Data Objects的简称,是一种数据库访问抽象层。PDO是用于多种数据库的一致接口。类比的说,PDO做的事情类似于JAVA中的持久层框架(Hibernate、OpenJPA)的功能,为异构数据库提供一个统一的编程接口,这样就不必再使用mysql_*、pg_*这样的函数,也不必再写自己的"GenericDAO"了。PDO在PHP5.1的时候一起发布,所以我们用的PHP5.2、PHP5.3都已经可以使用。

为了方便,我们使用MySQL5来做演示。

0. 建立实验环境数据库及相关表

Sql代码  

  1. CREATE TABLE `blog` (
  2. `id` int(10) NOT NULL AUTO_INCREMENT,
  3. `title` varchar(255) NOT NULL,
  4. PRIMARY KEY (`id`)
  5. ) ENGINE=InnoDB AUTO_INCREMENT=16 DEFAULT CHARSET=latin1

1. 使用PDO访问数据库 
    通过PDO访问数据库的步骤是:a)指定dsn、username、password,b)通过#a中的设置构造PDO对象,代码如下:

Php代码  

  1. file:pdo-access.php
  2. url:http://localhost:88/pdo/pdo-access.php
  3. <?php
  4. // 设置dsn、username、passwd
  5. $dsn = ‘mysql:host=localhost;dbname=pdotest‘;
  6. $username = ‘root‘;
  7. $passwd = ‘password‘;
  8. // 构造PDO对象
  9. try {
  10. $dbh = new PDO($dsn, $username, $passwd);
  11. echo ‘connect to database successfully!‘;
  12. } catch (Exception $e) {
  13. echo ‘Fail to connect to database!\n‘;
  14. echo $e->getMessage();
  15. }
  16. ?>

    备注:DSN即Data Source Name-数据源名称,提供数据库的连接信息,包括三部分,PDO驱动名称(MySQL、SQLite、PostgreSQL等)、冒号和驱动特定的语法。但是一般情况下,我们都很难记住这些,可以下载个php manual查,也可以到php的官网查。

2. 使用Query方法查询数据 
    在#1的基础上,连接数据库成功后,构造SQL语句,调用query方法返回结构数组,通过foreach进行数据结果遍历,代码如下:

Php代码  

  1. file:pdo-query.php
  2. url:http://localhost:88/pdo/pdo-query.php?title=title1
  3. <?php
  4. $dsn = ‘mysql:host=localhost;dbname=pdotest‘;
  5. $username = ‘root‘;
  6. $passwd = ‘password‘;
  7. try {
  8. $dbh = new PDO($dsn, $username, $passwd);
  9. echo ‘connect to database successfully!‘."\r\n";
  10. $title = ‘title1‘;
  11. // 构造SQL语句
  12. $sql = "SELECT * FROM blog WHERE title = ‘".$title."‘";
  13. // 执行查询并遍历结果
  14. foreach ($dbh->query($sql) as $row){
  15. print $row[‘id‘]."\t";
  16. print $row[‘title‘]."\t";
  17. }
  18. } catch (PDOException $e) {
  19. echo ‘Errors occur when query data!\n‘;
  20. echo $e->getMessage();
  21. }
  22. ?>

    备注:一般情况下, 通过构造SQL语句的方法来进行query、update、insert、delete,都会需要指定where条件,因此不可避免的需要防止SQL注入的问题出现。

例如,正常情况下,当用户输入“title1”时,我们构造的sql语句会是SELECT * FROM blog WHERE title=‘title1‘,但是对SQL比较熟悉的用户会输入‘OR id LIKE ‘%,此时我们构造的SQL就会变成SELECT * FROM blog where title=‘‘ OR id LIKE ‘%‘,这样整张blog 表中的数据都会被读取,因此需要避免,所以需要用到quote方法,把所有用户提供的数据进行转移,从而防止SQL注入的发生。使用quote方法后的sql为$sql = "SELECT * FROM blog WHERE title = ".$dbh->quote($title),转移出来后的sql是SELECT * FROM blog WHERE title = ‘\‘OR id LIKE \‘%‘,把所有的单引号(‘)都转移了。

3. 使用prepare和execute方法查询数据 
    如果我们用到的SQL查询是使用频率不高的查询,那么使用query或prepare和execute方法来查询都无太大差别,查询速度也不会差太远。两者不同的是,使用query时,php向数据库发送的sql,每执行一次都需要编译一次,而使用prepare和execute方法,则不需要,因此做大并发量的操作时,使用prepare和execute方法的优势会更加明显。

使用prepare和execute方法的步骤不多,a)构造SQL,b)将SQL传入PDO->prepart方法,得到一个PDOStatement对象,3)调用PDOStatement对象的execute方法,4)通过PDOStatement->fetch或PDOStatement->fetchObject遍历结果集。代码如下:

Php代码  

  1. file:pdo-prepare-fetch.php
  2. url:http://localhost:88/pdo/pdo-prepare-fetch.php?title=title1
  3. <?php
  4. $dsn = ‘mysql:host=localhost;dbname=pdotest‘;
  5. $username = ‘root‘;
  6. $passwd = ‘password‘;
  7. // 从请求获取title参数值
  8. $title = $_GET[‘title‘];
  9. try {
  10. $dbh = new PDO($dsn, $username, $passwd);
  11. echo ‘connect to database successfully!‘."<br/>";
  12. // 构造SQL语句,使用绑定变量
  13. $sql = "SELECT * FROM blog WHERE title = :title";
  14. // 编译SQL
  15. $stmt = $dbh->prepare($sql);
  16. // 为绑定变量赋值
  17. $stmt->bindParam(":title", $title, PDO::PARAM_STR);
  18. // 执行SQL
  19. $stmt->execute();
  20. // 以联合数组方式获取结果,并遍历结果
  21. while ($row = $stmt->fetch(PDO::FETCH_ASSOC)) {
  22. print $row[‘id‘]."\t";
  23. print $row[‘title‘]."\t";
  24. }
  25. } catch (PDOException $e) {
  26. echo ‘Errors occur when query data!\n‘;
  27. echo $e->getMessage();
  28. }
  29. ?>

除了使用上面的PDO::FETCH_ASSOC返回联合数组外,还可以使用fetchObject方法,返回结果集对象,代码如下:

Php代码  

  1. file:pdo-prepare-fetch-object.php
  2. url:http://localhost:88/pdo/pdo-prepare-fetch-object.php?title=title1
  3. <?php
  4. $dsn = ‘mysql:host=localhost;dbname=pdotest‘;
  5. $username = ‘root‘;
  6. $passwd = ‘password‘;
  7. $title = $_GET[‘title‘];
  8. try {
  9. $dbh = new PDO($dsn, $username, $passwd);
  10. echo ‘connect to database successfully!‘."<br/>";
  11. $sql = "SELECT * FROM blog WHERE title = :title";
  12. $stmt = $dbh->prepare($sql);
  13. $stmt->bindParam(":title", $title, PDO::PARAM_STR);
  14. $stmt->execute();
  15. // 以对象数组方式获取结果,并遍历结果
  16. while ($row = $stmt->fetchObject()) {
  17. print $row->id."\t";
  18. print $row->title."\t";
  19. }
  20. } catch (Exception $e) {
  21. echo ‘Errors occur when query data!\n‘;
  22. echo $e->getMessage();
  23. }
  24. ?>

4. 设置PDO的错误级别 
    PDO的错误级别分成PDO::ERRMODE_SILENT(默认)、PDO::ERRORMODE_WARNING、PDO::ERRORMODE_EXCEPTION三种。 
    PDO::ERRMODE_SILENT级别,当出现错误时,会自动设置PDOStatement对象的errorCode属性,但不进行任何其他操作,因此需要我们手工检查是否出现错误(使用empty($stmt->errorCode())),否则程序将继续走下去。 
    PDO::ERRORMODE_WARNING级别,基本与PDO::ERRMODE_SILENT一致,都是需要使用empty($stmt->errorCode())手工检查。 
    只需要在创建PDO对象后,加入以下代码即可:$dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_SILENT);或$dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_WARNING); 
    PDO::ERRORMODE_WARNING级别,当出现错误时,系统将抛出一个PDOException,并设置errorCode属性,程序可以通过try{...}catch{...}进行捕捉,否则未catch的exception会导致程序中断,加入以下代码即可:$dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);

Php代码  

  1. <?php
  2. ...
  3. try {
  4. ...
  5. } catch (Exception $e) {
  6. echo ‘Errors occur when operation!‘."<br/>";
  7. // 获取Exception信息
  8. echo $e->getMessage()."<br/>";
  9. // 获取错误码
  10. echo $e->getCode()."<br/>";
  11. // 获取出错文件名
  12. echo $e->getFile()."<br/>";
  13. // 获取出错行
  14. echo $e->getLine()."<br/>";
  15. // 把异常以字符串返回
  16. echo $e->getTraceAsString();
  17. }
  18. ?>

5. 使用prepare和execute方法插入/更新数据 
    方法和#3中进行查询的差不多,只是构造的SQL语句是insert语句或update语句,代码如下:

Php代码  

  1. file:pdo-prepare-insert.php
  2. url:http://localhost:88/pdo/pdo-insert.php?title=title11
  3. <?php
  4. $dsn = ‘mysql:host=localhost;dbname=pdotest‘;
  5. $username = ‘root‘;
  6. $passwd = ‘password‘;
  7. $title = $_GET[‘title‘];
  8. try {
  9. $dbh = new PDO($dsn, $username, $passwd);
  10. $dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  11. echo ‘connect to database successfully!‘."<br/>";
  12. // 构造Insert语句
  13. $sql = "INSERT INTO blog(title) VALUES(:title)";
  14. $stmt = $dbh->prepare($sql);
  15. $stmt->bindParam(":title", $title);
  16. $stmt->execute();
  17. } catch (Exception $e) {
  18. echo ‘Errors occur when query data!\n‘;
  19. echo $e->getMessage();
  20. }
  21. ?>

Php代码  

  1. file:pdo-prepare-update.php
  2. url:http://localhost:88/pdo/pdo-update.php?id=1&title=title12
  3. <?php
  4. $dsn = ‘mysql:host=localhost;dbname=pdotest‘;
  5. $username = ‘root‘;
  6. $passwd = ‘password‘;
  7. $id = $_GET[‘id‘];
  8. $title = $_GET[‘title‘];
  9. try {
  10. $dbh = new PDO($dsn, $username, $passwd);
  11. $dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  12. echo ‘connect to database successfully!‘."<br/>";
  13. // 构造update语句
  14. $sql = "UPDATE blog SET title=:title where id=:id";
  15. $stmt = $dbh->prepare($sql);
  16. $stmt->bindParam(":id", $id);
  17. $stmt->bindParam(":title", $title);
  18. $stmt->execute();
  19. } catch (Exception $e) {
  20. echo ‘Errors occur when query data!\n‘;
  21. echo $e->getMessage();
  22. }
  23. ?>

6. 获取返回的行数 
    使用#3中的prepare和execute方法,然后将sql语句改成count的,例如SELECT COUNT(id) FROM article ...,代码如下:

Php代码  

  1. file:pdo-prepare-fetch-column.php
  2. url:http://localhost:88/pdo/pdo-prepare-fetch-column.php?id=1&title=title12
  3. <?php
  4. $dsn = ‘mysql:host=localhost;dbname=pdotest‘;
  5. $username = ‘root‘;
  6. $passwd = ‘password‘;
  7. try {
  8. $dbh = new PDO($dsn, $username, $passwd);
  9. $dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_SILENT);
  10. echo ‘connect to database successfully!‘."<br/>";
  11. // 构造count语句
  12. $sql = "SELECT COUNT(id) FROM blog";
  13. $stmt = $dbh->prepare($sql);
  14. $stmt->execute();
  15. // 使用fetchColumn获取0列值
  16. echo $stmt->fetchColumn()." rows returned!";
  17. } catch (Exception $e) {
  18. echo ‘Errors occur when query data!\n‘;
  19. echo $e->getMessage();
  20. }
  21. ?>

7. 获取受影响的行数 
    使用#3中的prepare和execute方法,然后将SQL语句改成insert、update、delete语句即可,代码如下:

Php代码  

  1. file:pdo-prepare-row-count.php
  2. url:http://localhost:88/pdo/pdo-prepare-row-count.php?id=1
  3. <?php
  4. $dsn = ‘mysql:host=localhost;dbname=pdotest‘;
  5. $username = ‘root‘;
  6. $passwd = ‘password‘;
  7. $id = $_GET[‘id‘];
  8. try {
  9. $dbh = new PDO($dsn, $username, $passwd);
  10. $dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_SILENT);
  11. echo ‘connect to database successfully!‘."<br/>";
  12. $sql = "DELETE FROM blog WHERE id=:id";
  13. $stmt = $dbh->prepare($sql);
  14. $stmt->bindParam(":id", $id);
  15. $stmt->execute();
  16. // 获取update、insert、delete操作后影响的行数
  17. echo $stmt->rowCount()." rows affected!";
  18. } catch (Exception $e) {
  19. echo ‘Errors occur when data operation!\n‘;
  20. echo $e->getMessage();
  21. }
  22. ?>

8. 获得新插入行的ID值 
    为数据库表插入新数据行时,我们需要获得刚刚插入的新行的ID值,此时我们需要使用到PDO的lastInsertId()方法,代码如下:

Php代码  

  1. file:pdo-prepare-last-insertid.php
  2. url:http://localhost:88/pdo/pdo-prepare-last-insertid.php?title=title13
  3. <?php
  4. $dsn = ‘mysql:host=localhost;dbname=pdotest‘;
  5. $username = ‘root‘;
  6. $passwd = ‘password‘;
  7. $title = $_GET[‘title‘];
  8. try {
  9. $dbh = new PDO($dsn, $username, $passwd);
  10. $dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  11. echo ‘connect to database successfully!‘."<br/>";
  12. $sql = "INSERT INTO blog(title) VALUES(:title)";
  13. $stmt = $dbh->prepare($sql);
  14. $stmt->bindParam(":title", $title);
  15. $stmt->execute();
  16. // 获取上一个之行的insert语句插入的数据的id值
  17. echo $dbh->lastInsertId();
  18. } catch (Exception $e) {
  19. echo ‘Errors occur when query data!\n‘;
  20. echo $e->getMessage();
  21. }
  22. ?>

9. 使用PDO进行事务管理 
    事务是进行程序开发时,保证数据ACID(可分性、一致性、独立性、持久性)的工具。要不全部成功,要不全部不成功,这样才能保证关联数据的保存能够达到预期的目的。下面使用PDO的Transaction来进行实验,进行多比数据插入,开启事务,第一句sql是可以正常插入,第二句sql插入出错,检查是否rollback。

Php代码  

  1. file:pdo-prepare-transaction.php
  2. url:http://localhost:88/pdo/pdo-prepare-transaction.php
  3. <?php
  4. $dsn = ‘mysql:host=localhost;dbname=pdotest‘;
  5. $username = ‘root‘;
  6. $passwd = ‘password‘;
  7. try {
  8. $dbh = new PDO($dsn, $username, $passwd);
  9. $dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  10. echo ‘connect to database successfully!‘."<br/>";
  11. // 开启事务
  12. $dbh->beginTransaction();
  13. $sql = "INSERT INTO blog(title) VALUES(:title)";
  14. $stmt = $dbh->prepare($sql);
  15. $stmt->execute(array(‘:title‘=>‘insert title1‘));
  16. $stmt->execute(array(‘:title‘=>NULL));
  17. // 提交事务
  18. $dbh->commit();
  19. } catch (Exception $e) {
  20. echo ‘Errors occur when data operation!\n‘;
  21. echo $e->getMessage();
  22. // 回滚事务
  23. $dbh->rollBack();
  24. }
  25. ?>

10. 使用PDO进行数据库备份 
    使用system函数,将我们构造的mysqldump命令传入即可。下面为了演示,只做了简单的调用。

Php代码  

  1. file:pdo-backup.php
  2. url:http://localhost:88/pdo/pdo-backup.php
  3. <?php
  4. $username="root";
  5. $passwd="password";
  6. $dbname="pdotest";
  7. $file=‘d:/‘.$dbname.‘.sql‘;
  8. // 构造备份命令
  9. $cmd = "mysqldump -u".$username." -p".$passwd." ".$dbname." >".$file;
  10. // 执行备份命令
  11. system($cmd,$error);
  12. if($error){
  13. trigger_error("backup failed".$error);
  14. }
  15. ?>

采用工厂模式:

Php代码  

  1. file:AbstractMySQLDump.php
  2. <?php
  3. require_once ‘MySQLDump_Win.php‘;
  4. abstract class AbstractMySQLDump {
  5. protected $cmd;
  6. abstract function __construct($username, $passwd, $dbname, $file);
  7. // 依据操作系统类型,使用工厂方法构造备份类
  8. public static function factory($username, $passwd, $dbname, $file){
  9. if(strtoupper(substr(PHP_OS, 0, 3))===‘WIN‘){
  10. return new MySQLDump_Win($username, $passwd, $dbname, $file);
  11. }else{
  12. //              implement MySQLDump_NIX($username, $passwd, $dbname, $file);
  13. }
  14. }
  15. // 备份逻辑
  16. public function backup(){
  17. system($this->cmd, $error);
  18. // 判断是否出错及出错逻辑
  19. if($error){
  20. trigger_error("backup failure! command:".$this->cmd." Error:".$error);
  21. }
  22. }
  23. }
  24. ?>

Php代码  

  1. file:MySQLDump_Win.php
  2. <?php
  3. class MySQLDump_Win extends AbstractMySQLDump {
  4. // 覆盖父类的构造方法
  5. public function __construct($username, $passwd, $dbname, $file){
  6. $this->cmd = "mysqldump -u".$username." -p".$passwd." ".$dbname." > ".$file;
  7. }
  8. }
  9. ?>

Php代码  

    1. file:MySQLDumpTest.php
    2. url:http://localhost:88/pdo/MySQLDumpTest.php
    3. <?php
    4. require_once ‘AbstractMySQLDump.php‘;
    5. $username = "root";
    6. $passwd = "password";
    7. $dbname = "pdotest";
    8. $file = "d:/".$dbname.".sql";
    9. // 使用工厂方法生成备份类
    10. $dump = AbstractMySQLDump::factory($username, $passwd, $dbname, $file);
    11. // 执行备份类的backup方法
    12. $dump->backup();
    13. ?>

PHP开发笔记系列(一)-PDO使用,布布扣,bubuko.com

时间: 2024-10-18 05:51:45

PHP开发笔记系列(一)-PDO使用的相关文章

iOS开发笔记系列-基础3(多态、动态类型和动态绑定)

多态:相同的名称,不同的类 使不同的类共享相同方法名称的能力成为多态.它让你可以开发一组类,这组类中的每一个类都能响应相同的方法名.每个类的定义都封装了响应特定方法所需要的代码,这使得它独立于其他的类定义.这是因为Objective-C的运行时系统在执行方法时知道消息的接收者是哪个类的对象,它总是携带有关“一个对象属于哪个类”这样的信息,该信息能使系统在运行时做出决定,而不是在编译时. 动态绑定和id类型 id数据类型是一种通用的对象类型,可以用来存储属于任何类的对象.当使用id类型的时候,程序

iOS开发笔记系列-基础7(C语言特性)

Objective-C是C语言的扩展,因此,也具备很多C语言的基本特性,这里只罗列部分. 块(Blocks) 块是对C语言的一种扩展,它并未作为标准ANSI C所定义的部分,而是Apple添加到语言中的.它看起来很像函数,可以给它传递参数,它也具有返回值,与函数不同的是,块定义在函数或方法内部,并能够访问在函数或者方法范围内块之外的任何变量.一般来说,它可以访问到这些变量但是不能修改它们的值,有一个特殊的块修改器(由块前面含有两个下划线的字符组成)能够修改块内变量的值.块本身也能够作为参数传递给

【android】开发笔记系列:行为篇

1:键盘遮挡了输入框 在androidManifest.xml里,对应的activity里设置键盘模式 <activity android:name="活动名称" android:windowSoftInputMode="adjustPan"> 2:改变输入框回车键为“下一个” 前提:需要设置EditText为单行模式android:singleLine="true",否则回车键默认就是换行的. 在EditText的xml布局文件 a

iOS开发笔记系列-基础5(分类和协议)

分类 在Objective-C中,除了通过新建子类的方式来向类添加新方法外,还可以通过分类的方式.分类提供了一种简单的方式,将类的定义模块化到相关方法的组或分类中,它还提供了扩展现有类定义的简便方式,并且不需要访问类的源代码,也无须创建子类. 比如: //SomeClass+MathOps.h #import "SomeClass.h" @interface SomeClass (MathOPs) -(SomeClass *) add: (SomeClass *)s; -(SomeCl

【转】【系列】Openwrt 项目开发笔记

这个系列来自隔壁的坛友@Double_win,已自觉主动聚合,看起来很方便. “Openwrt 项目开发笔记”系列传送门: [Openwrt 项目开发笔记]:Openwrt平台搭建(一) (2014-07-11 00:11) [Openwrt 项目开发笔记]:Openwrt平台搭建(一)补遗 (2014-07-11 20:32) [Openwrt 项目开发笔记]:Openwrt必要设置(二) (2014-07-13 15:03) [Openwrt 项目开发笔记]:USB挂载& U盘启动(三) (

PHP微信公众开发笔记(一)

笔记是先写在印象笔记上的,晚上没事的时候整理整理. PHP微信公众开发笔记系列 日期:2014.9.1 今天开始正式开始做班上的微信公众平台的开发者服务了.因为是第一次做这个,对PHP也不了解,所以一边学PHP,一边整理微信的公众平台,当做是自己的练手学习好了. 在登录了微信的公众平台之后,左边功能区的最底部就是开发者中心了: 在没有开启开发者服务前,这里面啥也没有.启用这个服务后,在网站上编辑的自动回复功能便取消了,用户跟微信公众平台的交互信息便会转到开发者接入的功能.废话少说,开启开发者功能

PHP微信公众开发笔记(二)

PHP微信公众开发笔记系列 日期:2014.9.1 需求分析: 在成功的开启了微信开发者服务之后,我们便需要开始做响应用户操作的功能了.其实做微信公众平台,当初的目的我就是想做成一个服务平台,因为刚大学毕业,同学各奔东西.有的时候可能我们会到一个陌生的城市去,那时可能就会想啊,这个城市有班上的同学在么?这时掏出手机,点开服务号输入这个城市名,就会返回一串信息,假如有同学在这个城市,就会列出联系信息之类的.当然也许会说,既然是熟悉的同学,那么肯定会知道联系方式之类的,还需要这样多此一举么.而我的考

PHP微信公众开发笔记(三)

PHP微信公众开发笔记系列 日期:2014.9.2 今天主要的任务是昨天提到的那个处理缓存信息的问题,我需要保存一些消息用来做二次判断. 首先,记录一些PHP语法知识吧. 1.PHP中字符串的连接语法: 在lua中,两个字符串的连接很简单,加入有两个字符串 "aaa";"bbb"; 需要将这两个字符串连接起来,只需要用两个句点 .. 便可以了:"aaa" .. "bbb"; 这里要注意,字符串与句点直接需要有空格,没有空格会报

PHP微信公众开发笔记(五)

PHP微信公众开发笔记系列 日期:2014.9.3 今天做了身份验证的功能,然后完善了下搜索功能.其实主要的是将整个代码结构整理了一番,应该可以说是模块化设计吧. 模块化设计我们的公众号. 因为我们之前提的功能需求中有: 1.菜单--查询功能.我考虑到后期功能的扩展,就想将这些分模块来实现:菜单模块(这样,今后我们需要添加新的菜单功能,可以直接在这个模块里操作,这样修正和维护也简单,在考虑到后期可能会分工协作的时候各开发者之间不会产生冲突): 2.数据库模块(这里就主要是负责数据库相关的工作,如