Il decreto ‘di Natale’ sugli spostamenti piccoli comuni

Con Maurizio e Andrea abbiamo raccontato Il decreto “di Natale”, in chilometri, ovvero come calcolare le aree in cui sarà possibile spostarsi nei giorni 28, 29, 30 dicembre 2020 e 4 gennaio 2021, secondo quanto indicato nel Decreto Legge numero 172 del 18 dicembre 2020.
Maurizio le ha calcolate usando Python, Andrea l’ha fatto “a riga di comando”; io l’ho fatto con QGIS e SpatiaLite.

Vediamo come ho risolto il #30cappa del decreto “di Natale”, versione SpatiaLite.

Software usati

  • SpatiaLite 5.0 e le sue incredibile funzioni spaziali.
  • Per la visualizzazione ho usato QGIS 3.16 Hannover.

Osservazioni

  • nello script SQL utilizzerò la funzione ST_Cutter() che sfrutta le chiavi primarie delle geo-tabelle, quindi è necessario crearle tramite definizione diretta dei campi, e definire una primary key per ogni tabella;
  • uso sempre DROP TABLE IF EXISTS "nomeTabella"; ad inizio processo, perché se la tabella che sto creando esistesse già del database, verrebbe cancellata e non mi segnalerebbe errore (IF EXISTS);
  • uso sempre CastToMultiPolygon(geometry) sulla geometria per lavorare o forzare sempre il tipo di geometria come MultiPolygon;
  • quando creo una nuova geo-tabella, va aggiunto SELECT AddGeometryColumn ('nomeTabella','nomeCampoGeom',EPSG,'tipoGeometria','dimensione');
  • quando rigenero una geo-tabella, va aggiunto SELECT RecoverGeometryColumn('nomeTabella','nomeCampoGeom',EPSG,'tipoGeometria','dimensione');

Limiti amministrativi comunali

Shapefile ISTAT al 01/01/2020 con aggiunto il campo abitanti (01/01/2020), il processo è spiegato nel post di Andrea Borruso; non è un processo molto complesso, ma occorre una pulizia preventiva del dataset.

shapefile ISTAT 2020

Creo Buffer di 30 km

sui comuni con popolazione <= 5.000 (è uno dei requisiti del Decreto)

-- crea tabella - buffer da 30 km su comuni da 5k
DROP TABLE IF EXISTS "b30k_comuni5k"; -- se esiste cancella la tabella
CREATE TABLE "b30k_comuni5k"
      ('pk_uid' integer PRIMARY KEY autoincrement NOT NULL,'pro_com_t' text);
-- aggiunge campo geom, fissa EPSG, tipo di geometria e dimensione
SELECT AddGeometryColumn ('b30k_comuni5k','geom',32632,'MULTIPOLYGON','XY');
-- popola la tabella con abitanti minori di 5001
INSERT INTO "b30k_comuni5k"
(pk_uid, pro_com_t, geom)
SELECT pk_uid, pro_com_t, CastToMultiPolygon(ST_Buffer (geom,30000)) AS geom
FROM (SELECT pk_uid,pro_com_t, geom 
      FROM "Com01012020_g_WGS84"
      WHERE abitanti &lt;=5000); -- condizione sulla popolazione

ci sono alcuni comuni con enclave e quindi il buffer sarà come illustrato sotto:

Estraggo i soli capoluoghi di provincia

la condizione è sul campo CC_UTS della tabella attributi dello shapefile Comuni, che ha solo due valori 1, 0; serviranno successivamente per tagliare i buffer da 30 km

-- crea tabella - capoluoghi di provincia
DROP TABLE IF EXISTS "capoluoghi_prov";
CREATE TABLE "capoluoghi_prov" 
      ("pk_uid" integer PRIMARY KEY autoincrement NOT NULL,"pro_com_t" text);
-- aggiunge campo geom, fissa EPSG, tipo di geometria e dimensione
SELECT AddGeometryColumn ('capoluoghi_prov','geom',32632,'MULTIPOLYGON','XY');
-- popola la tabella
INSERT INTO "capoluoghi_prov"
(pk_uid, pro_com_t, geom)
SELECT pk_uid, pro_com_t, geom
FROM (SELECT pk_uid,pro_com_t,geom  
      FROM "Com01012020_g_WGS84"
      WHERE cc_uts != 0); -- condizione sui capoluoghi di provincia

Unisco tutti i comuni

servirà successivamente per ritagliare i buffer con i limiti dell’italico stivale.

Per creare questa tabella preferisco farlo come aggiornamento e quindi utilizzo UPDATE.

