如何统计QGIS里栅格图层的面积呢

现在有这样一个需求,就是需要统计一下加载在QGIS里的图层的面积,然后保存在csv里,第一列是图层名,第二列是面积,面积以平方千米计,直接在python控制台写python脚本比较方便。如下便是代码

python 复制代码
# -*- coding: utf-8 -*-
from qgis.core import QgsProject
from osgeo import gdal
import numpy as np
import processing
import csv
import os

# -------------------------------
# 设置输出 CSV 路径
# -------------------------------
output_csv = r"C:\D\tmp\tmp.csv"

# 确保目录存在
output_dir = os.path.dirname(output_csv)
if not os.path.exists(output_dir):
    try:
        os.makedirs(output_dir)
        print(f"✅ 已创建目录: {output_dir}")
    except Exception as e:
        print(f"❌ 创建目录失败: {e}")
        exit()

# -------------------------------
# 收集所有栅格图层
# -------------------------------
layers = QgsProject.instance().mapLayers().values()
raster_layers = [lyr for lyr in layers if lyr.type() == lyr.RasterLayer]

if not raster_layers:
    print("❌ 错误:没有找到任何栅格图层")
else:
    print(f"✅ 找到 {len(raster_layers)} 个栅格图层,开始处理...")

# 存储结果
results = []

# -------------------------------
# 遍历每个图层进行处理
# -------------------------------
for idx, layer in enumerate(raster_layers):
    name = layer.name()
    source = layer.source()
    print(f"\n📌 正在处理 [{idx+1}/{len(raster_layers)}]: {name}")

    # --- 1. 重投影到 EPSG:3857(内存中)
    params = {
        'INPUT': source,
        'TARGET_CRS': 'EPSG:3857',
        'OUTPUT': 'TEMPORARY_OUTPUT'
    }

    try:
        result = processing.run("gdal:warpreproject", params)
        reprojected_path = result['OUTPUT']
    except Exception as e:
        print(f"   ❌ 重投影失败: {e}")
        results.append({'name': name, 'area_km2': None})
        continue

    # --- 2. 使用 GDAL 读取数据
    ds = gdal.Open(reprojected_path)
    if ds is None:
        print(f"   ❌ 无法打开重投影数据")
        results.append({'name': name, 'area_km2': None})
        continue

    try:
        band = ds.GetRasterBand(1)
        gt = ds.GetGeoTransform()
        pixel_width = abs(gt[1])
        pixel_height = abs(gt[5])
        nodata = band.GetNoDataValue()

        data = band.ReadAsArray()
        pixel_area_m2 = pixel_width * pixel_height
        total_pixels = data.size

        # 统计有效像元
        if nodata is not None:
            valid_mask = (data != nodata) & (~np.isnan(data))
            valid_count = np.sum(valid_mask)
        else:
            valid_count = total_pixels

        # 计算面积(km²)
        total_area_km2 = (valid_count * pixel_area_m2) / 1_000_000

        print(f"   ✅ 分辨率: {pixel_width:.4f}×{pixel_height:.4f} 米, 有效像元: {valid_count:,}")
        print(f"   📊 面积: {total_area_km2:.6f} 平方公里")

        results.append({'name': name, 'area_km2': round(total_area_km2, 6)})

    except Exception as e:
        print(f"   ❌ 数据处理出错: {e}")
        results.append({'name': name, 'area_km2': None})
    finally:
        ds = None  # 释放资源

# -------------------------------
# 保存结果到 CSV
# -------------------------------
try:
    with open(output_csv, 'w', encoding='utf-8', newline='') as f:
        writer = csv.writer(f)
        writer.writerow(['图层名', '面积_平方千米'])  # 表头
        for r in results:
            area_str = f"{r['area_km2']:.6f}" if r['area_km2'] is not None else ""
            writer.writerow([r['name'], area_str])
    print(f"\n🎉 成功!结果已保存至:\n   {output_csv}")
except Exception as e:
    print(f"❌ 保存 CSV 失败: {e}")
相关推荐
零号全栈寒江独钓11 分钟前
visual studio编译wxWidgets
c++·visual studio
努力中的编程者12 分钟前
哈希表(C语言底层实现)
c语言·数据结构·c++·算法·哈希算法·散列表
mjhcsp15 分钟前
C++ 迭代加深搜索(IDDFS):从原理到实战的深度解析
c++·深度优先·迭代加深
摆烂小白敲代码25 分钟前
【数据结构与算法】汉诺塔问题(C++)
c语言·开发语言·数据结构·c++·算法·hanoi·汉诺塔问题
Trouvaille ~30 分钟前
【递归、搜索与回溯】专题(八):记忆化搜索——从暴力递归到动态规划的桥梁
c++·算法·leetcode·青少年编程·面试·蓝桥杯·动态规划
刚入坑的新人编程1 小时前
C++qt(3)-按钮类控件
开发语言·c++·qt
乐观勇敢坚强的老彭1 小时前
本周C++编程课笔记:for循环练习
java·c++·笔记
娇娇yyyyyy1 小时前
C++ 网络编程(22) beast网络库实现websocket服务器
网络·c++·websocket
西野.xuan1 小时前
【effective c++】条款四十三:学习处理模版化基类内的名称
java·c++·学习
8Qi81 小时前
LeetCode61. 旋转链表
c语言·数据结构·c++·算法·leetcode·链表·力扣