* * * *

Privacy Policy

Blog italiano

Clicca qui se vuoi andare al blog italiano su Lazarus e il pascal.

Forum ufficiale

Se non siete riusciti a reperire l'informazione che cercavate nei nostri articoli o sul nostro forum vi consiglio di visitare il
Forum ufficiale di Lazarus in lingua inglese.

Lazarus 1.0

Trascinare un file nel programma
DB concetti fondamentali e ZeosLib
Recuperare codice HTML da pagina web
Mandare mail con Lazarus
Stabilire il sistema operativo
Esempio lista in pascal
File INI
Codice di attivazione
Realizzare programmi multilingua
Lavorare con le directory
Utilizzare Unità esterne
TTreeView
TTreeview e Menu
Generare controlli RUN-TIME
LazReport, PDF ed immagini
Intercettare tasti premuti
Ampliare Lazarus
Lazarus e la crittografia
System Tray con Lazarus
UIB: Unified Interbase
Il file: questo sconosciuto
Conferma di chiusura di un applicazione
Liste e puntatori
Overload di funzioni
Funzioni a parametri variabili
Proprietà
Conversione numerica
TImage su Form e Panel
Indy gestiore server FTP lato Client
PopUpMenu sotto Pulsante (TSpeedButton)
Direttiva $macro
Toolbar
Evidenziare voci TreeView
Visualizzare un file Html esterno
StatusBar - aggirare l'errore variabile duplicata
Da DataSource a Excel
Le permutazioni
Brute force
Indy 10 - Invio email con allegati
La gestione degli errori in Lazarus
Pascal Script
Linux + Zeos + Firebird
Dataset virtuale
Overload di operatori
Lavorare con file in formato JSON con Lazarus
Zeos ... dietro le quinte (prima parte)
Disporre le finestre in un blocco unico (come Delphi)
Aspetto retrò (Cmd Line)
Lazarus 1.0
Come interfacciare periferica twain
Ubuntu - aggiornare free pascal e lazarus
fpcup: installazioni parallele di lazarus e fpc
Free Pascal e Lazarus sul Raspberry Pi
Cifratura: breve guida all'uso dell'algoritmo BlowFish con lazarus e free pascal.
Creare un server multithread
guida all'installazione di fpc trunk da subversion in linux gentoo
Indice
DB concetti fondamentali e connessioni standard
Advanced Record Syntax
DB concetti fondamentali e DBGrid
DB concetti fondamentali e TDBEdit, TDBMemo e TDBText
Advanced Record Syntax: un esempio pratico
Superclasse form base per programmi gestionali (e non)
Superclasse form base per programmi gestionali (e non) #2 - log, exception call stack, application toolbox
Superclasse form base per programmi gestionali (e non) #3 - traduzione delle form
Superclasse form base per programmi gestionali (e non) #4 - wait animation
Un dialog per la connessione al database:TfmSimpleDbConnectionDialog
Installare lazarus su mac osx sierra
immagine docker per lavorare con lazarus e free pascal
TDD o Test-Driven Development
Benvenuto! Effettua l'accesso oppure registrati.
Gennaio 15, 2025, 02:23:25 am

Inserisci il nome utente, la password e la durata della sessione.

91 Visitatori, 0 Utenti

Autore Topic: Usare il WITH - fate attenzione  (Letto 593 volte)

DragoRosso

  • Scrittore
  • Hero Member
  • *****
  • Post: 1453
  • Karma: +44/-0
  • Prima ascoltare, poi decidere
Usare il WITH - fate attenzione
« il: Gennaio 02, 2025, 11:42:43 am »
Mi collego al topic presente sul forum internazionale di Lazarus: https://forum.lazarus.freepascal.org/index.php/topic,69755.msg542298/topicseen.html

Il topic presenta un caso d'uso abbastanza comune tra i programmatori (in realtà io uso il WITH in rari casi), ossia l'uso del WITH come blocco per "accorciare" la scrittura del codice, e magari renderla più chiara (che poi è la funzione del WITH).

