DZone
Thanks for visiting DZone today,
Edit Profile
  • Manage Email Subscriptions
  • How to Post to DZone
  • Article Submission Guidelines
Sign Out View Profile
  • Post an Article
  • Manage My Drafts
Over 2 million developers have joined DZone.
Log In / Join
Refcards Trend Reports
Events Video Library
Refcards
Trend Reports

Events

View Events Video Library

Zones

Culture and Methodologies Agile Career Development Methodologies Team Management
Data Engineering AI/ML Big Data Data Databases IoT
Software Design and Architecture Cloud Architecture Containers Integration Microservices Performance Security
Coding Frameworks Java JavaScript Languages Tools
Testing, Deployment, and Maintenance Deployment DevOps and CI/CD Maintenance Monitoring and Observability Testing, Tools, and Frameworks
Culture and Methodologies
Agile Career Development Methodologies Team Management
Data Engineering
AI/ML Big Data Data Databases IoT
Software Design and Architecture
Cloud Architecture Containers Integration Microservices Performance Security
Coding
Frameworks Java JavaScript Languages Tools
Testing, Deployment, and Maintenance
Deployment DevOps and CI/CD Maintenance Monitoring and Observability Testing, Tools, and Frameworks

Curious about the future of data-driven systems? Join our Data Engineering roundtable and learn how to build scalable data platforms.

Data Engineering: The industry has come a long way from organizing unstructured data to adopting today's modern data pipelines. See how.

Threat Detection: Learn core practices for managing security risks and vulnerabilities in your organization — don't regret those threats!

Managing API integrations: Assess your use case and needs — plus learn patterns for the design, build, and maintenance of your integrations.

Related

  • Design Pattern: What You Need to Know to Improve Your Code Effectively

Trending

  • Designing Scalable and Secure Cloud-Native Architectures: Technical Strategies and Best Practices
  • Storybook: A Developer’s Secret Weapon
  • Smart Routing Using AI for Efficient Logistics and Green Solutions
  • Automate Web Portal Deployment in Minutes Using GitHub Actions

Observer Pattern Applied: Solving Sudoku, Easy Ones

By 
shivaji chelladurai user avatar
shivaji chelladurai
·
Dec. 03, 07 · Code Snippet
Likes (0)
Comment
Save
Tweet
Share
1.7K Views

Join the DZone community and get the full member experience.

Join For Free
We solve Sudoku by following certain routines in our mind. Thus we have to apply handful of patterns to solve each of these routines. I am taking just one of these patterns good enough for solving easy ones and delve in to detail. We all know, we will be given a Sudoku 9x9 array with few cells filled with values 1 to 9. The placement and values of numbers guarantees a unique solution. The puzzle is solved by placing appropriate values in the remaining cells such that the numbers are not repeated vertically, horizontally or within the 9 minor 3x3 arrays.

Routine followed
To start with, each cell can have any value between 1 and 9, inclusive, with equal probability. When each of the given numbers is placed in the corresponding cell, the adjacent cells (horizontally, vertically and within the minor array) loose the opportunity to have this number. This is the crux of the problem that we are discussing. Let’s implement a class to define a cell that holds all possible entries. 


public class Cell {
    private List values = new ArrayList();
    private int row;
    private int col;
    
    // Add all possible entries (1 to 9) in each cell
    public Cell(int row, int col) {
        this.row = row;
        this.col = col;
        for (int n = 1; n <= 9; n++) {
            values.add(new Integer(n));
        }
    }
    …
}

 
Placing a number in a cell reduces the array list to have just that number and remove the corresponding entry in the adjacent cells (8 vertically, 8 horizontally and 4 remaining cells in the minor array). As we place more numbers the size of array list in each of the cells gets reduced.

Pattern applied
Having said the routine that we want to follow in order to reduce the possibilities; how can we trigger the adjacent cells such that they respond to placing a number in a particular cell? The answer is to apply observer pattern. Here the observers are adjacent cells. See below code that has the criteria to identify the adjacent cells:


        for (int i = 0; i < 9; i++) {
            for (int j = 0; j < 9; j++) {
                boolean isSame = (i == row) && (j == col);
                boolean isSameLine = (i == row) || (j == col);
                boolean isMinor = (i/3 == row/3) && (j/3 == col/3);

                if ( !isSame && (isSameLine || isMinor)) { 
                    // logic to add observers goes here
                     …
                }
            }
        }


We need to re-look our definition of a cell as each cell is an observable to the adjacent cells and at the same time an observer of the adjacent cell. Refer below redefined cell class. The super class, observable, has the implementation to notify observers (here it refers to adjacent cells). Update method is the implementation of Observer interface that is called placing a value in a cell. Here we want to remove an entry from the array list. 


public class Cell extends Observable implements Observer {
   …

