Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Deep Tech Revolution: così Area Science Park apre i laboratori alle startup
Deep Tech Revolution: così Area Science Park apre i laboratori alle startup
Siamo tornati nel parco tecnologico di Trieste per il kick-off del programma che mette a disposizione di cinque startup le infrastrutture di ricerca, dal sincrotrone Elettra ai laboratori di genomica e HPC. Roberto Pillon racconta il modello e la visione
HP OMEN MAX 16 con RTX 5080: potenza da desktop replacement a prezzo competitivo
HP OMEN MAX 16 con RTX 5080: potenza da desktop replacement a prezzo competitivo
HP OMEN MAX 16-ak0001nl combina RTX 5080 Laptop e Ryzen AI 9 HX 375 in un desktop replacement potente e ben raffreddato, con display 240 Hz e dotazione completa. Autonomia limitata e calibrazione non perfetta frenano l'entusiasmo, ma a 2.609 euro è tra le proposte più interessanti della categoria.
Recensione Google Pixel 10a, si migliora poco ma è sempre un'ottima scelta
Recensione Google Pixel 10a, si migliora poco ma è sempre un'ottima scelta
Google ha appena rinnovato la sua celebre serie A con il Pixel 10a, lo smartphone della serie più conveniente se consideriamo il rapporto tra costo e prestazioni. Con il chip Tensor G4, un design raffinato soprattutto sul retro e l'integrazione profonda di Gemini, il colosso di Mountain View promette un'esperienza premium a un prezzo accessibile. E il retro non ha nessuno scalino
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 03-11-2009, 02:16   #1
malocchio
Senior Member
 
L'Avatar di malocchio
 
Iscritto dal: Feb 2007
Città: Verona
Messaggi: 1060
[Java] Design degli eventi

Ho scritto a scopo didattico un'applicazione desktop per il tris, human vs. cpu .

Ho fatto un disegno classi che definisco decente, forse troppo elaborato, ma a scopo didattico.
In sintesi, così formata:

alcune enum
Codice:
public enum Player {
    HUMAN("X"),
    CPU("O"),
    NONE(" ");

    Player(String s) {
        this.s = s;
    }

    private String s;

    @Override
    public String toString() {
        return s;
    }
}
Codice:
public enum TrisState {
    HUMAN_WON("Human player won!"),
    CPU_WON("Cpu player won!"),
    DRAW("Draw!"),
    INCOMLETE("Your turn...");

    TrisState (String label) {
        this.label = label;
    }

    private String label;

    @Override
    public String toString() {
        return this.label;
    }
}
Un modello che rappresenta la griglia, che gestisce l'inserimento dei simboli, il controllo sui turni, l'update dell'interfaccia grafica, il controllo dei possibili risultati raggiunti (vinto, pari, continua...) e, guardacaso, la notifica tramite eventi di eventuali ascoltatori (su cui baserò le prossime classi da scrivere). In sintesi:
Codice:
public class TrisGridModel extends AbstractTableModel {

    private Player[][] grid;

    public TrisGridModel() {...}

    public TrisGridModel(Player[][] grid) {...}

    public int getRowCount() {...}

    public int getColumnCount() {...}

    public Player getValueAt(int r, int c) {...}

    @Override
    public void setValueAt(Object aValue, int r, int c) {...}

    public TrisState getState() {...}

    public boolean complete() {...}

    public Player tris() {...}

    private void switchNow() {...}

    //Ecco la gestione degli eventi

    private List<TrisGameListener> listeners;

    {
        this.listeners = new ArrayList<TrisGameListener>();
    }

    public void addTrisGameListener (TrisGameListener tgl) {...}

    public boolean removeTrisGameListener (TrisGameListener tgl) {...}
    
    private void fireTrisGameEvent (TrisGameEvent tge) {
        //TODO notifica dell'evento
        ...
    }
}
Questa è l'interfaccia TrisGameListener, che è scritta così ma potrà prevedere altri tipi di eventi.
Codice:
public interface TrisGameListener extends java.util.EventListener {

    public void movePerformed (TrisGameEvent tge);
    public void stateChanged (TrisGameEvent tge);
    
}
I miei dubbi:
vorrei progettare una gestione degli eventi in perfetto stile Swing, con eventi, ascoltatori e oggetti che generano eventi. Però non ho capito bene come si deve definire la classe che descrive un evento. Nel caso sopra (TrisGameListener) ci sono due possibili tipi di evento:
- un giocatore inserisce un simbolo (movePerformed(...))
- la griglia cambia di stato (vedi TrisGameState) (stateChanged(...))
Cosa devo scrivere nella classe TrisGameEvent? I metodi dell'ascoltatore che parametri devono prendere in ingresso?