Il compilatore di FPC in questo caso presentato in realtà "ignora praticamente" il codice (diciamo che il codice scritto non ha alcun effetto). Questo è uno dei tipici casi in cui purtroppo si vengono a creare quelle situazioni che provocano dei bug clamorosi difficili poi da diagnosticare.

Sul WITH si è dibattuto più e più volte (ho partecipato a più webinar su questo argomento) perchè sembra che quel blocco "abbreviato" non sia molto gradito agli stessi compilatori (sia in Delphi che in FPC, ognuno a vari livelli). Anche se potrebbe sembrare, Delphi non ne è immune anche se è forse un pò più evoluto nel controllo.

Per cui ciò che voglio consigliare a tutti è di fare attenzione con l'uso del WITH, e di verificare più a fondo che ciò che stà all'interno del blocco sia congruo senza fidarsi ciecamente del compilatore.

Questo è un esempio di codice incriminato:

Codice: [Seleziona]
    program Project1;
     
    type
      TTest = record
        V: Integer;
      end;
     
    var
      X: TTest;
     
      function Test: TTest;
      begin
        Result := X;
      end;
     
    var
      T: TTest;
    begin
      with Test do
        V := 1;       //Nessun Errore
      Test.V := 1; //Errore: Argument cannot be assigned to
    end.

P.S.: gli effetti del codice entro il WITH e la linea successiva dovrebbero essere identici in quanto le due scritture sono esattamente equivalenti, ma il codice entro il WITH viene riportato come valido, quando in realtà non lo è.
« Ultima modifica: Gennaio 02, 2025, 11:46:30 am da DragoRosso »
:) Ogni alba è un regalo, ogni tramonto è una conquista :)

nomorelogic

  • Global Moderator
  • Hero Member
  • *****
  • Post: 2941
  • Karma: +20/-4
Re:Usare il WITH - fate attenzione
« Risposta #1 il: Gennaio 02, 2025, 05:19:53 pm »
questo errore del compilatore non lo conoscevo...
certo lascia un po' perplessi  ;D
Imagination is more important than knowledge (A.Einstein)

DragoRosso

  • Scrittore
  • Hero Member
  • *****
  • Post: 1453
  • Karma: +44/-0
  • Prima ascoltare, poi decidere
Re:Usare il WITH - fate attenzione
« Risposta #2 il: Gennaio 02, 2025, 06:24:09 pm »
questo errore del compilatore non lo conoscevo...
certo lascia un po' perplessi  ;D

Non lo sapevo per quello che riguardava FPC, per Delphi in passato c'erano già state delle avvisaglie su alcuni aspetti, non così impattanti ... ma un errore è sempre un errore, se scrivo:

a := 1;

e il compilatore non lo compila (senza errori) può essere grave o no, rimane il fatto che il codice che scrivo non viene "attuato" dal compilatore ed è ciò che è grave.
E su questo FPC e Delphi sono andati a braccetto su alcuni aspetti.
Come ho scritto, Delphi non soffre di questo "bug" ma nei vari webinar mi è sembrato di capire che il WITH non sia molto "digerito" dai compilatori, probabilmente perchè devono eseguire una "inference" sui dati e ciò sembra sia un pò ostico.

Ho sempre visto il WITH come un potenziale pericolo e ciò che è successo me lo ha semplicemente confermato.

Faccio un esempio banale, cosa che alcune volte (rare ..) mi è capitato di fare:

se uso il WITH con un oggetto che ha la proprietà ROW ad esempio ed uso il ROW entro il blocco WITH questo mi viene correttamente "agganciato" all'oggetto stesso.
Ma se per uno spiacevole equivoco (CHE NON DOVREBBE MAI ACCADERE) la proprietà ROW sparisse dall'oggetto (anche nell'immediato futuro), allora quel ROW verrebbe preso dal compilatore come una variabile normale e quindi potrebbe "agganciarla" ad una variabile globale ad esempio.

