Perl中著名的Schwartzian轉(zhuǎn)換問題解決實現(xiàn)
Perl中著名的Schwartzian轉(zhuǎn)換,其產(chǎn)生背景主要涉及到排序問題:
比如說,根據(jù)文件名以字母順序排序,代碼如下:
use strict;
use warnings;
my @files = glob "*.xml"; #perl中文件操作符glob提供相當于shell中的通配符的功能
my @sorted_files = sort @files; #sort(),排序,默認是字母順序排序
比如說,根據(jù)文件名長度排序,其代碼如下:
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大小時,要從硬盤中獲取兩次,所以次數(shù)是6次!也就是說,如果有1萬個文件,總共是6萬次。
其算法復雜度是: n*long(n),考慮到后兩項(比較文件大小,進行排序)必然要進行的操作,但第一項卻可以降低!
即一次性從硬盤中讀取所有文件大小,將其放置到Perl中的默認的變量,并存儲到內(nèi)存中!于是又下面算法實現(xiàn):
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;
看上去比較復雜,分三個步驟解釋下:
第一步:遍歷文件列表,對每個文件創(chuàng)建一個數(shù)組引用。數(shù)組引用包含兩個元素:
第一個是文件名($_),第二個是文件大小(-s $_)。這樣,處理每個文件只訪問一次磁盤。
第二步:對二維數(shù)組排序。因比較文件大小,所以需取元素[1],比較它們的值。得到另一個二維數(shù)組。
第三步:丟掉文件大小元素,創(chuàng)建一個只含文件名的列表。完成目標!
上面的代碼使用了兩個臨時數(shù)組,但這并不是必須的。我們可以一個語句就能完成所有的工作。為了達到目的,需要按照“數(shù)據(jù)從右流向左”的原理反轉(zhuǎn)句子順序,不如果將每個句子放在單獨一行,并且留出足夠的空間,我們依然可以寫出可讀性高的代碼。
my @quickly_sorted_files =
map { $_->[0] }
sort { $a->[1] <=> $b->[1] }
map { [$_, -s $_] }
@files;
這就是以Randal L. Schwartz命名的Schwartzian轉(zhuǎn)換,對數(shù)據(jù)量特多的情況下,其速度要比前者快數(shù)倍!
下面寫了小程序,包括在生成1萬個xml文件,在兩種情況下,完整代碼如下:
#!/usr/bin/perl -w
use strict;
use warnings;
use autodie;
use v5.10;
######################################
### 創(chuàng)建要比較的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";
######################################
### 常規(guī)轉(zhuǎn)換: 遍歷20次 ###
######################################
my $t1 = time();
foreach (1..20){
my @files = glob "*.xml";
my @sorted = sort { -s $a <=> -s $b } @files;
}
say "常規(guī)算法需要時間: => ", time()- $t1;
######################################
### Schwartzian轉(zhuǎn)換: 遍歷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;
輸出結(jié)果:
All the 10_1000 files created!
常規(guī)算法需要時間: => 185
Schwartzian算法需要時間: => 115
相關文章
PyQt+socket實現(xiàn)遠程操作服務器的方法示例
這篇文章主要介紹了PyQt+socket實現(xiàn)遠程操作服務器的方法示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-08-08Qt通過QGraphicsview實現(xiàn)簡單縮放及還原效果
本文主要介紹通過QGraphicsview實現(xiàn)簡單的縮放以及縮放后還原原始大小,通過scale可以對view進行放大或縮小,具體內(nèi)容詳情跟隨小編一起看看吧2021-09-09Qt6中重大改變的QtMultimedia多媒體模塊實現(xiàn)
本文主要介紹了Qt6中重大改變的QtMultimedia多媒體模塊實現(xiàn),文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-09-09Python實現(xiàn)的北京積分落戶數(shù)據(jù)分析示例
這篇文章主要介紹了Python實現(xiàn)的北京積分落戶數(shù)據(jù)分析,結(jié)合實例形式分析了Python針對北京積分落戶數(shù)據(jù)的分析、運算、展示等相關操作技巧,需要的朋友可以參考下2020-03-03Django動態(tài)隨機生成溫度前端實時動態(tài)展示源碼示例
本篇文章主要描述的是在動態(tài)隨機生成溫度,在前端動態(tài)實時展示,主要用到兩個東西,一個是APScheduler定時任務 和websocket,最后利用echarts將數(shù)據(jù)展示出來,下面對這兩個分別進行詳細的解說2021-09-09python中的位置參數(shù)和關鍵字參數(shù)詳解
位置參數(shù)和關鍵字參數(shù)是 Python 中的兩種不同類型的函數(shù)參數(shù)傳遞方式,位置參數(shù)依賴于參數(shù)的位置順序,而關鍵字參數(shù)通過參數(shù)名傳遞,不受位置影響,本文通過代碼示例給大家詳細介紹了python中的位置參數(shù)和關鍵字參數(shù),需要的朋友可以參考下2023-12-12