我们使用传统的 mysql_connect 、mysql_query方法来连接查询数据库时,如果过滤不严就有SQL注入风险,导致网站被攻击。

虽然可以用mysql_real_escape_string()函数过滤用户提交的值,但是也有缺陷。

而使用PHP的PDO扩展的 prepare 方法,就可以避免sql injection 风险。

PDO(PHP Data Object) 是PHP5新加入的一个重大功能,因为在PHP 5以前的php4/php3都是一堆的数据库扩展来跟各个数据库的连接和处理.

如 php_mysql.dll。 PHP6中也将默认使用PDO的方式连接,mysql扩展将被作为辅助 。

官方:http://php.net/manual/en/book.pdo.php

1、使用PDO连接前需要先确认PDO扩展是否已经打开。

使用PDO扩展之前,先要启用这个扩展,PHP.ini中。

去掉"extension=php_pdo.dll"前面的";"号,若要连接数据库,还需要去掉与PDO相关的数据库扩展前面的";"号。

(一般用的是php_pdo_mysql.dll),然后重启Apache服务器即可。

extension=php_pdo.dll
extension=php_pdo_mysql.dll

2、PDO连接数据库:


class MysqlPdo
{
private $config = [
// 数据库类型
'type' => 'mysql',
// 服务器地址
'hostname' => '127.0.0.1',
// 数据库名
'database' => 'test',
// 用户名
'username' => 'root',
// 密码
'password' => 'root',
// 端口
'hostport' => '3306',
// 数据库编码默认采用utf8
'charset' => 'utf8',
];
private $pdo;

public function __construct()
{
$dsn = "{$this->config['type']}:host={$this->config['hostname']};";
$dsn.= "dbname={$this->config['database']};charset={$this->config['charset']}";
$pdo = new PDO($dsn, $this->config['username'], $this->config['password']);
$pdo->setAttribute(PDO::ATTR_CASE, PDO::CASE_LOWER);
$pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
$pdo->setAttribute(PDO::ATTR_EMULATE_PREPARES, false);
$this->pdo = $pdo;
unset($dsn);
}
public function select($sql = '')
{
$res = $this->pdo->query($sql);
$res = $res->fetchAll(PDO::FETCH_ASSOC);
return $res;
}
}

3、PDO设置属性:

PDO有三种错误处理方式:

1、PDO::ERrmODE_SILENT 不显示错误信息,只设置错误码

2、PDO::ERrmODE_WARNING 显示警告错

3、PDO::ERrmODE_EXCEPTION 抛出异常

1
$pdo->setAttribute(\PDO::ATTR_ERrmODE, \PDO::ERrmODE_EXCEPTION);

1) :当设置为PDO::ERrmODE_SILENT时可以通过调用errorCode() 或errorInfo()来获得错误信息,当然其他情况下也可以。

2) :因为不同数据库对返回的字段名称大小写处理不同,所以PDO提供了PDO::ATTR_CASE设置项(包括PDO::CASE_LOWER,PDO::CASE_NATURAL,PDO::CASE_UPPER),来确定返回的字段名称的大小写。

3) :通过设置PDO::ATTR_ORACLE_NULLS类型(包括PDO::NULL_NATURAL,PDO::NULL_EmpTY_STRING,PDO::NULL_TO_STRING)来指定数据库返回的NULL值在php中对应的数值。

 

4、PDO常用方法及其应用:

PDO::query() 主要是用于有记录结果返回的操作,特别是SELECT操作

PDO::exec() 主要是针对没有结果集合返回的操作,如INSERT、UPDATE等操作

PDO::prepare() 主要是预处理操作,需要通过$rs->execute()来执行预处理里面的SQL语句,这个方法可以绑定参数,功能比较强大(防止sql注入就靠这个)

PDO::lastInsertId() 返回上次插入操作,主键列类型是自增的最后的自增ID

PDOStatement::fetch() 是用来获取一条记录

PDOStatement::fetchAll() 是获取所有记录集到一个集合

PDOStatement::fetchColumn() 是获取结果指定第一条记录的某个字段,缺省是第一个字段

PDOStatement::rowCount() 主要是用于PDO::query()和PDO::prepare()进行DELETE、INSERT、UPDATE操作影响的结果集,对PDO::exec()方法和SELECT操作无效。

 

5、PDO操作实例:

 

【示例6,说明】,防sql注入操作:

1、使用PDO访问MySQL数据库时,真正的real prepared statements 默认情况下是不使用的。

为了解决这个问题,你必须禁用 prepared statements的仿真效果。

2、设置禁用prepared statements:$pdo->setAttribute(\PDO::ATTR_EMULATE_PREPARES, false);

它会告诉PDO禁用模拟预处理语句,并使用 real parepared statements 。

这可以确保SQL语句和相应的值在传递到mysql服务器之前是不会被PHP解析的(禁止了所有可能的恶意SQL注入攻击)。

