使用PHP从Mysql数据库中导出大数据量的Excel或csv文件

<?php

/**

* <方法描述>导出excel或csv文件

* 需求描述:在项目开发中遇到了需要导出大数据量excel的问题,发现用框架在现有的基础上无法实现,特此总结

* 使用该方法我已成功导出100W的数据分别到excel和csv文件中,

* 下载文件大概1分钟左右,读取文件,excel时间长点大概需要4、5分钟才能打开,在2003版本中打开只能显示到65536行

* 打开csv文件则快多了,而且不区分2003和2007,都可以显示100W的数据

* excel 2003版本最多允许65536条数据

* 2007版本以上允许1048576条数据

* 采用数据库直连方式,否则可能会内存溢出,无法导出excel,由于PHP版本不同可能会报mysql连接函数已废弃,建议使用PDO

* 我使用的是PHP5.3版本,因为我们正式环境的版本就是5.3,尽量保证开发环境与生产环境的一致

* PHP可以从数据库中取出上百万的数据,但是取出的数据放哪里呢,数组?对象?此时就会有内存溢出的问题

* php.ini中修改memory_limit 的值可以将内存设置为很大,为-1则不限制,但是数据量大时会特别卡,

* 而且有时你不一定有权限修改php.ini

* 可在程序中临时修改php.ini中的值

* ini_set('memory_limit', '128M');//设置脚本允许分配的字节的最大内存大小

* @param [int] limit 导出条数

* @param [int] isexcel 1导出excel,2导出csv

* @return 直接输出excel或csv文件到浏览器下载

* @author guangzhengren@sina.com

* @date 2017-07-21

*/

//设置超时时间,PHP默认30秒超时时间,当数据量大时若超时写入文件的内容则不全

set_time_limit(1800);

//默认导出20条数据

$limit = $_GET['limit'] ? intval($_GET['limit']) : 20;

//默认导出excel文件

$isexcel = $_GET['isexcel'] ? intval($_GET['isexcel']) : 1;

//连接数据库

$con = mysql_connect('localhost','root','123456') or die('DB connect failed!'."\n");

//选库

mysql_select_db('db_ljlj',$con);

//sql语句

$wechatSql = 'SELECT id,appid,mch_appid,wx_appid,openid,consume_id,contract_number,out_trade_no,transaction_id,pass_trade_no,value,charge,pay_result,out_refund_no,refund_id,pass_refund_no,is_subscribe,bank_type,refund_channel,add_time,modify_time  FROM `wechat_annal` ORDER BY id DESC LIMIT '.$limit;

//执行查询,返回类型为 resource 的结果集

//mysql_unbuffered_query() 向 MySQL 发送一条 SQL 查询 query ,但不像 mysql_query() 那样自动获取并缓存结果集。一方面,这在处理很大的结果集时会节省可观的内存。另一方面,可以在获取第一行后立即对结果集进行操作,而不用等到整个 SQL 语句都执行完毕

$resource = mysql_unbuffered_query($wechatSql);

//文件名称,不含后缀

$filename = 'test';

//数据,表头

$tableHeader = array('ID','商户appid','商户号','微信号','openid','收款表主键','合同号','商户订单号','微信订单号','商户单号','金额','手续费','支付结果','商户退款单号','微信退款单号','通道退款单号','是否关注公众号','付款银行','退款手续费','订单生成时间','订单修改时间');

//判断导出excel还是csv

if($isexcel == 2){

//导出csv文件

exportCsv($resource,$tableHeader,$filename);

}else{

//导出excel

exportExcel($resource,$tableHeader,$filename);

}

/**

* 导出excel文件

* @param $resource Mysql结果集

* @param $tableHeader 文件表头

* @param $filename 文件名称,不含后缀

*/

function exportExcel($resource,$tableHeader,$filename)

{

// 输出excel文件头

header("Content-Type:application/vnd.ms-excel");

header("Content-Disposition:attachment;filename=".iconv("utf-8",  "GB2312",  $filename).".xls");

header("Content-type: text/html; charset=utf-8");

echo "<!DOCTYPE html PUBLIC '-//W3C//DTD XHTML 1.0 Transitional//EN' 'http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd'><html xmlns='http://www.w3.org/1999/xhtml'><head><meta http-equiv='Content-Type' content='text/html;  charset=UTF-8' /></head><title>".$filename."</title><body><table width='90%'><tr>";

foreach ($tableHeader as $tab)

{

echo '<td >'.$tab.'</td>';

}

echo '</tr>';

while($rowCon = mysql_fetch_assoc($resource))

{

echo '<tr>';

foreach ($rowCon as $v)

{

echo '<td>'.$v.'</td>';

}

echo '</tr>';

}

echo '</table>';

exit;

}

/**

* 导出csv文件

* @param $resource Mysql结果集

* @param $tableHeader 文件表头

* @param $filename 文件名称,不含后缀

*/

function exportCsv($resource,$tableHeader,$filename)

{

// 输出csv文件头

header('Content-Type: application/vnd.ms-excel;charset=gbk');

header('Content-Disposition: attachment;filename="'.$filename.'.csv"');

header('Cache-Control: max-age=0');

// PHP文件句柄,php://output 表示直接输出到浏览器

$fp = fopen('php://output', 'a');

// 输出csv列头信息

foreach ($tableHeader as $i => $v)

{

// CSV的Excel支持GBK编码,一定要转换,否则乱码

$tableHeader[$i] = iconv('utf-8', 'gbk', $v);

}

// 写入列头

fputcsv($fp, $tableHeader);

// 计数器

$cnt = 0;

// 每隔$limit行,刷新一下输出buffer,节约资源

$limit = 10000;

while($rowCon = mysql_fetch_assoc($resource))

{

if ($limit == $cnt)

{

//刷新一下输出buffer,防止由于数据过多造成问题

ob_flush();

flush();

$cnt = 0;

}

foreach ($rowCon as $j=>$val)

{

$row[$j] = iconv('utf-8', 'gbk', $val);

}

$cnt++;

fputcsv($fp, $row);

}

exit;

}

?>

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容

  • 1.创建文件夹 !/bin/sh mkdir -m 777 "%%1" 2.创建文件 !/bin/sh touch...
    BigJeffWang阅读 10,015评论 3 53
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,585评论 18 139
  • php.ini设置,上传大文件: post_max_size = 128Mupload_max_filesize ...
    bycall阅读 6,736评论 3 64
  • 个人学习批处理的初衷来源于实际工作;在某个迭代版本有个BS(安卓手游模拟器)大需求,从而在测试过程中就重复涉及到...
    Luckykailiu阅读 4,678评论 0 11
  • 我的博客:https://blog.thuol.com 说明 亲测原稿。以 D:\wnmp 目录为例(实际目录...
    00天火00阅读 1,102评论 1 7