澳门新葡萄京官网注册PHP把导入和导出CSV文件到mysql数据库方法

制作网站时,经常会遇到检索数据列表的情况。通常用户希望下载这些列表数据并存储到客户端。当然下载这些数据的时候是需要固定的格式,以便于用Excel等软件阅览。说的简单一些就是CSV/Excel数据导出

下面只是一个简单的实例

例子,生成csv文件并下载
//要生成csv文件的数组
$csvArr=array();
$csvArr[]=array(‘用户编号1′,’上班日期1′,’签到时间1′,’签退时间1’);
$csvArr[]=array(‘用户编号2′,’上班日期2′,’签到时间2′,’签退时间2’)

注意事项:

我们先准备mysql数据表,假设项目中有一张记录学生信息的表student,并有id,name,sex,age分别记录学生的姓名、性别、年龄等信息。

download_send_headers(“data_export_” . date(“Y-m-d”) . “.csv”);
$head=array(‘用户编号’,’上班日期’,’签到时间’,’签退时间’);

输出文档的基本格式为:列1,列2,列3,,列nn

 代码如下

echo array2csv($csvArr,$head);
unset($csvArr);
die();

格式化数据的同时要对特殊的字符进行过滤。譬如,如果不转换为全角,会导致数据格式的混乱。

CREATE TABLE `student` ( 
  `id` int(11) NOT NULL auto_increment, 
  `name` varchar(50) NOT NULL, 
  `sex` varchar(10) NOT NULL, 
  `age` smallint(3) NOT NULL default ‘0’, 
  PRIMARY KEY  (`id`) 
) ENGINE=MyISAM  DEFAULT CHARSET=utf8; 

function array2csv(array &$array,$head)
{
   if (count($array) == 0) {
     return null;
   }
   ob_start();
   $df = fopen(“php://output”,
‘w’);
   if(!$head){
        $head=array_keys(reset($array));
   }
   fputcsv($df,$head);
   foreach ($array as $row)
{
      fputcsv($df, $row);
   }
   fclose($df);
   return ob_get_clean();
}

实现方法:

我们还需要一个html交互页面,放置导入表单和导出按钮。

function download_send_headers($filename) {
    // disable caching
    $now = gmdate(“D, d M Y H:i:s”);
    header(“Expires: Tue, 03 Jul 2001 06:00:00 GMT”);
    header(“Cache-Control: max-age=0, no-cache, must-revalidate,
proxy-revalidate”);
    header(“Last-Modified: {$now} GMT”);

举例:检索数据库表,将结果保存为字符串,进行格式和特殊字符的过滤后,导出到客户端的CSV文件。

 代码如下

    // force download 
    header(“Content-Type: application/force-download”);
    header(“Content-Type: application/octet-stream”);
    header(“Content-Type: application/download”);

OrderSearchEdit_saveCSV.php

<form id=”addform” action=”do.php?action=import” method=”post” enctype=”multipart/form-data”> 
    <p>请选择要导入的CSV文件:<br/><input type=”file” name=”file”> <input type=”submit” 
    class=”btn” value=”导入CSV”> 
    <input type=”button” class=”btn” value=”导出CSV” onclick=”window.location.href=’do.php? 
    action=export'”></p> 
</form> 

    // disposition / encoding on response body
    header(“Content-Disposition: attachment;filename={$filename}”);
    header(“Content-Transfer-Encoding: binary”);
}

?phpinclude($_SERVER[”DOCUMENT_ROOT”]./ftcart/OrderInfoManager.class.php);include($_SERVER[”DOCUMENT_ROOT”]./ftcart/CommonUtil.php);include($_SERVER[”DOCUMENT_ROOT”]./ftcart/CommonConst.php);include($_SERVER[”DOCUMENT_ROOT”]./ftcart/CommonErrorMsg.php);include($_SERVER[”DOCUMENT_ROOT”]./webadmin.php);$searchcase=newOrderInfoManager();$nowTime=microtime_float();$searchcase=unserialize($_SESSION[ORDER_SEARCH_CASE]);$listcsv=$searchcase-doCSV();//调用doCSV()方法$filename=$nowTime;//$filename=str_replace(:,,$nowTime);//$filename=trim($filename);//echo$filename;header(Content-Disposition:attachment;filename=.$filename..csv);header(”Content-Type:APPLICATION/OCTET-STREAM”);echo注文No.,注文日,氏名,住所,支

选择好本地csv文件后,点击导入,提交到do.php?action=import处理,而点击导出按钮则请求地址do.php?action=export进行数据导出处理。

php array生成csv文件

1.导入CSV

