最近项目需要用到批量更新数据库里的数据,在网上找了一下这方面的例子,觉得这个还不错,分享给大家。

在这个业务里里面涉及到了更新两张数据表,那么大家是不是会想到非常简单,马上上代码

$sql = "update newhouse_clicks set clicks=6,type=1,update_time=time() where is=$value['id']";

其中数据表名为newhouse_clicks,有四个字段,主键id,type(类型-整型)字段,clicks(点击量-整型)字段,update_time(整型)字段
这样做确实没有问题吗?比如说批量更新当前城市下的所有楼盘,比如说北京,打个比方1000条数据,那么在业务里面是不是就要这样写

$data = array(id=>1,id=>2,..........id=>1000);//省略数据
foreach($data as $key=>$value) {
$sql = "update newhouse_clicks set clicks=6,type=1,update_time=time() where is=$value['id']";
}

这样在涉及到多张表*1000条数据,那么会不会有很大的延时呢?

结果是的,她这样写确实是导致了服务器的超时!

大家如果熟悉JAVA等语言应该知道,JAVA会内部提供了批量更新sql的功能,那么作为世界上做好的语言PHP能不能做到呢?答案是可以滴!

那么我们学习一下批量更新的sql语句。

UPDATE newhouse_clicks
SET clicks = CASE id
WHEN 1 THEN 1000
WHEN 2 THEN 2000
WHEN 3 THEN 3000
END
WHERE id IN (1,2,3)

稍安勿躁,咱们详解一下这条sql语句的意思:
更新newhouse_clicks数据表中的clicks字段,当id=1是设置值为1000,当id=2时设置值为2000,当id=3时设置值为3000


那么更新多个字段能不能做到呢?当然可以,贴代码:

UPDATE newhouse_clicks
SET clicks = CASE id
WHEN 1 THEN 1000
WHEN 2 THEN 2000
WHEN 3 THEN 3000
END,
type = CASE id
WHEN 1 THEN 1
WHEN 2 THEN 6
WHEN 3 THEN 8
END
WHERE id IN (1,2,3)

这条sql语句的含义就是更新newhouse_clicks数据表中的clicks字段,当id=1是设置值为1000,当id=2时设置值为2000,当id=3时设置值为3000,更新type字段,当id=1时更新为type字段为1,当id=2时更新type字段为6,当id=3时,更新type字段为8。
那么,世界上最好的语言PHP不就可以拼出来sql了吗?

//查询数据库返回的数据格式
$newhouse_clicks = array(
      1 => 2,
      2 => 3,
      3 => 8,
      4 => 9,
  );
$ids = implode(',', array_keys($newhouse_clicks ));
$sql = "UPDATE newhouse_clicks SET clicks = CASE id ";
foreach ( $newhouse_clicks as $key => $value) {
$sql .= sprintf("WHEN %d THEN %d ", $key, $value);
}
$sql .= "END WHERE id IN ($ids)";
echo $sql;

大家查看一下是不是和咱们上面的sql语句是一样的呀!

那么,咱们真是的数据是不是要比这复杂呀?肯定的,看题,咱们通常在数据库里面取出来的数据格式是不是都是这样的。

//查询数据库返回的数据格式
$newhouse_clicks = array(
1 => array('clicks'=>1,'type'=>1,'update_time'=>time()),
2 => array('clicks'=>2,'type'=>2,'update_time'=>time()),
3 => array('clicks'=>3,'type'=>3,'update_time'=>time()),
4 => array('clicks'=>4,'type'=>4,'update_time'=>time()),
);
?>

那么,这种情况怎么办?

   
<?php
//查询数据库返回的数据格式
$newhouse_clicks = array(
1 => array('clicks'=>1,'type'=>1,'update_time'=>time()),
2 => array('clicks'=>2,'type'=>2,'update_time'=>time()),
3 => array('clicks'=>3,'type'=>3,'update_time'=>time()),
4 => array('clicks'=>4,'type'=>4,'update_time'=>time()),
);
//获取所有的id
$newhouse_clicks_keys = array_keys($newhouse_clicks);
//拼接批量更新sql语句
$sql = "UPDATE newhouse_clicks SET ";
//合成sql语句
foreach ($newhouse_clicks[1] as $key => $value) {
$sql .= "{$key} = CASE id ";
foreach ($newhouse_clicks as $newhouse_clicks_key=>$newhouse_clicks_value) { $sql .= sprintf("WHEN %d THEN %d ", $newhouse_clicks_key, $newhouse_clicks_value[$key]);
}
$sql .= "END, ";
}
//把最后一个,去掉
$sql = substr($sql, 0, strrpos($sql,','));
//合并所有id
$ids = implode(',', $newhouse_clicks_keys);
//拼接sql
$sql .= " WHERE ID IN ({$ids})";
echo $sql;

其实,写了这么一大堆,是不是就是为了拼装成mysql语句呀。

大功告成!速度是不是像丝般顺滑!

老多程序员特别是初学者,很容易进入一个误区,把在sql中取数据套在for循环里面。这样写导致一个问题,就是严重的阻塞,现实生活中就有这样一个例子:

比如说,你在12层办公,这是快递员给你打电话让你去楼下取快递(总共12件),你去取快递有两种方式:

1.拿到第一件快递,跑回12层,放好后,接着去领取下一件快递,放好12层后再接着去取下一件快递。

2.一次性把所有的快递都拿到12层。

大家一定都会选第二个方案吧,没人会傻不拉几的去跑12次才能拿完快递的。