L'interfaccia grafica comunica con la griglia (TrisGameModel) tramite gli eventi caratteristici della classe TableModel, infatti, nella classe della gui:
Codice:
public class TrisPanel extends JPanel {

    private JButton[][] grid;
    private TrisGridModel model;

    public TrisPanel () {...}

    public TrisPanel(TrisGridModel model) {
        this.model = model;
        this.model.addTableModelListener( new TrisModelListener() );
        ... 
    }

    public class TrisModelListener implements TableModelListener {

        public void tableChanged(TableModelEvent e) {...}

    }

}
Vorrei aggiungere anche la funzionalità "Hai vinto!" o "Hai perso!" o "Pari" proprio tramite eventi TrisGameEvent.

Altro dubbio: se ho un oggetto Swing con alcuni ascoltatori registrati ad esso, una volta che genera un evento e attiva i metodi degli ascoltatori, qual'è il thread in esecuzione? Posso agire su oggetti Swing direttamente oppure devo usare l'invokeLater() ??
__________________
malocchio è offline   Rispondi citando il messaggio o parte di esso
Old 03-11-2009, 11:28   #2
banryu79
Senior Member
 
L'Avatar di banryu79
 
Iscritto dal: Oct 2007
Città: Padova
Messaggi: 4131
Quote:
I miei dubbi:
vorrei progettare una gestione degli eventi in perfetto stile Swing, con eventi, ascoltatori e oggetti che generano eventi. Però non ho capito bene come si deve definire la classe che descrive un evento. Nel caso sopra (TrisGameListener) ci sono due possibili tipi di evento:
- un giocatore inserisce un simbolo (movePerformed(...))
- la griglia cambia di stato (vedi TrisGameState) (stateChanged(...))
Cosa devo scrivere nella classe TrisGameEvent?
In TrisGameEvent ci metterai le cose che possono interessare a chi ascolta i due eventi.
Ad esmepio, il listener su cui viene invocato il metodo statChanged() sarà interessato a recuperare quello stato, per esaminarlo, percui il tuo TrisGameEvent potrebbe avere un membro di tipo TrisState e un metodo getter.

Nel caso di movePerformed() non so cosa possa interessare: se movePerformed rappresenta l'evento che accade quando un giocatore inserisce un simbolo potrebbe essere utile sapere quale istanza Palyer della griglia è cambiata e che valore ha.
Il valore lo si ricava dall'istanza Player stessa (codifica una String e ha ridefinito un toString() che la restituisce, e questa stringa rappresenta il suo valore).
L'istanza Player cambiata è identificata dalla sua posizione nella griglia, ovvero dai due indici che la identificano nell'array bidimensionale del TrisGridModel: perciò potrebbe essere utile, per l'ascoltatore, poter chiedere a TrisGameEvent la posizione dell'istanza Player cambiata.

Quote:
I metodi dell'ascoltatore che parametri devono prendere in ingresso?
Che io sappia i metodi che definiscono un listener in ingresso prendono solo un parametro: l'evento che ascoltano, cioè TrisGameEvent, nel tuo caso.


Quote:
Altro dubbio: se ho un oggetto Swing con alcuni ascoltatori registrati ad esso, una volta che genera un evento e attiva i metodi degli ascoltatori, qual'è il thread in esecuzione? Posso agire su oggetti Swing direttamente oppure devo usare l'invokeLater() ??
Se l'ascoltatore è stato registrato su una sorgente Swing, il thread che esegue il metodo invocato dell'ascoltatore è l'EDT.
Puoi vedere quale thread stia eseguendo un certo pezzo di codice chiamando Thread.currentThread() che torna una reference al thread corrente
__________________

As long as you are basically literate in programming, you should be able to express any logical relationship you understand.
If you don’t understand a logical relationship, you can use the attempt to program it as a means to learn about it.
(Chris Crawford)

Ultima modifica di banryu79 : 03-11-2009 alle 12:30.
banryu79 è offline   Rispondi citando il messaggio o parte di esso
Old 08-11-2009, 16:55   #3
malocchio
Senior Member
 
L'Avatar di malocchio
 