-- crea tabella temporanea unendo tutti i comuni
DROP TABLE IF EXISTS "tmp_italia";
CREATE TABLE "tmp_italia" 
      ("pk_uid" INTEGER PRIMARY KEY);
-- aggiunge campo geom, fissa EPSG, tipo di geometria e dimensione
SELECT AddGeometryColumn ('tmp_italia','geom',32632,'MULTIPOLYGON','XY');
-- popola la tabella
INSERT INTO "tmp_italia" VALUES (1,NULL);
-- aggiorna tabella
UPDATE "tmp_italia" SET geom = 
      (SELECT CastToMultiPolygon(ST_Union(geom)) AS geom
       FROM "Com01012020_g_WGS84") 
       WHERE  pk_uid = 1;

Suddivido lo stivale

la suddivisione evita lunghi tempi di analisi (vale per grandi poligoni); la funzione è presente da SpatiaLite 5, suddivide un poligono a partire dal numero di nodi indicati come argomento:

  • ST_Subdivide(geom,2048): richiede il nome del campo geometrico e il numero di nodi, più grande è il numero meno suddivisioni farà;
  • in questo caso occorre recuperare la geometria tramite: SELECT RecoverGeometryColumn('nomeTabella','nomeTabellaGeometria',EPSG,'tipoGeometria','dimensione');, per maggiori info, vedi riferimenti a fine post;
  • la funzione restituisce unica feature, quindi se la geometria di input fosse un Polygon restituirebbe un Multipolygon;
-- suddivide l'intero poligono nazionale
DROP TABLE IF EXISTS "italia_subd";
CREATE TABLE "italia_subd" AS
SELECT ST_Subdivide(geom,2048) AS geom FROM "tmp_italia";
-- recupero la geometria
SELECT RecoverGeometryColumn('italia_subd','geom',32632,'MULTIPOLYGON','XY');

Estraggo le parti elementari

come detto sopra, la suddivisione crea unica feature e quindi occorre esploderla nelle sue parti elementari usando:

  • SELECT ElementaryGeometries( 'nomeTabellaInput','nomeCampo Geometry', 'nomeTabellaOutput', 'chiavePrimariaTabella', 'chiavePrimariaTabellaInput', 1 )
-- estrae i poligoni elementari del poligono nazionale
SELECT ElementaryGeometries( 'italia_subd',
                             'geom',
                             'italia_subd_elem',
                             'pk_elem',
                             'out_multi_id', 1 ) as num;

Taglio i buffer da 30 km

con l’Italia suddivisa a pezzi: per il taglio (topologico) utilizzo ST_Cutter(), una funzione presente a partire dalla versione 4.4 di SpatiaLite: la funzione vuole una geo-tabelle come input e un’altra geo-tabella da usare come lama per il taglio:

  • SELECT ST_Cutter(NULL, 'nomeTabellaInput', NULL, NULL, 'nomeTabellaLAMA', NULL, 'nomeTabellaOutput', 1, 1); per maggiori info, vedi riferimenti a fine post.
-- taglia i buffer secondo il vettore dato
SELECT ST_Cutter(NULL, 'b30k_comuni5k', NULL, NULL, 'italia_subd_elem', NULL, 'b30k_com5k_italy_subd_elem', 1, 1);

Ricompongo i pezzi

fondo (ST_Union()) i buffer da 30 km tagliati in precedenza, il valore NULL indica che l’elemento è fuori dalla lama e quindi li scartiamo (WHERE "blade_italia_subd_elem_pk_elem" is NOT NULL)

<img src="./imgs/img_071.png" width="600">

-- ricompone i pezzi - ricorda che pk_uid si riferisce alla tabella comuni 'Com01012020_g_WGS84'
DROP TABLE IF EXISTS "b30k_com5k_italy";
CREATE TABLE "b30k_com5k_italy" 
      ("pk_uid" INTEGER PRIMARY KEY);
-- aggiunge campo geom, fissa EPSG, tipo di geometria e dimensione
SELECT AddGeometryColumn ('b30k_com5k_italy','geom',32632,'MULTIPOLYGON','XY');
INSERT INTO "b30k_com5k_italy" 
(pk_uid,geom)
SELECT k."input_b30k_comuni5k_pk_uid" AS pk_uid, k.geom
FROM (SELECT "input_b30k_comuni5k_pk_uid", "blade_italia_subd_elem_pk_elem", CastToMultiPolygon(ST_Union("geom")) AS geom
      FROM "b30k_com5k_italy_subd_elem"
      WHERE "blade_italia_subd_elem_pk_elem" is NOT NULL -- scarta gli elementi NULL
      GROUP BY "input_b30k_comuni5k_pk_uid") k;