Codice normale:
Codice: [Seleziona]
type
  TProva = record
      ROW: integer;
  end;

var Prova: TProva;

begin
    WITH Prova do
       ROW := 5;
end;

Codice futuro .... bacato:
Codice: [Seleziona]
//In una UNIT molto lontana e dimenticata una variabile globale ROW venne dichiarata ....
var ROW: integer;
.....
.....
type
  TProva = record
      ROWS: integer;   //cambio del nome..... DA NON FARE MAI DOPO AVER PUBBLICATO IL CODICE !!!!
  end;

var Prova: TProva;

begin
    WITH Prova do
       ROW := 5;         //NON E' il ROW del codice precedente ... vai a scoprirlo poi .....
                                // .... non ti bastano gli inferni disponibili di una vita da quante bestemmie tiri
end;
« Ultima modifica: Gennaio 02, 2025, 06:26:35 pm da DragoRosso »
:) Ogni alba è un regalo, ogni tramonto è una conquista :)

xinyiman

  • Administrator
  • Hero Member
  • *****
  • Post: 3293
  • Karma: +12/-0
Re:Usare il WITH - fate attenzione
« Risposta #3 il: Gennaio 03, 2025, 08:56:14 am »
Ma daiiiii, che comportamento anomalo (per non dire altro). Fortuna non lo uso da una vita.
Ieri è passato, domani è futuro, oggi è un dono...

DragoRosso

  • Scrittore
  • Hero Member
  • *****
  • Post: 1453
  • Karma: +44/-0
  • Prima ascoltare, poi decidere
Re:Usare il WITH - fate attenzione
« Risposta #4 il: Gennaio 03, 2025, 12:45:27 pm »
La discussione in quel topic si è prolungata nel frattempo.

E sembra che il bug (perchè è un bug e non ci sono ne santi ne madonne che tengano) non venga ritenuto un bug in quanto comunque il codice scritto è errato e quindi non intendono sprecare energie con un intervento.

Uno dei più titolati del forum se l'è un pò presa (sempre civilmente) perchè ho accennato che Delphi rileva ciò come un errore  :-X e perchè ritengo (ma avevo premesso che sono l'ultimo a cui dare credito) che il compilatore debba segnalare ciò che non è corretto nel codice.

Devo dire che quanto ho visto non mi è piaciuto per niente, perchè mi pare si punti più a difendere l'orticello piuttosto che vederla a campo largo, e se ciò potrebbe andare bene in un forum come il nostro, in un forum di riferimento come quello internazionale ...

Io rimango dell'idea che il compilatore DEVE segnalare codice errato, sempre e comunque. Non può essere che io scriva un schifezza e che il compilatore me la passi per buona, indifferentemente se ciò ha un peso o meno.
 
Ciao
:) Ogni alba è un regalo, ogni tramonto è una conquista :)

nomorelogic

  • Global Moderator
  • Hero Member
  • *****
  • Post: 2941
  • Karma: +20/-4
Re:Usare il WITH - fate attenzione
« Risposta #5 il: Gennaio 03, 2025, 03:40:07 pm »
comunque nel thread in inglese c'è uno che ha criticato chi ha criticato te  ;D

sul quel caso specifico di WITH c'è poco da obiettare: quel codice è errato ed il compilatore dovrebbe segnalarlo
che poi "romperebbe" codice pre-esistente può anche darsi, ma sarebbe un bene!
soprattutto in un linguaggio che fa del type-safe una bandiera

per altri usi WITH mi piace molto ma lo uso sempre con cautela
per esempio lo evito volentieri quando ci sono parametri di una funzione che si chiamano come proprietà di qualcosa
tipo

Codice: [Seleziona]
function saluta(const nome: string): string;
type
   TEsempio = record
      nome: string;
      saluto: string;
   end;

var e: TEsempio;
begin

   with e do begin
      saluto := 'ciao';
      nome := nome;  // chissà cosa verrà assegnato qua...
      result := saluto + ' ' + nome;  // chissà cosa verrà concatenato qua...
   end;