Iscritto dal: Feb 2007
Città: Verona
Messaggi: 1060
Quote:
Originariamente inviato da banryu79 Guarda i messaggi
In TrisGameEvent ci metterai le cose che possono interessare a chi ascolta i due eventi.
Ad esmepio, il listener su cui viene invocato il metodo statChanged() sarà interessato a recuperare quello stato, per esaminarlo, percui il tuo TrisGameEvent potrebbe avere un membro di tipo TrisState e un metodo getter.

Nel caso di movePerformed() non so cosa possa interessare: se movePerformed rappresenta l'evento che accade quando un giocatore inserisce un simbolo potrebbe essere utile sapere quale istanza Palyer della griglia è cambiata e che valore ha.
Il valore lo si ricava dall'istanza Player stessa (codifica una String e ha ridefinito un toString() che la restituisce, e questa stringa rappresenta il suo valore).
L'istanza Player cambiata è identificata dalla sua posizione nella griglia, ovvero dai due indici che la identificano nell'array bidimensionale del TrisGridModel: perciò potrebbe essere utile, per l'ascoltatore, poter chiedere a TrisGameEvent la posizione dell'istanza Player cambiata.


Che io sappia i metodi che definiscono un listener in ingresso prendono solo un parametro: l'evento che ascoltano, cioè TrisGameEvent, nel tuo caso.



Se l'ascoltatore è stato registrato su una sorgente Swing, il thread che esegue il metodo invocato dell'ascoltatore è l'EDT.
Puoi vedere quale thread stia eseguendo un certo pezzo di codice chiamando Thread.currentThread() che torna una reference al thread corrente
Intanto grazie della risposta, chiara come sempre.
Scusami se non ho risposto subito, ho avuto da fare, ma il 3d rimane sempre aperto.

Ho inserito le modifiche nel codice sorgente e questo è quello che ho ottenuto:

Classe TrisGameEvent
Codice:
public class TrisGameEvent extends java.awt.AWTEvent {
    
    public static final int MOVE_PERFORMED = RESERVED_ID_MAX + 1;
    public static final int STATE_CHANGED = RESERVED_ID_MAX + 2;

    private int row;
    private int column;
    private Player player;
    private TrisState state;

    public TrisGameEvent(Object source, int type, int row, int column) {
        super(source, type);
        this.row = row;
        this.column = column;
    }

    public TrisGameEvent (Object source, int type, int row, int column, TrisState state) {
        this(source, type, row, column);
        this.state = state;
    }

    public TrisGameEvent(Object source, int type, int row, int column, Player player) {
        this(source, type, row, column);
        this.player = player;
    }

    public int getRow() {
        return row;
    }

    public int getColumn() {
        return column;
    }

    public TrisState getState() {
        return state;
    }

}
Classe TrisGridModel
Codice:
public class TrisGridModel extends AbstractTableModel {

    private Player[][] grid;
    private Player now;
    private TrisState prevState;

    public TrisGridModel() {...}

    public TrisGridModel(Player[][] grid) {...}

    public int getRowCount() {...}

    public int getColumnCount() {...}

    public Player getValueAt(int r, int c) {
        return this.grid[r][c];
    }

    public void setValueAt(Object aValue, int r, int c) {
        if (this.now == Player.NONE) return;
        if (r < this.getRowCount() && c < this.getColumnCount() && aValue instanceof Player) {
            if ( (Player)aValue == this.now ) {
                this.grid[r][c] = this.now;
                this.switchNow();
                this.fireMovePerformed(r, c, (Player) aValue);
            }
            
        } //FIXME else lancia una BRUTTA eccezione
    }

    public TrisState getState() {...}

    public boolean complete() {...}

    public Player tris() {...}

    private TrisGameListener listener;

    public void addTrisGameListener (TrisGameListener tgl) {
        listener = TrisGameEventMulticaster.add(listener, tgl);
    }

    public void removeTrisGameListener (TrisGameListener tgl) {
        listener = TrisGameEventMulticaster.remove(listener, tgl);
    }

    private void fireMovePerformed(int row, int column, Player player) {
        this.listener.movePerformed( new TrisGameEvent(this, TrisGameEvent.MOVE_PERFORMED, row, column, player) );
        this.fireTableCellUpdated(row, column);
        if (this.getState() != this.prevState) {
            this.prevState = this.getState();
            this.fireStateChanged(row, column, prevState);
        }
    }

    private void fireStateChanged(int row, int column, TrisState state) {
        this.listener.stateChanged( new TrisGameEvent(this, TrisGameEvent.STATE_CHANGED, row, column, state) );
    }
}
Ti sembra corretto?
__________________
malocchio è offline   Rispondi citando il messaggio o parte di esso
Old 09-11-2009, 09:49   #4
banryu79
Senior Member
 
