0

0

封装了一个PDO的类,希望CSDN的大牛们指点一下不足之处

php中文网

php中文网

发布时间:2016-06-23 13:26:06

|

1186人浏览过

|

来源于php中文网

原创

封装了一个PDO的类,希望CSDN的大牛们指点一下不足之处。

大师们,come on !

<?php/** * PDO封装类,目的是为了使用起来更简单方便 * modify Date: 2014-07-01 */class PDOX {	private $pdo 	   = null;		public  $statement = null;		public  $options = array(		PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES ",	);		public function __construct($dsn, $user = '', $pass = '', $persistent = false, $charset = "utf8"){		$this->options[PDO::MYSQL_ATTR_INIT_COMMAND] .= $charset;		if($persistent){			$this->options[PDO::ATTR_PERSISTENT] = true;		}		$this->pdo = new PDO($dsn, $user, $pass, $this->options);	}	/**	全局属性设置,包括:列名格式和错误提示类型    可以使用数字也能直接使用参数	*/	public function setAttr($param, $val = ''){		if(is_array($param)){			foreach($param as $key => $val){				$this->pdo->setAttribute($key, $val);			}		}else{			if($val != '' ){				$this->pdo->setAttribute($param, $val);			}else{				return false;			}		}	}	/**	生成一个编译好的sql语句模版 你可以使用 ? :name 的形式	返回一个statement对象	*/	public function prepare($sql){		if(empty($sql)){			return false;		}		$this->statement = $this->pdo->prepare($sql);		return $this->statement;	}	/**	执行Sql语句,一般用于 增、删、更新或者设置  返回影响的行数	*/	public function exec($sql){		if(empty($sql)){			return false;		}		try{			return $this->pdo->exec($sql);		}catch(Exception $e){			return $e->getMessage();		}	}	/**	执行有返回值的查询,返回PDOStatement  可以通过链式操作,可以通过这个类封装的操作获取数据	*/	public function query($sql){		if(empty($sql)){			return false;		}		$this->statement = $this->pdo->query($sql);		return $this->statement;	}	/**	开启事务	*/	public function beginTransaction(){		return $this->pdo->beginTransaction();	}	/**	提交事务	*/	public function commit(){		return $this->pdo->commit();	}	/**	事务回滚	*/	public function rollBack(){		return $this->pdo->rollBack();	}		public function lastInertId(){		return $this->pdo->lastInsertId();	}			//**   PDOStatement 类操作封装    **//		/**	让模版执行SQL语句,1、执行编译好的 2、在执行时编译	*/	public function execute($param = ""){		if(is_array($param)){			try{				return $this->statement->execute($param);			}catch (Exception $e){				//return $this->errorInfo();				return $e->getMessage();			}		}else{			try{				return $this->statement->execute();			}catch(Exception $e){				/* 返回的错误信息格式				[0] => 42S22    			[1] => 1054    			[2] => Unknown column 'col' in 'field list'    			return $this->errorInfo();    			*/				return $e->getMessage();			}		}	}		/**	参数1说明:	PDO::FETCH_BOTH		也是默认的,两者都有(索引,关联)	PDO::FETCH_ASSOC	关联数组	PDO::FETCH_NUM		索引	PDO::FETCH_OBJ			对象	PDO::FETCH_LAZY		对象 会附带queryString查询SQL语句	PDO::FETCH_BOUND	如果设置了bindColumn,则使用该参数	*/	public function fetch($fetch_type = PDO::FETCH_ASSOC){		if(is_object($this->statement)){			return $this->statement->fetch($fetch_type);		}		return false;	}	/**	参数1说明:	PDO::FETCH_BOTH		也是默认的,两者都有(索引,关联)	PDO::FETCH_ASSOC	关联数组	PDO::FETCH_NUM		索引	PDO::FETCH_OBJ		对象	PDO::FETCH_COLUMN	指定列 参数2可以指定要获取的列	PDO::FETCH_CLASS	指定自己定义的类	PDO::FETCH_FUNC		自定义类 处理返回的数据	PDO_FETCH_BOUND	如果你需要设置bindColumn,则使用该参数	参数2说明:	给定要处理这个结果的类或函数	*/	public function fetchAll($fetch_type = PDO::FETCH_ASSOC, $handle = ''){		if(empty($handle)){			return $this->statement->fetchAll($fetch_type);		}		return $this->statement->fetchAll($fetch_type, $handle);	}	/**	以对象形式返回 结果 跟fetch(PDO::FETCH_OBJ)一样	*/	public function fetchObject($class_name){		if(empty($clss_name)){			return $this->statement->fetchObject();		}		return $this->statement->fetchObject($class_name);	}		public function fetchColumn($intColumn = 0){		return $this->statement->fetchColumn($intColumn);	}		/**	public function bindColumn($array=array(),$type=EXTR_OVERWRITE){		if(count($array)>0){			extract($array,$type);		}		//$this->statement->bindColumn()	}	*/		/**	以引用的方式绑定变量到占位符(可以只执行一次prepare,执行多次bindParam达到重复使用的效果)	*/	public function bindParam($parameter, $variable, $data_type = 'STR', $length = 0){		switch ($data_type){			case  'STR':				$data_type = PDO::PARAM_STR;				break;			case 'INT':				$data_type = PDO::PARAM_INT;				break;			default :				$data_type = '';				break;		}		return $this->statement->bindParam($parameter, $variable, $data_type, $length);	}		/**	返回statement记录集的行数	*/	public function rowCount(){		return $this->statement->rowCount();	}	public function count(){		return $this->statement->rowCount();	}	public function columnCount(){		$this->statement->execute();		return $this->statement->columnCount();	}	public function getColumnMeta($intColumn){		return $this->statement->getColumnMeta($intColumn);	}			/**	关闭	*/	public function close(){		return $this->statement->closeCursor();	}		public function closeCursor(){		return $this->statement->closeCursor();	}	/**	返回错误信息也包括错误号	*/	private function errorInfo(){		return $this->statement->errorInfo();	}	/**	返回错误号	*/	private function errorCode(){		return $this->statement->errorCode();	}				//简化操作	public function insert($table, $data){		if(!is_array($data)){			return false;		}		$cols = array();		$vals = array();		foreach($data as $key => $val){			$cols[] = $key;			$vals[] = "'" . $val . "'";		}		$sql  = "INSERT INTO {$table} (";		$sql .= implode(",", $cols) . ") VALUES (";				$sql .= implode(",", $vals) . ")";		return $this->exec($sql);	}	public function insertBind($table, $arrayData){		if(!is_array($arrayData)){			return false;		}		$vals = array_keys($arrayData);		$cols = array();		/*		$arrayobject  = new  ArrayObject( $arrayData );		$iterator  	  =  $arrayobject->getIterator();		while($iterator->valid()) {			$vals[] =  ':' . $iterator->key() . '';			$iterator->next();		}		*/		$c	  = implode('', $vals);		$cols = array_filter(explode(':', $c));		$sql  = "INSERT INTO {$table} (";		$sql .= implode(",", $cols) . ") VALUES (";				$sql .= implode(",", $vals) . ")";				$this->statement = $this->pdo->prepare($sql);		$this->statement->execute($arrayData);		return  $this->statement->rowCount();	}		public function update($table, $data, $where){		if(!is_array($data)){			return false;		}		$set = array();		foreach($data as $key => $val){			$set[] = $key . "='" . $val  . "'";		}		$sql = "UPDATE {$table} SET ";		$sql .= implode(",", $set);		$sql .= " WHERE " . $where;		return $this->exec($sql);	}	public function updateBind($sql, $arrayWhere){		if(empty($sql) || !is_array($arrayWhere)){			return false;		}		$this->statement = $this->pdo->prepare($sql);		$this->statement->execute($arrayWhere);		return $this->statement->rowCount();	}		public function delete($table, $where){		if(empty($table) || empty($where)){			return false;		} 		$sql = "DELETE FROM {$table} WHERE " . $where;		return $this->exec($sql);	}	public function deleteBind($sql, $arrayWhere){		if(empty($sql) || !is_array($arrayWhere)){			return false;		}		$this->statement = $this->pdo->prepare($sql);		$this->statement->execute($arrayWhere);		return $this->statement->rowCount();	}}?>




毕竟在一个项目当中,不可能到处的写pdo自身的一些东西。也不要说pdo已经封装好了,无须再次封装之类的废话。


回复讨论(解决方案)

几点建议:
1、如果我
$db = new PDOX;
$a = $db->query($sql1);
$b = $db->query($sql2);
print_r($a->fetchall());
print_r($b->fetchall());
可以吗?显然是不可以的,因为
$this->statement = $this->pdo->query($sql);
前一次的查询结果被后一次的覆盖了
2、事务(Transaction)操作宜封装成整体,只传入一组 sql 而隐藏相关操作
3、没考虑使用存储过程
4、PDOX 可直接继承于 PDO 而无需抄写 PDO 已有方法


总的来说还不错,但感觉上有些方法过于细琐,可以有必要的整合在一起

几点建议:
1、如果我
$db = new PDOX;
$a = $db->query($sql1);
$b = $db->query($sql2);
print_r($a->fetchall());
print_r($b->fetchall());
可以吗?显然是不可以的,因为
$this->statement = $this->pdo->query($sql);
前一次的查询结果被后一次的覆盖了
2、事务(Transaction)操作宜封装成整体,只传入一组 sql 而隐藏相关操作
3、没考虑使用存储过程
4、PDOX 可直接继承于 PDO 而无需抄写 PDO 已有方法


谢谢你的宝贵意见,我仅仅是想让它操作更简单化一点。
1、像你举的这个例子,可有解决方法。
2、事务我很少用到,也不知道该怎样封装。如果你能指点一下更好。
3、存储过程也很少用到。
4、抄写PDO的方法,只是想在使用上和PDO无异样,方便直接写PDO的朋友直接使用此类。无须过多的学习。不像其它的数据库类,在学完原生的数据库操作后,又要再次学习一个新的数据库操作,可能它很简单,但学习也是要时间的。



总的来说还不错,但感觉上有些方法过于细琐,可以有必要的整合在一起


请问怎样整合会更简洁此呢??

封装的目的是为了简化操作,不常用的就没有不要封装了

class PDOX extends PDO {  //执行各种 sql 指令,并可通过参数 $param 进行扩展  function query($sql, $param=null) {}  //查询并返回单条记录  function fetch($sql) {};  //查询并以数组方式返回多条记录  function fetchall($sql) {}}

数据库类只要有这三个方法就可满足绝大多数应用的需要了
由于是继承于 PDO,所以 PDO 原有的方法一个也不会少

@xuzuning

$db = new PDOX;
$a = $db->query($sql1);
$b = $db->query($sql2);
print_r($a->fetchall());
print_r($b->fetchall());

这种写法的程序员应该不多吧?如果这样的话,可以考虑先放弃这种操作支持。如果我的类在效率或者并发上有问题的话,这才是大问题。

这个类内部是否有错误或者不妥之处??

各位大牛,请多多发言。

嵌套甚至递归的查询在读取分类树时是经常用到的,怎么能说不多呢?
嗯,做个留言板是不需要的

简单的插入数据:
$data = array(':title'=>$title, ':content'=>$content);
$pdo  = new PDOX;
$lastID = $pdo->insertBind('article', $data);


修改数据:
$data = array(':title'=>$title, ':id'=>$id);
$sql   = 'update article set title=:title where id=:id';
$yesorno = $pdo->updateBind($sql, $data);

删除数据:
$data = array(':id'=>$id);
$sql   = 'delete article  where id=:id';
$yesorno = $pdo->deleteBind($sql, $data);

最主要的是这种写法,可以省下很多的代码量。这才是我写此类的关键。

嵌套甚至递归的查询在读取分类树时是经常用到的,怎么能说不多呢?
嗯,做个留言板是不需要的



你说的这个还真是,在递归中需要使用。想使用此类的朋友要注意了。

我会尽量优化PHP代码或者使用其它方法替代递归,因为递归有些耗资源。

谢谢你提醒。

千帆大模型平台
千帆大模型平台

面向企业开发者的一站式大模型开发及服务运行平台

下载

表示看的挺纠结的

表示看的挺纠结的



杂纠结了???

我觉得在操作数据时挺方便的呀

大牛们,继续呀。。

封装的目的是为了简化操作,不常用的就没有不要封装了

class PDOX extends PDO {  //执行各种 sql 指令,并可通过参数 $param 进行扩展  function query($sql, $param=null) {}  //查询并返回单条记录  function fetch($sql) {};  //查询并以数组方式返回多条记录  function fetchall($sql) {}}

数据库类只要有这三个方法就可满足绝大多数应用的需要了
由于是继承于 PDO,所以 PDO 原有的方法一个也不会少



不知道版主可有更好的PDO操作类???
可否贡献一个。

不怎么样!
另外预处理部分没有封装

	/**	 * 将数组转换为SQL语句	 * @param array $where 要生成的数组	 * @param string $font 连接串。	 */	final public function sqls($where, $font = ' AND ') {				# 先将bindColumn存起来				$this->bindColumn = array();		if (is_array($where)) {			$sql = '';			foreach ($where as $key=>$val) {				$sql .= $sql ? " $font `$key` = :$key " : " `$key` = :$key";				$this->bindColumn = array($key, $val);			}			return ;		} else {			return $where;		}	}

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

705

2026.02.13

微博网页版主页入口与登录指南_官方网页端快速访问方法
微博网页版主页入口与登录指南_官方网页端快速访问方法

本专题系统整理微博网页版官方入口及网页端登录方式,涵盖首页直达地址、账号登录流程与常见访问问题说明,帮助用户快速找到微博官网主页,实现便捷、安全的网页端登录与内容浏览体验。

233

2026.02.13

Flutter跨平台开发与状态管理实战
Flutter跨平台开发与状态管理实战

本专题围绕Flutter框架展开,系统讲解跨平台UI构建原理与状态管理方案。内容涵盖Widget生命周期、路由管理、Provider与Bloc状态管理模式、网络请求封装及性能优化技巧。通过实战项目演示,帮助开发者构建流畅、可维护的跨平台移动应用。

117

2026.02.13

TypeScript工程化开发与Vite构建优化实践
TypeScript工程化开发与Vite构建优化实践

本专题面向前端开发者,深入讲解 TypeScript 类型系统与大型项目结构设计方法,并结合 Vite 构建工具优化前端工程化流程。内容包括模块化设计、类型声明管理、代码分割、热更新原理以及构建性能调优。通过完整项目示例,帮助开发者提升代码可维护性与开发效率。

22

2026.02.13

Redis高可用架构与分布式缓存实战
Redis高可用架构与分布式缓存实战

本专题围绕 Redis 在高并发系统中的应用展开,系统讲解主从复制、哨兵机制、Cluster 集群模式及数据分片原理。内容涵盖缓存穿透与雪崩解决方案、分布式锁实现、热点数据优化及持久化策略。通过真实业务场景演示,帮助开发者构建高可用、可扩展的分布式缓存系统。

61

2026.02.13

c语言 数据类型
c语言 数据类型

本专题整合了c语言数据类型相关内容,阅读专题下面的文章了解更多详细内容。

30

2026.02.12

雨课堂网页版登录入口与使用指南_官方在线教学平台访问方法
雨课堂网页版登录入口与使用指南_官方在线教学平台访问方法

本专题系统整理雨课堂网页版官方入口及在线登录方式,涵盖账号登录流程、官方直连入口及平台访问方法说明,帮助师生用户快速进入雨课堂在线教学平台,实现便捷、高效的课程学习与教学管理体验。

15

2026.02.12

豆包AI网页版入口与智能创作指南_官方在线写作与图片生成使用方法
豆包AI网页版入口与智能创作指南_官方在线写作与图片生成使用方法

本专题汇总豆包AI官方网页版入口及在线使用方式,涵盖智能写作工具、图片生成体验入口和官网登录方法,帮助用户快速直达豆包AI平台,高效完成文本创作与AI生图任务,实现便捷智能创作体验。

669

2026.02.12

PostgreSQL性能优化与索引调优实战
PostgreSQL性能优化与索引调优实战

本专题面向后端开发与数据库工程师,深入讲解 PostgreSQL 查询优化原理与索引机制。内容包括执行计划分析、常见索引类型对比、慢查询优化策略、事务隔离级别以及高并发场景下的性能调优技巧。通过实战案例解析,帮助开发者提升数据库响应速度与系统稳定性。

58

2026.02.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
TypeScript 教程
TypeScript 教程

共19课时 | 3万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号