end;





Imagination is more important than knowledge (A.Einstein)

DragoRosso

  • Scrittore
  • Hero Member
  • *****
  • Post: 1453
  • Karma: +44/-0
  • Prima ascoltare, poi decidere
Re:Usare il WITH - fate attenzione
« Risposta #6 il: Gennaio 06, 2025, 02:48:22 pm »
Ultimo aggiornamento, ultimo in tutti i sensi.

Sul forum internazionale dalla 1 pagina e mezza di discussione siamo intanto passati alle 6 pagine, e dentro ci hanno messo di tutto, scoprendo che c'è gente che scrive una applicazione intera DENTRO il WITH  :o ::)

Sembra che per il compilatore il WITH sia un mondo a se stante, dove è possibile creare variabili inline (chiamate "shadow" o "masked" in fpc) simil DELPHI (mio pensiero).

Certo che se ne scoprono una dietro l'altra e non sempre sono scoperte positive ... io rimango convinto che il compilatore debba fare ciò che le regole gli impongono perchè questo è quello che si aspetta lo sviluppatore ... che si possano dichiarare variabili a casaccio senza definizione in PASCAL non l'avevo mai sentita. Comunque tant'è, ho smesso di seguire quella discussione perchè stà diventando surreale, e preferisco rimanere nell'ignoranza ... che è meglio.

Questo è ciò che fà Delphi (dalla versione 10.3 RIO di qualche anno fà), ma anche se assolutamente discutibile dal punto di vista del PASCAL, è sicuramente più chiaro e "corretto" rispetto a quello indicato da quella discussione.

Codice: [Seleziona]
program Project1;
     
    type
      TTest = record
        V: Integer;
      end;
     
    var
      X: TTest;
     
      function Test: TTest;
      begin
        Result := X;
      end;
     
    var
      T: TTest;
    begin
      with Test do
         begin
           //V := 1;       //Nessun Errore, scrittura valida in FPC attualmente, ma dalla valenza discutibile (mio pensiero)
           var V := 1;       <-----   QUESTA E' UNA DICHIARAZIONE DI VARIABILE INLINE DI DELPHI, CON SCOPO RIDOTTO AL BLOCCO WITH
         end;
      Test.V := 1; //Errore: Argument cannot be assigned to
    end.

Generalmente nei nuovi programmi di Delphi, chi dovrà convertirli si troverà a che fare con questa nuova sintassi tipica di Delphi (e per molti assolutamente scorretta dal punto di vista del linguaggio) che secondo me và molto bene e migliora la leggibilità, la comprensione, la funzionalità e soprattutto evita alcune problematiche tipiche delle forme invece standard:

Codice: [Seleziona]
 for var i:= 0 to 10 do
  begin
    //qui la variabile "i" esiste ed è l'unico blocco in cui esiste (si dice variabile di scopo)
  end;
  //Da qui in poi (così come prima del "for") la variabile "i" non esiste più. Ne posso creare altre di var "i", tutte però con esistenza ed uno "scopo" ben precisi.
  //Ovviamente la variabile "i" non può essere definita come variabile locale.
 
« Ultima modifica: Gennaio 06, 2025, 02:53:12 pm da DragoRosso »
:) Ogni alba è un regalo, ogni tramonto è una conquista :)

 

Recenti

How To

Utenti
  • Utenti in totale: 803
  • Latest: FGB
Stats
  • Post in totale: 19361
  • Topic in totale: 2313
  • Online Today: 118
  • Online Ever: 900
  • (Gennaio 21, 2020, 08:17:49 pm)
Utenti Online
Users: 0
Guests: 91
Total: 91

Disclaimer:

Questo blog non rappresenta una testata giornalistica poiché viene aggiornato senza alcuna periodicità. Non può pertanto considerarsi un prodotto editoriale ai sensi della legge n. 62/2001.