虽然你可以配置文件中设置字符集的属性(charset=utf8),但是需要格外注意的是,老版本的 PHP( < 5.3.6)在DSN中是忽略字符参数的。

但是我们需要注意以下几种情况,PDO并不能帮助我们防范SQL注入的:

1、你不能让占位符 ? 代替一组值,如:

select * from xc_company where id in( ? );

2、你不能让占位符代替数据表名或列名,如:

select * from xc_company order by ?;

3、你不能让占位符 ? 代替任何其他SQL语法,如:

select EXTRACT( ? from date) as times from xc_company;

查看PDO都有那些方法

echo '<pre>';
print_r(get_class_methods('PDO'));
echo '</pre>';

结果:

Array
(
[0] => __construct
[1] => prepare
[2] => beginTransaction
[3] => commit
[4] => rollBack
[5] => inTransaction
[6] => setAttribute
[7] => exec
[8] => query
[9] => lastInsertId
[10] => errorCode
[11] => errorInfo
[12] => getAttribute
[13] => quote
[14] => __wakeup
[15] => __sleep
[16] => getAvailableDrivers
)

常用实例源码:

<?php

# 【PDO连接mysql数据库】(数据库名:lmgg ,账户:root,密码:root)
$pdo = new \PDO('mysql:host=localhost;dbname=lmgg;charset=utf8','root','root');
# 设置数据库编码为utf-8(防止乱码)当然上面的连接设置了。
# 上面可设置也可不设置,有没有charset=utf8都可以连接上数据库
$pdo->exec('set names utf8'); # 【示例1:查询数据】:查询company表中的id和name字段。存到$data数组中去
$res = $pdo->query("select id,name from xc_company");
$data = []; # 1:FETCH_ASSOC 关联数组形式返回
# 2:FETCH_NUM 数字索引数组形式返回
# 设置返回数据类型方法1:
$res->setFetchMode(\PDO::FETCH_NUM);
while($row = $res->fetch()){
$data[] = $row;
}
echo '<pre>';
print_r($data);
echo '</pre>'; # 设置返回数据类型方法2:
while($row = $res->fetch(\PDO::FETCH_ASSOC)){
$data[] = $row;
}
echo '<pre>';
print_r($data);
echo '</pre>'; # 【示例2:添加数据】:添加数据到company表中,并返回数据在表中的ID是多少!
$res = $pdo->exec("insert into xc_company(name) values('小川编程添加111')");
if($res){
echo '11添加成功数据ID为:'.$pdo->lastinsertid().'<br/>';
}
$res = $pdo->query("insert into xc_company(name) values('小川编程添加222')");
if($res){
echo '22添加成功数据ID为:'.$pdo->lastinsertid().'<br/>';
} # 【示例3:更新数据】:
$res = $pdo->exec("update xc_company set name='小川编程更新111' where id=26");
if($res){
echo '更新数据成功<br/>';
}
$res = $pdo->query("update xc_company set name='小川编程更新222' where id=26");
if($res){
echo '成功更新数据【'.$res->rowCount().'】条<br/>';
} # 【示例4:删除数据】:
$res = $pdo->exec("delete from xc_company where id=38");
if($res){
echo '删除数据成功<br/>';
}
$res = $pdo->query("delete from xc_company where id=38");
if($res){
echo '成功删除数据【'.$res->rowCount().'】条<br/>';
} # 【示例5:统计数据】:统计company表有多少条数据
$num = $pdo->query("select count(*) from xc_company");
echo '共有数据:【'.$num->fetchColumn().'】条'; # 【示例6:防sql注入】:实例,使用前先看文字说明,会理解的更透彻!
# PDO连接mysql数据库(数据库名:lmgg ,账户:root,密码:root)
$pdo = new \PDO("mysql:host=localhost; dbname=lmgg", "root", "root");
# 禁用prepared statements的仿真效果
$pdo->setAttribute(\PDO::ATTR_EMULATE_PREPARES, false);
$pdo->query("set names 'utf8'");
$sql="select * from xc_company where name = ?";
$res = $pdo->prepare($sql);
$name = '小川编程';
$exeres = $res->execute(array($name));
$data = [];
if($exeres){
while ($row = $res->fetch(\PDO::FETCH_ASSOC)){
$data[] = $row;
}
}
echo '<pre>';
print_r($data);
echo '</pre>';
$pdo = null; ?>
$dsn= "{$this->config['type']}:host={$this->config['hostname']};";
$dsn.= "dbname={$this->config['database']};charset={$this->config['charset']}";
$pdo = new PDO($dsn, $this->config['username'], $this->config['password']);

如果有帮到您,就给小编打个赏吧,谢谢哦!