   // add the known value … and notify observers
    public void setValue(int value) {
        …
        super.notifyObservers(new Integer(value));
    } 

    // Observe and remove the entry set in the observable
    public void update(Observable o, Object arg) {
        values.remove(arg); 
        … 
    }
}


Class definitions
Let us put together the definition of cell class below: 


package my.apps.sudoku;

import java.util.Observable;
import java.util.Observer;
import java.util.ArrayList;
import java.util.List; 

public class Cell extends Observable implements Observer {
    private List values = new ArrayList();
    private boolean isSolved = false; 
    private int row;
    private int col;

    // Add all possible entries (1 to 9) in each cell
    public Cell(int row, int col) {
        this.row = row;
        this.col = col;
        for (int n = 1; n <= 9; n++) {
            values.add(new Integer(n));
        }
    }

    // add cells that are in the same line or same box as observers
    public synchronized void addObserver(Cell[][] cells) {
        for (int i = 0; i < 9; i++) {
            for (int j = 0; j < 9; j++) {
                boolean isSame = (i == row) && (j == col);
                boolean isSameLine = (i == row) || (j == col);
                boolean isSecondary = (i/3 == row/3) && (j/3 == col/3);
                if ( !isSame && (isSameLine || isSecondary)) {
                    super.addObserver(cells[i][j]);
                }
            }
        }
    } 

	// add the known value after clearing and notify observers
    public void setValue(int value) {
        values.clear();
        values.add(new Integer(value));
        isSolved = true;
        super.setChanged();
        super.notifyObservers(new Integer(value));
    }

    // Observe and remove the entry set in the observable
    public void update(Observable o, Object arg) {
        values.remove(arg);
        if (!isSolved && values.size() == 1) {
            Integer value = (Integer)values.get(0);
            setValue(value.intValue());
        }
    }

    // A cell is solved if the it has just one value
    public int getValue() {
        if (values.size() == 1) {
            return ((Integer)values.get(0)).intValue();
        }
        return 0;
    }
}

 
Below is the code for creating a collection of cell and setting up the values. It also demonstrates with a sample: 


package my.apps.sudoku;

import java.util.Observable;
import java.util.Observer;
import java.util.ArrayList;
import java.util.List;

public class Sudoku {
    private Cell [][] cells = new Cell[9][9];

    public Sudoku() {
        // initialize the cell
        for (int i = 0; i < 9; i++) {
            for (int j = 0; j < 9; j++) {
                cells[i][j] = new Cell(i, j);
            }
        } 

        // add observers
        for (int i = 0; i < 9; i++) {
            for (int j = 0; j < 9; j++) {
                cells[i][j].addObserver(cells);
            }
        }
    }
    
    // set known values
    public void setup(int [][] puzzle) {
        for (int i = 0; i < 9; i++) {
            for (int j = 0; j < 9; j++) {
                if (puzzle[i][j] != 0) {
                    cells[i][j].setValue(puzzle[i][j]);
                }
            }
        }
    }
    
    public int getCellValue(int i, int j) {
        return cells[i][j].getValue();
    }
    
    public static void main(String [] args) {
        int [][] puzzle = {
                {0, 6, 2,    3, 0, 0,    9, 0, 0},
                {0, 0, 0,    0, 8, 0,    0, 0, 7},
                {8, 0, 0,    0, 0, 5,    0, 0, 4}, 

                {0, 0, 5,    0, 2, 0,    0, 0, 6},
                {0, 3, 0,    9, 4, 6,    0, 5, 0},
                {4, 0, 0,    0, 5, 0,    7, 0, 0}, 

                {7, 0, 0,    6, 0, 0,    0, 0, 3},
                {5, 0, 0,    0, 3, 0,    0, 0, 0},
                {0, 0, 3,    0, 0, 7,    8, 9, 0}
                        };

        Sudoku sudoku = new Sudoku();
        sudoku.setup(puzzle);
        for (int i = 0; i < 9; i++) {
            for (int j = 0; j < 9; j++) {
                System.out.print(sudoku.getCellValue(i, j) + "|");
            }
            System.out.println();
        }
    }
}

Observer pattern

Opinions expressed by DZone contributors are their own.

Related

  • Design Pattern: What You Need to Know to Improve Your Code Effectively

Partner Resources


Comments

The likes didn't load as expected. Please refresh the page and try again.

ABOUT US

  • About DZone
  • Support and feedback
  • Community research
  • Sitemap

ADVERTISE

  • Advertise with DZone

CONTRIBUTE ON DZONE

  • Article Submission Guidelines
  • Become a Contributor
  • Core Program
  • Visit the Writers' Zone

LEGAL

  • Terms of Service
  • Privacy Policy

CONTACT US

  • 3343 Perimeter Hill Drive
  • Suite 100
  • Nashville, TN 37211
  • support@dzone.com

Let's be friends: