PHP 数组到 CSV
Posted
技术标签:
【中文标题】PHP 数组到 CSV【英文标题】:PHP Array to CSV 【发布时间】:2012-10-17 23:20:09 【问题描述】:我正在尝试将产品数组转换为 CSV 文件,但似乎没有计划。 CSV 文件是一长行,这是我的代码:
for($i=0;$i<count($prods);$i++)
$sql = "SELECT * FROM products WHERE id = '".$prods[$i]."'";
$result = $mysqli->query($sql);
$info = $result->fetch_array();
$header = '';
for($i=0;$i<count($info);$i++)
$row = $info[$i];
$line = '';
for($b=0;$b<count($row);$b++)
$value = $row[$b];
if ( ( !isset( $value ) ) || ( $value == "" ) )
$value = "\t";
else
$value = str_replace( '"' , '""' , $value );
$value = '"' . $value . '"' . "\t";
$line .= $value;
$data .= trim( $line ) . "\n";
$data = str_replace( "\r" , "" , $data );
if ( $data == "" )
$data = "\n(0) Records Found!\n";
header("Content-type: application/octet-stream");
header("Content-Disposition: attachment; filename=your_desired_name.xls");
header("Pragma: no-cache");
header("Expires: 0");
array_to_CSV($data);
function array_to_CSV($data)
$outstream = fopen("php://output", 'r+');
fputcsv($outstream, $data, ',', '"');
rewind($outstream);
$csv = fgets($outstream);
fclose($outstream);
return $csv;
此外,标头不会强制下载。我一直在复制和粘贴输出并保存为 .csv
编辑
问题已解决:
如果其他人也在寻找同样的东西,找到更好的方法:
$num = 0;
$sql = "SELECT id, name, description FROM products";
if($result = $mysqli->query($sql))
while($p = $result->fetch_array())
$prod[$num]['id'] = $p['id'];
$prod[$num]['name'] = $p['name'];
$prod[$num]['description'] = $p['description'];
$num++;
$output = fopen("php://output",'w') or die("Can't open php://output");
header("Content-Type:application/csv");
header("Content-Disposition:attachment;filename=pressurecsv.csv");
fputcsv($output, array('id','name','description'));
foreach($prod as $product)
fputcsv($output, $product);
fclose($output) or die("Can't close php://output");
【问题讨论】:
使用 $info[]=$result->fetch_array();否则它将有最后的产品详细信息 @JohnnyFaldo:谢谢伙计!正是我需要的。 嘿!我知道这是一个老问题,但你可以回答你自己的问题。它可以帮助人们找到类似问题的解决方案。 还有人把第一行和第二行的前两块留空吗?? 【参考方案1】:考虑使用fputcsv()
,而不是写出值。
这可能会立即解决您的问题。
评论注释:我应该提到这将在您的服务器上创建一个文件,因此您需要在输出之前读取该文件的内容,如果您不想这样做保存副本,然后您需要在完成后重新链接文件
【讨论】:
我应该提到这将在您的服务器上创建一个文件,因此您需要在输出之前读取该文件的内容,如果您不想保存副本,那么您完成后需要重新链接文件。 感谢 Vyktor 的编辑,我今天的打字很糟糕!我通常很小心。 我可以编辑上面的代码来做到这一点吗?因为它正在输出所有字段,并且使用手册中的 fputcsv 示例我无法做到这一点 array_to_CSV($data);函数 array_to_CSV($data) $outstream = fopen("php://output", 'r+'); fputcsv($outstream, $data, ',', '"'); rewind($outstream); $csv = fgets($outstream); fclose($outstream); return $csv; 是的,已经解决了,谢谢,在原问题中添加了解决方案【参考方案2】:尝试使用;
PHP_EOL
终止 CSV 输出中的每一行。
我假设文本是定界的,但没有移动到下一行?
这是一个 PHP 常量。它将确定您需要的正确行尾。
例如,Windows 使用“\r\n”。当我的输出没有突破到新的行时,我用那个破坏了我的大脑。
how to write unified new line in PHP?
【讨论】:
【参考方案3】:在我的例子中,我的数组是多维的,可能以数组作为值。所以我创建了这个递归函数来完全分解数组:
function array2csv($array, &$title, &$data)
foreach($array as $key => $value)
if(is_array($value))
$title .= $key . ",";
$data .= "" . ",";
array2csv($value, $title, $data);
else
$title .= $key . ",";
$data .= '"' . $value . '",';
由于我的数组的各个级别不适合平面 CSV 格式,因此我使用子数组的键创建了一个空白列,作为对下一级数据的描述性“介绍”。示例输出:
agentid fname lname empid totals sales leads dish dishnet top200_plus top120 latino base_packages
G-adriana ADRIANA EUGENIA PALOMO PAIZ 886 0 19 0 0 0 0 0
您可以轻松删除该“介绍”(描述性)列,但在我的情况下,我在每个子数组中都有重复的列标题,即 inbound_leads,因此在下一节之前给了我一个中断/标题。删除:
$title .= $key . ",";
$data .= "" . ",";
在 is_array() 之后进一步压缩代码并删除多余的列。
由于我想要标题行和数据行,我将两个变量传递给函数,并在完成对函数的调用后,用 PHP_EOL 终止这两个变量:
$title .= PHP_EOL;
$data .= PHP_EOL;
是的,我知道我留下了一个额外的逗号,但为了简洁起见,我没有在这里处理它。
【讨论】:
这似乎是对一个非常普遍的问题的具体答案,并且您承认留下了额外的逗号(由于使用字符串而不是数组的真正问题)以及没有引用标题行和有一些额外的不需要的电话。我重塑了这一切inside an answer to another question。【参考方案4】:我知道这是旧的,我有一个案例我需要将数组键也包含在 CSV 中,所以我更新了 Jesse Q 的脚本来做到这一点。 我使用一个字符串作为输出,因为 implode 无法添加新行(新行是我添加的,应该确实存在)。
请注意,这只适用于单值数组(key, value)
。但可以轻松更新以处理多维(key, array())
。
function arrayToCsv( array &$fields, $delimiter = ',', $enclosure = '"', $encloseAll = false, $nullToMysqlNull = false )
$delimiter_esc = preg_quote($delimiter, '/');
$enclosure_esc = preg_quote($enclosure, '/');
$output = '';
foreach ( $fields as $key => $field )
if ($field === null && $nullToMysqlNull)
$output = '';
continue;
// Enclose fields containing $delimiter, $enclosure or whitespace
if ( $encloseAll || preg_match( "/(?:$delimiter_esc|$enclosure_esc|\s)/", $field ) )
$output .= $key;
$output .= $delimiter;
$output .= $enclosure . str_replace($enclosure, $enclosure . $enclosure, $field) . $enclosure;
$output .= PHP_EOL;
else
$output .= $key;
$output .= $delimiter;
$output .= $field;
$output .= PHP_EOL;
return $output ;
【讨论】:
【参考方案5】:通过内置 php 函数 fputcsv 处理逗号、引号等,将数据数组转换为 csv 'text/csv' 格式。 看看https://coderwall.com/p/zvzwwa/array-to-comma-separated-string-in-phphttp://www.php.net/manual/en/function.fputcsv.php
【讨论】:
欢迎提供解决方案的链接,但请确保您的答案在没有它的情况下有用:add context around the link 这样您的其他用户就会知道它是什么以及为什么会出现,然后引用最相关的内容您链接到的页面的一部分,以防目标页面不可用。 Answers that are little more than a link may be deleted.【参考方案6】:这是一个将数组导出为 csv 字符串的简单解决方案:
function array2csv($data, $delimiter = ',', $enclosure = '"', $escape_char = "\\")
$f = fopen('php://memory', 'r+');
foreach ($data as $item)
fputcsv($f, $item, $delimiter, $enclosure, $escape_char);
rewind($f);
return stream_get_contents($f);
$list = array (
array('aaa', 'bbb', 'ccc', 'dddd'),
array('123', '456', '789'),
array('"aaa"', '"bbb"')
);
var_dump(array2csv($list));
Reference
【讨论】:
【参考方案7】:它对我有用。
$f=fopen('php://memory','w');
$header=array("asdf ","asdf","asd","Calasdflee","Start Time","End Time" );
fputcsv($f,$header);
fputcsv($f,$header);
fputcsv($f,$header);
fseek($f,0);
header('content-type:text/csv');
header('Content-Disposition: attachment; filename="' . $filename . '";');
fpassthru($f);```
【讨论】:
以上是关于PHP 数组到 CSV的主要内容,如果未能解决你的问题,请参考以下文章