Perl中著名的Schwartzian转换问题解决实现
Perl中著名的Schwartzian转换,其产生背景主要涉及到排序问题:
比如说,根据文件名以字母顺序排序,代码如下:
use strict;
use warnings;
my @files = glob "*.xml"; #perl中文件操作符glob提供相当于shell中的通配符的功能
my @sorted_files = sort @files; #sort(),排序,默认是字母顺序排序
比如说,根据文件名长度排序,其代码如下:
use strict;
use warnings;
#length求长度。 太空船操作符,默认变量是$a,$b,返回值为-1,0,1分别表示大于,==,小于。 sort进行排序
my $files = ".xml";
my @sorted_length = sort { length($a) length($b) } @files;
上面的两种情况,对很多文件操作来说,速度还不算慢,如果是下面这种情况。
比如说:要批量比较文件大小,其代码如下:
use strict;
use warnings;
my @files = glob "*.xml";
my @sort_size = sort { -s $a -s $b } @files; #比较大小
上面的代码设计到三重(次)操作:
1. 从硬盘上获取文件大小(-s $b)
2. 比较文件大小(太空船操作)
3. 对其进行排序(sort操作)
考虑到要比较$a,$b大小时,要从硬盘中获取两次,所以次数是6次!也就是说,如果有1万个文件,总共是6万次。
其算法复杂度是: n*long(n),考虑到后两项(比较文件大小,进行排序)必然要进行的操作,但第一项却可以降低!
即一次性从硬盘中读取所有文件大小,将其放置到Perl中的默认的变量,并存储到内存中!于是又下面算法实现:
use strict;
use warnings;
my @files = glob "*.xml";
my @unsorted_pairs = map { [$_, -s $_] } @files;
my @sorted_pairs = sort { $a->[1] $b->[1] } @unsorted_pairs;
my @sorted_files = map { $_->[0] } @sorted_pairs;
看上去比较复杂,分三个步骤解释下:
第一步:遍历文件列表,对每个文件创建一个数组引用。数组引用包含两个元素:
第一个是文件名($_),第二个是文件大小(-s $_)。这样,处理每个文件只访问一次磁盘。
第二步:对二维数组排序。因比较文件大小,所以需取元素[1],比较它们的值。得到另一个二维数组。
第三步:丢掉文件大小元素,创建一个只含文件名的列表。完成目标!
上面的代码使用了两个临时数组,但这并不是必须的。我们可以一个语句就能完成所有的工作。为了达到目的,需要按照“数据从右流向左”的原理反转句子顺序,不如果将每个句子放在单独一行,并且留出足够的空间,我们依然可以写出可读性高的代码。
my @quickly_sorted_files =
map { $_->[0] }
sort { $a->[1] $b->[1] }
map { [$_, -s $_] }
@files;
这就是以Randal L. Schwartz命名的Schwartzian转换,对数据量特多的情况下,其速度要比前者快数倍!
下面写了小程序,包括在生成1万个xml文件,在两种情况下,完整代码如下:
#!/usr/bin/perl -w
use strict;
use warnings;
use autodie;
use v5.10;
######################################
### 创建要比较的10,000个.xml文件 ###
######################################
my $profix = ".xml";
foreach my $num (1..10000) {
open(my $fh, '>', $num . $profix) || die "Can not create the file: $!\n";
print $fh "This is file size testing!";
}
print "All the 10_1000 files created! \n";
######################################
### 常规转换: 遍历20次 ###
######################################
my $t1 = time();
foreach (1..20){
my @files = glob "*.xml";
my @sorted = sort { -s $a -s $b } @files;
}
say "常规算法需要时间: => ", time()- $t1;
######################################
### Schwartzian转换: 遍历20次 ###
######################################
my $t2 = time();
foreach (1..20){
my @files = glob "*.xml";
my @sorted =
map {$_->[0]}
sort {$a->[1] $b->[1]}
map {[$_, -s $_]}
@files;
}
say "Schwartzian算法需要时间: => ", time()- $t2;
输出结果:
All the 10_1000 files created!
常规算法需要时间: => 185
Schwartzian算法需要时间: => 115

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

