Java: Example - Roman Numerals

This is a decimal to roman numeral conversion program. It accepts decimal numbers in the range 1 to 3999. The Romans didn't build their empire with only this small range of numbers, but this restricted range is used because their extended notation doesn't translate well into ASCII characters, for example, bars over numbers (meaning multiply by 1000) and a backward C were common, but many other symbols have been used. And contrary to popular opinion, they did have zero, written as N (nullus or nihil) or as a dash. And there were variations, so you might find other possibilities. The advantage of the hindu-arabic number system that we now use is the positional power notation, which makes computation vastly easier.

This program illustrates a few things:

User interface

  1 
  2 
  3 
  4 
  5 
  6 
  7 
  8 
  9 
 10 
 11 
 12 
 13 
 14 
 15 
 16 
 17 
 18 
 19 
 20 
 21 
 22 
 23 
 24 
 25 
 26 
 27 
 28 
 29 
 30 
 31 
 32 
 33 
 34 
 35 
 36 
 37 
 38 
 39 
 40 
 41 
 42 
 43 
 44 
 45 
 46 
 47 
 48 
 49 
 50 
 51 
 52 
 53 
 54 
 55 
 56 
 57 
 58 
 59 
// File   : gui/componenents/calculators/Dec2Rom.java
// Description: Converts decimal to roman numerals.
// Shows  : Simple GUI, anonymous inner class listener.
//          Separate logic.
// Author : Fred Swartz - 2006-12-29 - Placed in public domain.

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.event.*;

//////////////////////////////////////////////////////////////// Dec2Rom
public class Dec2Rom extends JFrame {
    
    //=============================================== instance variables
    private JTextField _decimalTf = new JTextField(6);
    private JTextField _romanTf   = new JTextField(8);
    
    //============================================================= main
    public static void main(String[] args) {
        Dec2Rom win = new Dec2Rom();
    }
    
    //====================================================== constructor
    public Dec2Rom() {
        JButton convertButton = new JButton("Convert");
        
        //... Create a listener with anonymous inner class syntax.
        ActionListener doIt = new ActionListener() {
            public void actionPerformed(ActionEvent e) {
                try {
                    int val = Integer.parseInt(_decimalTf.getText());
                    String roman = Roman.int2roman(val);
                    _romanTf.setText("" + roman);
                } catch (NumberFormatException ex) {
                    _romanTf.setText("Error");
                }
            }};
        
        //... Add this same listener to two components.
        convertButton.addActionListener(doIt);
        _decimalTf.addActionListener(doIt);
        
        //... Create panel, set layout, add components.
        JPanel content = new JPanel();
        content.setLayout(new FlowLayout());
        content.add(_decimalTf); 
        content.add(convertButton);
        content.add(_romanTf); 
        
        //... Set characteristics of this window.
        this.setContentPane(content);
        this.setTitle("Decimal to Roman");
        this.pack();                       // Do layout.
        this.setLocationRelativeTo(null);  // Center window.
        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        this.setVisible(true);
    }
}

Model / logic

  1 
  2 
  3 
  4 
  5 
  6 
  7 
  8 
  9 
 10 
 11 
 12 
 13 
 14 
 15 
 16 
 17 
 18 
 19 
 20 
 21 
 22 
 23 
 24 
 25 
 26 
 27 
 28 
 29 
 30 
 31 
 32 
 33 
 34 
 35 
 36 
 37 
 38 
 39 
 40 
 41 
 42 
 43 
 44 
 45 
 46 
 47 
 48 
 49 
 50 
 51 
 52 
 53 
 54 
 55 
 56 
 57 
 58 
 59 
// File   : gui/componenents/calculators/Roman.java
// Description: A static method for converting binary integers to Roman numbers.
// Illustrates: Static inner value class, StringBuffer, throw exceptions.
// Author : Fred Swartz - 2006-12-29 - Placed in public domain.

/////////////////////////////////////////////////////////////////// class Roman
public class Roman {
    //================================================================ constant
    // This could be alternatively be done with parallel arrays.
    // Another alternative would be Pair<Integer, String>
    final static RomanValue[] ROMAN_VALUE_TABLE = {
        new RomanValue(1000, "M"),
        new RomanValue( 900, "CM"),
        new RomanValue( 500, "D"),
        new RomanValue( 400, "CD"),
        new RomanValue( 100, "C"),
        new RomanValue(  90, "XC"),
        new RomanValue(  50, "L"),
        new RomanValue(  40, "XL"),
        new RomanValue(  10, "X"),
        new RomanValue(   9, "IX"),
        new RomanValue(   5, "V"),
        new RomanValue(   4, "IV"),
        new RomanValue(   1, "I")
    };
    
    //============================================================== int2roman
    public static String int2roman(int n) {
        if (n >= 4000  || n < 1) {
            throw new NumberFormatException("Numbers must be in range 1-3999");
        }
        StringBuffer result = new StringBuffer(10);
        
        //... Start with largest value, and work toward smallest.
        for (RomanValue equiv : ROMAN_VALUE_TABLE) {
            //... Remove as many of this value as possible (maybe none).
            while (n >= equiv.intVal) {
                n -= equiv.intVal;            // Subtract value.
                result.append(equiv.romVal);  // Add roman equivalent.
            }
        }
        return result.toString();
    }
    
    ///////////////////////////////////////////////////////// inner value class
    private static class RomanValue {
        //============================================================== fields
        //... No need to make this fields private because they are
        //    used only in this private value class.
        int    intVal;     // Integer value.
        String romVal;     // Equivalent roman numeral.
        
        //========================================================= constructor
        RomanValue(int dec, String rom) {
            this.intVal = dec;
            this.romVal = rom;
        }
    }
}

Exercises

  1. Provide an additional conversion from roman numerals to decimal.