php之PDO连接mysql数据库,增删改查等等操作实例的更多相关文章

  1. Java连接MySQL数据库增删改查通用方法

    版权声明:本文为博主原创文章,未经博主允许不得转载. Java连接MySQL数据库增删改查通用方法 运行环境:eclipse+MySQL 以前我们Java连接MySQL数据库都是一个数据库写一个类,类 ...

  2. python操作mysql数据库增删改查的dbutils实例

    python操作mysql数据库增删改查的dbutils实例 # 数据库配置文件 # cat gconf.py #encoding=utf-8 import json # json里面的字典不能用单引 ...

  3. jsp-2 简单的servlet连接mysql数据库 增删改查

    连接mysql数据库的操作 有增删改查 用的包有 commons-lang3-3.5 mysql-connector-java-5.1.40-bin 但是实际上也就是 数据查询和数据处理两种 所以对数 ...

  4. java连接mysql数据库增删改查操作记录

    1. 连接数据库.得到数据库连接变量 注意连接数据库的时候 (1)打开DB Browser 新建一个Database Driver,注意加入Driver JARs的时候加入的包,我的是mysql-co ...

  5. Asp.Net操作MySql数据库增删改查

    Asp.Net操作MySql数据库增删改查,话不多说直接步入正题.git源码地址:https://git.oschina.net/gxiaopan/NetMySql.git  1.安装MySQL数据库 ...

  6. MySQL数据库(增删改查语句)

    MySQL数据库(增删改查语句)一.登录数据库:---->  mysql -uroot -proot;(对应用户名和密码)二.SQL语句:    数据定义语言DDL  用来定义数据库.表.列,关 ...

  7. python2.7入门---操作mysql数据库增删改查

    Python 标准数据库接口为 Python DB-API,Python DB-API为开发人员提供了数据库应用编程接口.Python 数据库接口支持非常多的数据库,你可以选择适合你项目的数据库: G ...

  8. Python实现mysql数据库增删改查

    利用python操作mysql数据库用法简单,环境配置容易,本文将实现对库增.删.改.查的简易封装!   1. 环境配置 安装第三方包  ,导入模块 mysql.connector  pip inst ...

  9. java连接mysql以及增删改查操作

    java连接数据库的代码基本是固定的,步骤过程觉得繁琐些,代码记起来对我来说是闹挺.直接上代码: (温馨提醒:你的项目提前导入连接数据库的jar包才有的以下操作 ) class DBConnectio ...

随机推荐

  1. 通过BulkLoad快速将海量数据导入到Hbase(TDH,kerberos认证)

    一.概念 使用BlukLoad方式利用Hbase的数据信息是 按照特点格式存储在HDFS里的特性,直接在HDFS中生成持久化的Hfile数据格式文件,然后完成巨量数据快速入库的操作,配合MapRedu ...

  2. Linux 设置静态IP

    由于工作需要,安装一套Linux系统.安装完成后发现这个家伙居然不能上网,然后看了下IP,(命令 ip a)发现是127.0.0.1 下面是我的界面: inet 是127.0.0.1/8 还有6个网卡 ...

  3. Solon rpc 之 SocketD 协议

    1. 简介 SocketD 是一种二进制的点对点通信协议,是一种新的网络通信第七层协议.旨在用于分布式应用程序中.从这个意义上讲,SocketD可以是RSocket等其他类似协议的替代方案.它的消息协 ...

  4. C# 设置默认关联程序

    以下代码做个Mark /// <summary> /// Create an associaten for a file extension in the windows registry ...

  5. 【Java】流程控制 - 顺序结构、 选择(分支)结构(单分支、双分支、多分支、嵌套)、循环结构(for、while、do...while)、跳转语句(break、continue)

    流程控制语句结构 文章目录 流程控制语句结构 一. 顺序结构 1. 输出语句 2. 输入语句 3.code 二.复合语句 三. 分支结构 1. 条件判断 1.单分支结构 2.双分支结构 3.多分支结构 ...

  6. 【SpringBoot】Spring Boot,开发社区讨论交流网站首页。

    初识Spring Boot,开发社区讨论交流网站首页. 文章目录 初识Spring Boot,开发社区讨论交流网站首页. 1.项目简介 2. 搭建开发环境 JDK Apache Maven Intel ...

  7. [CPP] 智能指针

    介绍 C++ 的智能指针 (Smart Pointers) 相关 API. C++ 中的智能指针是为了解决内存泄漏.重复释放等问题而提出的,它基于 RAII (Resource Acquisition ...

  8. 到底什么是哈希Hash?

    有次面试被问到这个问题? 我说是经过运算的一串字符串,这个回答显然是让人不满意,连自己都不满意! 但是又对其很模糊,那么到底什么是Hash呢? 定义 Hash一般翻译为散列,还有音译为哈希,本文我们统 ...

  9. Java中的NIO进阶

    目录 前言 NIO与多线程 Readable和Writeable的空触发 请求与返回的处理 事件的处理机制 NIO多线程使用的一个例子 前言 之前一篇文章简单介绍了NIO,并附了一个简单的例子,但是自 ...

  10. 一文读懂k8s之Pod安全策略

    导读 Pod容器想要获取集群的资源信息,需要配置角色和ServiceAccount进行授权.为了更精细地控制Pod对资源的使用方式,Kubernetes从1.4版本开始引入了PodSecurityPo ...