L'Avatar di banryu79
 
Iscritto dal: Oct 2007
Città: Padova
Messaggi: 4131
Beh, mi pare di sì.
A te la cosa funziona?
__________________

As long as you are basically literate in programming, you should be able to express any logical relationship you understand.
If you don’t understand a logical relationship, you can use the attempt to program it as a means to learn about it.
(Chris Crawford)
banryu79 è offline   Rispondi citando il messaggio o parte di esso
Old 09-11-2009, 12:06   #5
malocchio
Senior Member
 
L'Avatar di malocchio
 
Iscritto dal: Feb 2007
Città: Verona
Messaggi: 1060
Quote:
Originariamente inviato da banryu79 Guarda i messaggi
Beh, mi pare di sì.
A te la cosa funziona?
Devo ancora utilizzarlo
Mi interessava la correttezza "logica"
__________________
malocchio è offline   Rispondi citando il messaggio o parte di esso
Old 09-11-2009, 12:45   #6
banryu79
Senior Member
 
L'Avatar di banryu79
 
Iscritto dal: Oct 2007
Città: Padova
Messaggi: 4131
Quote:
Originariamente inviato da malocchio Guarda i messaggi
Devo ancora utilizzarlo
Mi interessava la correttezza "logica"
Bene, allora prendi tutto quello che ti ho scritto, che è il mio ragionamento in base alle mie conscenze (e ti avverto: sono molto lacunose in realtà, Swing lo sto ancora studiando e so poco) e leggile con occhio critico.

Se quello che cercavi era un cosiglio per implementare "in modo idiomatico" il tuo modello ad eventi allora non dare retta a quello che ti ho detto: ne so quanto te
__________________

As long as you are basically literate in programming, you should be able to express any logical relationship you understand.
If you don’t understand a logical relationship, you can use the attempt to program it as a means to learn about it.
(Chris Crawford)
banryu79 è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Deep Tech Revolution: così Area Science Park apre i laboratori alle startup Deep Tech Revolution: così Area Science P...
HP OMEN MAX 16 con RTX 5080: potenza da desktop replacement a prezzo competitivo HP OMEN MAX 16 con RTX 5080: potenza da desktop ...
Recensione Google Pixel 10a, si migliora poco ma è sempre un'ottima scelta Recensione Google Pixel 10a, si migliora poco ma...
6G, da rete che trasporta dati a rete intelligente: Qualcomm accelera al MWC 2026 6G, da rete che trasporta dati a rete intelligen...
CHUWI CoreBook Air alla prova: design premium, buona autonomia e qualche compromesso CHUWI CoreBook Air alla prova: design premium, b...
Il nuovo MacBook Neo ha una memoria SSD ...
Xbox Project Helix, le prime specifiche ...
Annunci pubblicitari sulla TV quando cam...
Prezzi aumentati del 50% durante la nott...
Sconti studiati per singolo utente: Sony...
Addio alla Kia Niro EV, il crossover sar...
Apple crede nel suo iPhone Fold: la prod...
Fortnite, un nuovo listino per i pacchet...
Ecco i nuovi Sonos Play ed Era 100 SL: d...
Razer svela il futuro del gaming potenzi...
Tre robot Narwal in offerta: pulizia aut...
Gracenote denuncia OpenAI: ChatGPT addes...
Microsoft AI Tour Milano: dall'efficienz...
Asus ExpertBook Ultra: Intel Core Ultra ...
Intel presenta i processori desktop Core...
Chromium
GPU-Z
OCCT
LibreOffice Portable
Opera One Portable
Opera One 106
CCleaner Portable
CCleaner Standard
Cpu-Z
Driver NVIDIA GeForce 546.65 WHQL
SmartFTP
Trillian
Google Chrome Portable
Google Chrome 120
VirtualBox
Tutti gli articoli Tutte le news Tutti i download

Strumenti

Regole
Non Puoi aprire nuove discussioni
Non Puoi rispondere ai messaggi
Non Puoi allegare file
Non Puoi modificare i tuoi messaggi

Il codice vB è On
Le Faccine sono On
Il codice [IMG] è On
Il codice HTML è Off
Vai al Forum


Tutti gli orari sono GMT +1. Ora sono le: 01:48.


Powered by vBulletin® Version 3.6.4
Copyright ©2000 - 2026, Jelsoft Enterprises Ltd.
Served by www3v