计算机就是上述原理,在for循环里面去资源取数据,就是类似第一种方案。批量取数据就是类似于第二种方案。(PS一下:不要以为在mysql中取数据有这种问题,redis也会

有这种问题,要不然怎么会有pipeline取批量数据呢,这一点在面试的时候我是经常当面试题去问的,绝对会成为衡量一个人技术水平的标准)

转自 https://blog.csdn.net/wuming19900801/article/details/62893429

PHP如何批量更新MYSQL中的数据的更多相关文章

  1. PHP批量更新MYSQL中的数据

    原文链接:https://blog.csdn.net/wuming19900801/article/details/62893429 $sql = "update newhouse_clic ...

  2. 使用DataAdpater自动批量更新DataSet中的数据到数据库

    使用DataAdpater的批量更新,好处是不用写具体的sql脚本,尤其是做web服务这个优势非常吸引人. [WebMethod(Description = "Update服务提供的方法,将 ...

  3. 【Redis 向Redis中批量导入mysql中的数据(亲自测试)】

    转自:https://blog.csdn.net/kenianni/article/details/84910638 有改动,仅供个人学习 问题提出:缓存的冷启动问题 应用系统新版本上线,这时候 re ...

  4. 批量更新mysql表数据

    1.批量更新表中某个字段,如果表比较大,每条记录都执行一条update,1s执行10条数据,10万条数据就要1W秒,3个多小时. 2.可以用case when 来实现,模板如下 UPDATE cate ...

  5. SQL批量更新数据库中所有用户数据表中字段类型为tinyint为int

    --SQL批量更新数据库中所有用户数据表中字段类型为tinyint为int --关键说明:--1.从系统表syscolumns中的查询所有xtype='48'的记录得到类型为[tinyint]的字段- ...

  6. 用python批量向数据库(MySQL)中导入数据

    用python批量向数据库(MySQL)中导入数据 现有数十万条数据,如下的经过打乱处理过的数据进行导入 数据库内部的表格的数据格式如下与下面的表格结构相同 Current database: pyt ...

  7. 多线程查询数据,将结果存入到redis中,最后批量从redis中取数据批量插入数据库中【我】

    多线程查询数据,将结果存入到redis中,最后批量从redis中取数据批量插入数据库中 package com.xxx.xx.reve.service; import java.util.ArrayL ...

  8. 利用Ajax实现数据的同步传输,从mysql中提取数据,通过echarts可视化

    如何将mysql数据库中的方式通过echarts可视化呢,以下面这个简单的例子向大家进行演示:   步骤一:mysql的创表和插入数据,当然这些数据也可以是你通过爬虫抓取的.     步骤二:   创 ...

  9. Vusual C++连接Mysql和从MySql中取出数据的API介绍

    .1 mysql_real_connect() 2.1.1 函数原型: MYSQL *mysql_real_connect(MYSQL *mysql, const char *host, const ...

随机推荐

  1. 开发创建XMPP“发布订阅”扩展(xmpp pubsub extend)

    发布订阅(PubSub)是一个功能强大的XMPP协议扩展.用户订阅一个项目(在xmpp中叫做node),得到通知时,也即当事项节点更新时.xmpp服务器通知用户(通过message格式). 节点类型: ...

  2. app绘制手势密码 、九宫格分解

    什么是九宫格? 即是我们常见的手势绘制.一共有9个点,让我们进行绘制手势.我们手动操作的时候,通过是按住第一个点,然后移动到最后一点,然后松开,就完成手势的操作,那么,如果要用自动化代码,来让其自动绘 ...

  3. js 开发过程中经验及总结记录

    一   let 和 var 作用域    1  普通用法 for (var i = 0; i < 5; i++) { console.log(i); } console.log(i); //-- ...

  4. 使用Docker发布应用

    新建spring boot应用demo-docker,添加web依赖 <dependency> <groupId>org.springframework.boot</gr ...

  5. 磁盘分区(20G升50G)

      不多说,直接上干货! 本博文的主要内容有 .磁盘分区的概述 .常用的磁盘管理工具    ./下分5G,给/home扩容    .系统自带的fdisk和parted这两款工具 .磁盘空间管理 前言 ...

  6. Java之建造者模式(Builder Pattern)(转)

    1.概念 将一个复杂的构建与其表示相分离,使得同样的构建过程可以创建不同的表示. [构建与表示分离,同构建不同表示] 与抽象工厂的区别:在建造者模式里,有个指导者,由指导者来管理建造者,用户是与指导者 ...

  7. 装饰者模式——Java设计模式

    装饰模式 1.概念 动态地为对象附加上额外的职责 其目的是包装一个对象,从而可以在运行时动态添加新的职责.每个装饰器都可以包装另一个装饰器,这样理论上来说可以对目标对象进行无限次的装饰. 2.装饰器类 ...

  8. php获取 POST请求的数据

    普通键值对的数据: $_POST[‘username’]; // 获取 username的信息: $_REQUEST; //则会获取 整个请求中的键值对,返回结果为数组: 如果是,流数据,则需要使用: ...

  9. Java 使用pipeline对redis进行批量读写

    code import redis.clients.jedis.Jedis; import redis.clients.jedis.Pipeline; import java.util.List; p ...

  10. Spring事务内方法调用自身事务 增强的三种方式

    ServiceA.java文件: 查看Spring Tx的相关日志:  可以看到只创建了一个事物ServiceA.service方法的事务,但是callSelf方法却没有被事务增强; 分析原因:Spr ...