<?php
$data = array(
        array( ‘row_1_col_1’, ‘row_1_col_2’, ‘row_1_col_3’ ),
        array( ‘row_2_col_1’, ‘row_2_col_2’, ‘row_2_col_3’ ),
        array( ‘row_3_col_1’, ‘row_3_col_2’, ‘row_3_col_3’ ),
    );
$filename = “example”;
 
    header(“Content-type: text/csv”);
    header(“Content-Disposition: attachment;
filename={$filename}.csv”);
    header(“Pragma: no-cache”);
    header(“Expires: 0”);
 
outputCSV($data);
 
function outputCSV($data) {
        $outputBuffer = fopen(“php://output”, ‘w’);
        foreach($data as $val) {
        foreach ($val as $key => $val2) {
         $val[$key] =
iconv(‘utf-8’, ‘gbk’, $val2);
//
CSV的Excel支持GBK编码,一定要转换,否则乱码
         }
            fputcsv($outputBuffer, $val);
        }
        fclose($outputBuffer);
    }
 
?>

do.php需要根据get过来的参数,分别处理导入和导出过程,php结构如下:

解决 fgetcsv函数在php5.2.8 中的bug

include_once (“connect.php”); //连接数据库 
 

环境linux

 代码如下

问题解析出来的数据不完整,有为空的字段
网上查了下说是在php5.2.8 中存在bug
解决办法是使用自定义函数

$action = $_GET[‘action’]; 
if ($action == ‘import’) { //导入CSV 
   //导入处理 
} elseif ($action==’export’) { //导出CSV 
   //导出处理 

function __fgetcsv(& $handle, $length = null, $d = ‘,’, $e = ‘”‘) {
     $d = preg_quote($d);
     $e = preg_quote($e);
     $_line = “”;
     $eof=false;
     while ($eof != true) {
         $_line .= (empty ($length) ? fgets($handle) : fgets($handle,
$length));
         $itemcnt =
preg_match_all(‘/’ . $e .
‘/’, $_line, $dummy);
         if ($itemcnt % 2 == 0)
             $eof = true;
     }
     $_csv_line = preg_replace(‘/(?: |[ ])?$/’, $d,
trim($_line));
     $_csv_pattern = ‘/(‘ . $e . ‘[^’ . $e . ‘]*(?:’ . $e . $e .
‘[^’ . $e . ‘]*)*’ . $e . ‘|[^’ . $d . ‘]*)’ . $d . ‘/’;
     preg_match_all($_csv_pattern, $_csv_line, $_csv_matches);
     $_csv_data = $_csv_matches[1];
     for ($_csv_i = 0; $_csv_i < count($_csv_data);
$_csv_i++) {
         $_csv_data[$_csv_i] = preg_replace(‘/^’ . $e . ‘(.*)’ .
$e . ‘$/s’, ‘$1’ , $_csv_data[$_csv_i]);
         $_csv_data[$_csv_i] = str_replace($e . $e, $e,
$_csv_data[$_csv_i]);
     }
     return empty ($_line) ? false : $_csv_data;
}

导入CSV处理流程:校验csv文件合法性(本文忽略)->打开读入并解析csv文件中的字段->循环获取各字段值->批量添加到数据表中->完成。

excel无法正确读取长度超过32K的CSV域问题

 代码如下

php 导出csv文件用excel打开后,产品表述字段分两行显示。
查看了下这个字段发现这个字段超过32K的字符,excel会把字符串打断成两行,如果小于32K,显示正常。
这是EXCEL的限制,目前还没有找到解决办法。
excel一个单元格最多是32767个字符。

if ($action == ‘import’) { //导入CSV 
    $filename = $_FILES[‘file’][‘tmp_name’]; 
    if (empty ($filename)) { 
        echo ‘请选择要导入的CSV文件!’; 
        exit; 
    } 
    $handle = fopen($filename, ‘r’); 
    $result = input_csv($handle); //解析csv 
    $len_result = count($result); 
    if($len_result==0){ 
        echo ‘没有任何数据!’; 
        exit; 
    } 
    for ($i = 1; $i < $len_result; $i++) { //循环获取各字段值 
        $name = iconv(‘gb2312’, ‘utf-8’, $result[$i][0]); //中文转码 
        $sex = iconv(‘gb2312’, ‘utf-8’, $result[$i][1]); 
        $age = $result[$i][2]; 
        $data_values .= “(‘$name’,’$sex’,’$age’),”; 
    } 
    $data_values = substr($data_values,0,-1); //去掉最后一个逗号 
    fclose($handle); //关闭指针 
    $query = mysql_query(“insert into student (name,sex,age) values $data_values”);//批量插入数据表中 
    if($query){ 
        echo ‘导入成功!’; 
    }else{ 
        echo ‘导入失败!’; 
    } 

解决PHP生成UTF-8编码的CSV文件用Excel打开乱码的问题

注意php自带的fgetcsv函数可以轻松处理csv,使用该函数可以从文件指针中读入一行并解析CSV字段。下面的函数将csv文件字段解析并以数组的形式返回。

PHP生成UTF-8编码的CSV文件用Excel打开中文显示乱码,是由于输出的CSV文件中没有BOM。

 代码如下

<?php
$now = gmdate(“D, d M Y H:i:s”);
header(“Expires: Tue, 03 Jul 2001 06:00:00 GMT”);
header(“Cache-Control: max-age=0, no-cache, must-revalidate,
proxy-revalidate”);
header(“Last-Modified: {$now} GMT”);
 
// force download
header(“Content-Type: application/force-download”);
header(“Content-Type: application/octet-stream”);
header(“Content-Type: application/download”);
 
// disposition / encoding on response body
header(“Content-Disposition: attachment;filename={$filename}”);
header(“Content-Transfer-Encoding: binary”);
$items_data=array(
‘0’=>array(‘title’=>’test test test1’),
‘1’=>array(‘title’=>’test test test2’),
‘2’=>array(‘title’=>’test test test3’)
)
print(chr(0xEF).chr(0xBB).chr(0xBF));
//设置utf-8 + bom ,处理汉字显示的乱码
echo array2csv($items_data);
 
function array2csv(array &$array)
{
   if (count($array) == 0) {
     return null;
   }
   ob_start();
   $df = fopen(“php://output”, ‘w’);
   fputcsv($df, array_keys(reset($array)));
   foreach ($array as $row) {
      fputcsv($df, $row);
   }
   fclose($df);
   return ob_get_clean();
}
?>

function input_csv($handle) { 
    $out = array (); 
    $n = 0; 
    while ($data = fgetcsv($handle, 10000)) { 
        $num = count($data); 
        for ($i = 0; $i < $num; $i++) { 
            $out[$n][$i] = $data[$i]; 
        } 
        $n++; 
    } 
    return $out; 

导出csv文件数字会自动变科学计数法的解决方法

此外在导入到数据库中时,我们采用的是批量插入而不是一条条插入的,因此在构建SQL语句时,要稍作处理,见代码。

用程序导出的csv文件,当字段中有比较长的数字字段存在时,在用excel软甲查看csv文件时就会变成科学技术法的表现形式。
其实这个问题跟用什么语言导出csv文件没有关系。Excel显示数字时,如果数字大于12位,它会自动转化为科学计数法;如果数字大于15位,它不仅用于科学技术费表示,还会只保留高15位,其他位都变0。
解决这个问题
只要把数字字段后面加上显示上看不见的字符即可,字符串结尾加上制表符”t”.
php 程序可以这样判断,注意一定是”t”,不是’t’.
is_numeric($val)?$val.”t”:$val;

2.导出CSV

我们知道csv文件是由逗号分割符组成的纯文本文件,你可以用excel打开,效果跟xls表个一样。

导出CSV处理流程:读取学生信息表->循环记录构建逗号分隔的字段信息->设置header信息->导出文件(下载)到本地

 代码如下

… 
} elseif ($action==’export’) { //导出CSV 
    $result = mysql_query(“select * from student order by id asc”); 
    $str = “姓名,性别,年龄n”; 
    $str = iconv(‘utf-8′,’gb2312’,$str); 
    while($row=mysql_fetch_array($result)){ 
        $name = iconv(‘utf-8′,’gb2312’,$row[‘name’]); //中文转码 
        $sex = iconv(‘utf-8′,’gb2312’,$row[‘sex’]); 
        $str .= $name.”,”.$sex.”,”.$row[‘age’].”n”; //用引文逗号分开 
    } 
    $filename = date(‘Ymd’).’.csv’; //设置文件名 
    export_csv($filename,$str); //导出 

要将数据导出到本地即下在,需要修改header信息,代码如下:

 代码如下

function export_csv($filename,$data) { 
    header(“Content-type:text/csv”); 
    header(“Content-Disposition:attachment;filename=”.$filename); 
    header(‘Cache-Control:must-revalidate,post-check=0,pre-check=0’); 
    header(‘Expires:0’); 
    header(‘Pragma:public’); 
    echo $data; 

注意导入和导出的过程中,因为我们使用的是统一UTF-8编码,遇到中文字符一定要记得转码,否则可能会出现中文乱码的情况。

PHP把导入和导出CSV文件下载:

发表评论

电子邮件地址不会被公开。 必填项已用*标注