iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > PHP编程 >使用PHP怎么在两个文件中找出相同的记录
  • 857
分享到

使用PHP怎么在两个文件中找出相同的记录

2023-06-15 07:06:10 857人浏览 泡泡鱼
摘要

使用PHP怎么在两个文件中找出相同的记录?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。引言给定a,b两个文件, 分别有x,y行数据, 其中(x, y均大于10亿

使用PHP怎么在两个文件中找出相同的记录?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。

引言

给定a,b两个文件, 分别有x,y行数据, 其中(x, y均大于10亿), 机器内存限制100M,该如何找出其中相同的记录?<!--more-->

思路

  • 处理该问题的困难主要是无法将这海量数据一次性读内内存中.

  • 一次性读不进内存中,那么是否可以考虑多次呢?如果可以,那么多次读入要怎么计算相同的值呢?

  • 我们可以用分治思想, 大而化小。相同字符串的值hash过后是相等的, 那么我们可以考虑使用hash取模, 将记录分散到n个文件中。这个n怎么取呢? php 100M内存,数组大约可以存100w的数据, 那么按a,b记录都只有10亿行来算, n至少要大于200。

  • 此时有200个文件,相同的记录肯定在同一个文件中,并且每个文件都可以全部读进内存。那么可以依次找出这200个文件中各自相同的记录,然后输出到同一个文件中,得到的最终结果就是a, b两个文件中相同的记录。

  • 找一个小文件中相同的记录很简单了吧,将每行记录作为hash表的key, 统计key的出现次数>=2就可以了。

实操

10亿各文件太大了,实操浪费时间,达到实践目的即可。

问题规模缩小为:  1M内存限制, a, b各有10w行记录, 内存限制可以用PHP的ini_set('memory_limit', '1M');来限制。

生成测试文件

生成随机数用于填充文件:

function generate(string $filename, int $batch=1000, int $batchSize=10000){    for ($i=0; $i<$batch; $i++) {        $str = '';        for ($j=0; $j<$batchSize; $j++) {            $str .= rand($batch, $batchSize) . PHP_EOL; // 生成随机数        }        file_put_contents($filename, $str, FILE_APPEND);  // 追加模式写入文件    }}generate('a.txt', 10);generate('b.txt', 10);

分割文件

  • a.txt, b.txt通过hash取模的方式分割到n个文件中.

function spiltFile(string $filename, int $mod=20, string $dir='files'){    if (!is_dir($dir)){        mkdir($dir);    }    $fp = fopen($filename, 'r');    while (!feof($fp)){        $line = fgets($fp);        $n = crc32(hash('md5', $line)) % $mod; // hash取模        $filepath = $dir . '/' . $n . '.txt';  // 文件输出路径        file_put_contents($filepath, $line, FILE_APPEND); // 追加模式写入文件    }    fclose($fp);}spiltFile('a.txt');spiltFile('b.txt');
  • 执行splitFile函数, 得到如下图files目录的20个文件。

查找重复记录

现在需要查找20个文件中相同的记录, 其实也就是找一个文件中的相同记录,操作个20次。

  • 找一个文件中的相同记录:

function search(string $inputFilename, $outputFilename='output.txt'){    $table = [];    $fp = fopen($inputFilename, 'r');    while (!feof($fp))    {        $line = fgets($fp);        !isset($table[$line]) ? $table[$line] = 1 : $table[$line]++; // 未设置的值设1,否则自增    }    fclose($fp);    foreach ($table as $line => $count)    {        if ($count >= 2){ // 出现大于2次的则是相同的记录,输出到指定文件中            file_put_contents($outputFilename, $line, FILE_APPEND);        }    }}
  • 找出所有文件相同记录:

function searchAll($dirs='files', $outputFilename='output.txt'){    $files = scandir($dirs);    foreach ($files as $file)    {        $filepath = $dirs . '/' . $file;        if (is_file($filepath)){            search($filepath, $outputFilename);        }    }}
  • 到这里已经解决了大文件处理的空间问题,那么时间问题该如何处理? 单机可通过利用CPU的多核心处理,不够的话通过多台服务器处理。

完整代码

<?phpini_set('memory_limit', '1M'); // 内存限制1Mfunction generate(string $filename, int $batch=1000, int $batchSize=10000){    for ($i=0; $i<$batch; $i++) {        $str = '';        for ($j=0; $j<$batchSize; $j++) {            $str .= rand($batch, $batchSize) . PHP_EOL; // 生成随机数        }        file_put_contents($filename, $str, FILE_APPEND);  // 追加模式写入文件    }}function spiltFile(string $filename, int $mod=20, string $dir='files'){    if (!is_dir($dir)){        mkdir($dir);    }    $fp = fopen($filename, 'r');    while (!feof($fp)){        $line = fgets($fp);        $n = crc32(hash('md5', $line)) % $mod; // hash取模        $filepath = $dir . '/' . $n . '.txt';  // 文件输出路径        file_put_contents($filepath, $line, FILE_APPEND); // 追加模式写入文件    }    fclose($fp);}function search(string $inputFilename, $outputFilename='output.txt'){    $table = [];    $fp = fopen($inputFilename, 'r');    while (!feof($fp))    {        $line = fgets($fp);        !isset($table[$line]) ? $table[$line] = 1 : $table[$line]++; // 未设置的值设1,否则自增    }    fclose($fp);    foreach ($table as $line => $count)    {        if ($count >= 2){ // 出现大于2次的则是相同的记录,输出到指定文件中            file_put_contents($outputFilename, $line, FILE_APPEND);        }    }}function searchAll($dirs='files', $outputFilename='output.txt'){    $files = scandir($dirs);    foreach ($files as $file)    {        $filepath = $dirs . '/' . $file;        if (is_file($filepath)){            search($filepath, $outputFilename);        }    }}// 生成文件generate('a.txt', 10);generate('b.txt', 10);// 分割文件spiltFile('a.txt');spiltFile('b.txt');// 查找记录searchAll('files', 'output.txt');

看完上述内容是否对您有帮助呢?如果还想对相关知识有进一步的了解或阅读更多相关文章,请关注编程网PHP编程频道,感谢您对编程网的支持。

--结束END--

本文标题: 使用PHP怎么在两个文件中找出相同的记录

本文链接: https://www.lsjlt.com/news/278699.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作