-- recupero la geometria
SELECT RecoverGeometryColumn('b30k_com5k_italy','geom',32632,'MULTIPOLYGON','XY');

Taglio i buffer

dai buffer 30 km sagomati con lo stivale, taglio i capoluoghi di provincia

-- taglia i buffer sagomati con lo stivale secondo il vettore dei capoluoghi
SELECT ST_Cutter(NULL, 'b30k_com5k_italy', NULL, NULL, 'capoluoghi_prov', NULL, 'b30k_com5k_italy_subd_elem_capolp', 1, 1);

Ricompongo i pezzi

buco e pulisco i buffer (WHERE "blade_capoluoghi_prov_pk_uid" IS NULL AND ST_Area(geom) &gt; 1) e li fondo (ST_Union("geom")) secondo il criterio usato sopra:

<img src="./imgs/img_091.png" width="600">

-- ricompone i pezzi - ricorda che pk_uid si riferisce alla tabella comuni 'Com01012020_g_WGS84'
DROP TABLE IF EXISTS "finale";
CREATE TABLE "finale" 
      ("pk_uid" INTEGER PRIMARY KEY);
-- aggiunge campo geom, fissa EPSG, tipo di geometria e dimensione
SELECT AddGeometryColumn ('finale','geom',32632,'MULTIPOLYGON','XY');
INSERT INTO "finale" 
(pk_uid,geom)
SELECT k."input_b30k_com5k_italy_pk_uid" AS pk_uid, k.geom
FROM (SELECT "input_b30k_com5k_italy_pk_uid", CastToMultiPolygon(ST_Union("geom")) AS geom
      FROM "b30k_com5k_italy_subd_elem_capolp"
      WHERE "blade_capoluoghi_prov_pk_uid" IS NULL AND ST_Area(geom) &gt; 1 -- pulisco
      group by "input_b30k_com5k_italy_pk_uid") k;
SELECT RecoverGeometryColumn('finale','geom',32632,'MULTIPOLYGON','XY');

Associo i dati necessari

ricostruisco la tabella con i dati necessari

-- associa campo pro_com_t ed altri
DROP TABLE IF EXISTS "aree30cappa";
CREATE TABLE "aree30cappa" AS
SELECT f."pk_uid",c."pro_com_t",c."cod_reg",c."comune",c."abitanti",f."geom"
FROM "finale" f left join "Com01012020_g_WGS84" c USING ("pk_uid");
SELECT RecoverGeometryColumn('aree30cappa','geom',32632,'MULTIPOLYGON','XY');

Cancello le tabelle

il processo necessita di tabelle intermedie non più utili

-- cancella geotabelle inutili
DROP TABLE IF EXISTS "b30k_comuni5k";
DROP TABLE IF EXISTS "capoluoghi_prov";
DROP TABLE IF EXISTS "tmp_italia";
DROP TABLE IF EXISTS "italia_subd";
DROP TABLE IF EXISTS "italia_subd_elem";
DROP TABLE IF EXISTS "b30k_com5k_italy_subd_elem";
DROP TABLE IF EXISTS "b30k_com5k_italy";
DROP TABLE IF EXISTS "b30k_com5k_italy_subd_elem_capolp";
DROP TABLE IF EXISTS "finale";

Aggiorno statistiche del database e avvio ottimizzazione

-- aggiorno statistiche e VACUUM (nel mio vecchio laptop impiega circa 4 minuti)
--
SELECT UpdateLayerStatistics('aree30cappa');
VACUUM;

Riferimenti utili

Canale Telegram, isciviti!!! – https://t.me/pigrecoinfinito

Raduno #ODS2021 Il valore dei dati: (Presentazione di Maurizio Napolitano)

Pubblicità

2 pensieri su “Il decreto ‘di Natale’ sugli spostamenti piccoli comuni

Rispondi

Inserisci i tuoi dati qui sotto o clicca su un'icona per effettuare l'accesso:

Logo di WordPress.com

Stai commentando usando il tuo account WordPress.com. Chiudi sessione /  Modifica )

Foto Twitter

Stai commentando usando il tuo account Twitter. Chiudi sessione /  Modifica )

Foto di Facebook

Stai commentando usando il tuo account Facebook. Chiudi sessione /  Modifica )

Connessione a %s...

Questo sito utilizza Akismet per ridurre lo spam. Scopri come vengono elaborati i dati derivati dai commenti.