本文介紹基於Python中whitebox模塊,對大量長時間序列柵格遙感影像的每一個像元進行忽略NoData值的多時序平均值求取~ ...
本文介紹基於Python中whitebox模塊,對大量長時間序列柵格遙感影像的每一個像元進行忽略NoData值的多時序平均值求取。
在文章Python ArcPy批量計算多時相遙感影像的各像元平均值中,我們介紹了基於Python中Arcpy模塊實現多時相遙感影像數據的平均值求取方法。但是這一方法具有一個問題,即對於任意一個像元,只要該像元在任意一個時相的圖像中是無效值(即為NoData),那麼該像元在最終求出的平均值結果圖中像素值也將會是無效值NoData。這就導致在我們最終計算得到的平均值結果圖層中,具有很多空白區域(像素值為NoData的區域)。
為瞭解決這一問題,這裡我們再介紹一種基於Python中另一個地理空間數據分析庫——whitebox
,實現多時像遙感影像數據逐像元平均值的求取方法。
首先,需要下載並安裝whitebox
這一模塊。如果大家電腦中已經有了Anaconda環境,就可以直接按照Python地理分析庫whitebox在Anaconda中的配置這篇文章中介紹的方法下載、安裝whitebox
。
本文要實現的需求和文章Python ArcPy批量計算多時相遙感影像的各像元平均值中的一致,這裡就不再贅述。本文所需用到的代碼如下。
# -*- coding: utf-8 -*-
"""
Created on Sun Apr 17 15:04:29 2022
@author: fkxxgis
"""
import glob
from whitebox import WhiteboxTools
tif_file_path="E:/LST/Data/MODIS/test/"
average_file_path="E:/LST/Data/MODIS/06_Average/"
wbt=WhiteboxTools()
wbt.work_dir=tif_file_path
tif_file_name=glob.glob(tif_file_path+"*.tif")
tif_file_year=tif_file_name[0][-18:-14]
one_year_tif_list=[]
for tif_file in tif_file_name:
if tif_file[-18:-14]==tif_file_year:
one_year_tif_list.append(tif_file)
tif_file_temp=tif_file
if tif_file==tif_file_name[len(tif_file_name)-1]:
wbt.average_overlay(inputs=';'.join(one_year_tif_list),
output=average_file_path+tif_file_year+"_Ave.tif")
else:
wbt.average_overlay(inputs=';'.join(one_year_tif_list),
output=average_file_path+tif_file_year+"_Ave.tif")
one_year_tif_list=[]
one_year_tif_list.append(tif_file)
tif_file_year=tif_file[-18:-14]
其中,tif_file_path
是原有計算平均值前遙感圖像的保存路徑,average_file_path
是我們新生成的求取平均值後遙感影像的保存路徑,也就是結果保存路徑。
上述代碼的整體思路其實和文章Python ArcPy批量計算多時相遙感影像的各像元平均值這篇文章是非常類似的。首先,同樣需要在資源管理器中,將tif_file_path
路徑下的各文件以“名稱”排序的方式進行排序;隨後,利用arcpy.ListRasters()
函數,獲取路徑下原有的全部.tif
格式的圖像文件,並截取第一個文件的部分文件名,從而獲取其成像時間的具體年份。
接下來,遍歷tif_file_path
路徑下全部.tif
格式圖像文件。其中,我們通過一個簡單的判斷語句if tif_file[0:4]==tif_file_year:
,來確定某一年的遙感影像是否已經讀取完畢——如果已經讀取完畢,例如假如2001
年成像的8
幅遙感影像都已經遍歷過了,那麼就對這8
景遙感影像加以逐像元的平均值求取,並開始對下一個年份(即2005
年)成像的遙感影像繼續加以計算;如果還沒有讀取完畢,例如假如2001
年成像的8
幅遙感影像目前僅遍歷到了第5
幅,那麼就不求平均值,繼續往下遍歷,直到遍歷完2001
年成像的8
幅遙感影像。
這裡相信大家也看到了為什麼我們要在前期先將文件夾中的文件按照“名稱”排序——是為了保證同一年成像的所有遙感影像都排列在一起,遍歷時只要遇到一個新的年份,程式就知道上一個年份的所有圖像都已經遍歷完畢了,就可以將上一個年份的所有柵格圖像加以平均值求取。
本文代碼和前期博客中代碼不一樣的部分就在於,這裡是用到whitebox
模塊而非arcpy
模塊來實現同一年份遙感影像的逐像元平均值求取。在這裡,wbt.average_overlay()
函數就是我們實現這一步驟的關鍵,其中inputs
參數表示需要進行平均值計算的同一年份的所有遙感影像,output
表示求取平均值後得到的結果圖像。
最後,通過if tif_file==tif_file_name[len(tif_file_name)-1]:
這個判斷,來確認是否目前已經遍歷到文件夾中的最後一個圖像文件。如果是的話,就需要將當前成像年份的所有圖像進行平均值的求取,並宣告代碼完成運行。
這裡需要註意,由於我們在此沒有用到arcpy
模塊,因此代碼也就不一定非要在 IDLE (Python GUI) 中運行了,常見的編譯器都可以運行。在代碼運行過程中,還可以看到具體運行情況與進度。
代碼運行完畢後,即可得到求解平均值後的結果圖層。
最後還有一個問題——在我用這一代碼進行實踐後發現,如果計算平均值前的圖層具有兩個或兩個以上的波段,那麼得到的結果圖層整體看還好,如下圖所示。
但放大後,會發現得到的結果呈現出如下所示的條帶狀。
而如果計算平均值前的圖層僅具有一個波段的話,就不會出現這種問題;如下圖所示。
因此,大家在使用本文的代碼對大量長時間序列柵格遙感影像的每一個像元進行忽略Nodata值的多時序平均值求取時,一定註意輸入圖層要僅含有一個波段;否則結果就會出現條帶狀的錯誤。