使用Vue.js和Perl語言開發高效的網路爬蟲和資料抓取工具近年來,隨著網路的快速發展和資料的日益重要,網路爬蟲和資料抓取工具的需求也越來越大。在這個背景下,結合Vue.js和Perl語言開發高效率的網路爬蟲和資料抓取工具是個不錯的選擇。本文將介紹如何使用Vue.js和Perl語言開發這樣一個工具,並附上對應的程式碼範例。一、Vue.js和Perl語言的介

使用Vue.js和Perl語言開發系統腳本和自動化工具在目前的軟體開發環境中,系統腳本和自動化工具已成為開發人員節省時間和提高效率的重要工具。在這篇文章中,我們將介紹如何使用Vue.js和Perl語言開發系統腳本和自動化工具,並提供一些程式碼範例。 Vue.js是一個流行的JavaScript框架,用於建立使用者介面。它採用組件化的開發方式,使得開發人員可以將代

在當今數位時代,Web應用程式越來越普遍,讓我們的生活更加便利和有效率。 Python和Perl是兩種廣泛使用的程式語言,它們都是建立Web應用程式的理想選擇。但是,要建立高效能的Web應用程序,需要掌握一些最佳實踐,本文將介紹一些Python和Perl建立高效能Web應用程式的最佳實踐。選擇合適的Web框架選擇一個合適的Web框架是一個設計高效能Web應用程

perl+fastcgi+nginx搭建nginx+fastcgi是php下最流行的一套環境了,那perl會不會也有fastcgi呢,當然有,今天來搭建下nginx下perl的fastcgi.性能方面也不亞於php,但現在web程式php的流行程度perl無法比擬了,性能再好也枉然,但是部分小功能可以考慮使用perl的fastcgi來搞定.進入正題.1.準備軟體環境:nginxperl:系統自備fastcgi1.2perl安裝一般linux都有自備perl,可以不用安裝,如果確實沒有,請執行:

Redis和Perl語言開發:建立高效的命令列工具引言:Redis是一個開源的記憶體資料儲存系統,使用C語言編寫,具有高效能和靈活的特性,被廣泛用於快取、訊息佇列和即時分析等場景。 Perl是一種腳本語言,具有強大的文字處理和正規表示式功能,非常適合用於快速開發命令列工具。本文將介紹如何使用Perl語言和Redis建立高效的命令列工具,並提供相關的程式碼範例。一

文本翻譯中的多語種轉換問題,需要具體程式碼範例隨著全球化的進一步發展,文本翻譯在日常生活和商業交流中變得越來越重要。而在進行文本翻譯時,常會面臨多語種轉換的問題。本文將討論多語種轉換問題,並提供一些具體的程式碼範例來幫助讀者更好地理解和應用。多語種轉換問題主要涉及將一段文字從一種語言轉換為另一種語言。在實際應用中,我們常常需要將一段英文文本轉換為中文、法文、西

如何使用Redis和Perl開發推薦系統功能推薦系統是現代網路應用中非常重要的一部分,它可以幫助使用者發現他們可能感興趣的內容或產品。在本文中,我們將介紹如何使用Redis和Perl開發一個簡單的推薦系統功能,並提供具體的程式碼範例。首先,讓我們來了解Redis和Perl的基本概念。 Redis是一個開源的記憶體資料儲存系統,它可用作資料庫、快取和訊息中間件。它

Golang和Perl在系統程式設計上有以下關鍵差異:並發性:Golang是並發語言,而Perl則是使用外部模組實作並發性。記憶體管理:Golang採用垃圾收集,Perl使用手動記憶體管理。編譯與解釋:Golang是編譯性語言,Perl是解釋性語言。實戰案例:Golang在檢查檔案是否存在並傳回其大小的範例中表現出簡潔和有效率。
