2015-11-11 20:20:04 -08:00
|
|
|
// onjava/Countries.java
|
2016-12-30 17:23:13 -08:00
|
|
|
// (c)2017 MindView LLC: see Copyright.txt
|
2015-11-15 15:51:35 -08:00
|
|
|
// We make no guarantees that this code is fit for any purpose.
|
2016-09-23 13:23:35 -06:00
|
|
|
// Visit http://OnJava8.com for more book information.
|
2016-01-25 18:05:55 -08:00
|
|
|
// "Flyweight" Maps and Lists of sample data
|
2016-07-28 12:48:23 -06:00
|
|
|
// {java onjava.Countries}
|
2015-11-11 20:20:04 -08:00
|
|
|
package onjava;
|
2015-06-15 17:47:35 -07:00
|
|
|
import java.util.*;
|
|
|
|
|
|
|
|
public class Countries {
|
|
|
|
public static final String[][] DATA = {
|
|
|
|
// Africa
|
2016-11-01 11:43:22 -07:00
|
|
|
{"ALGERIA","Algiers"},
|
|
|
|
{"ANGOLA","Luanda"},
|
|
|
|
{"BENIN","Porto-Novo"},
|
|
|
|
{"BOTSWANA","Gaberone"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"BURKINA FASO","Ouagadougou"},
|
|
|
|
{"BURUNDI","Bujumbura"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"CAMEROON","Yaounde"},
|
|
|
|
{"CAPE VERDE","Praia"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"CENTRAL AFRICAN REPUBLIC","Bangui"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"CHAD","N'djamena"},
|
|
|
|
{"COMOROS","Moroni"},
|
|
|
|
{"CONGO","Brazzaville"},
|
|
|
|
{"DJIBOUTI","Dijibouti"},
|
|
|
|
{"EGYPT","Cairo"},
|
|
|
|
{"EQUATORIAL GUINEA","Malabo"},
|
|
|
|
{"ERITREA","Asmara"},
|
|
|
|
{"ETHIOPIA","Addis Ababa"},
|
|
|
|
{"GABON","Libreville"},
|
|
|
|
{"THE GAMBIA","Banjul"},
|
|
|
|
{"GHANA","Accra"},
|
|
|
|
{"GUINEA","Conakry"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"BISSAU","Bissau"},
|
|
|
|
{"COTE D'IVOIR (IVORY COAST)","Yamoussoukro"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"KENYA","Nairobi"},
|
|
|
|
{"LESOTHO","Maseru"},
|
|
|
|
{"LIBERIA","Monrovia"},
|
|
|
|
{"LIBYA","Tripoli"},
|
|
|
|
{"MADAGASCAR","Antananarivo"},
|
|
|
|
{"MALAWI","Lilongwe"},
|
|
|
|
{"MALI","Bamako"},
|
|
|
|
{"MAURITANIA","Nouakchott"},
|
|
|
|
{"MAURITIUS","Port Louis"},
|
|
|
|
{"MOROCCO","Rabat"},
|
|
|
|
{"MOZAMBIQUE","Maputo"},
|
|
|
|
{"NAMIBIA","Windhoek"},
|
|
|
|
{"NIGER","Niamey"},
|
|
|
|
{"NIGERIA","Abuja"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"RWANDA","Kigali"},
|
|
|
|
{"SAO TOME E PRINCIPE","Sao Tome"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"SENEGAL","Dakar"},
|
|
|
|
{"SEYCHELLES","Victoria"},
|
|
|
|
{"SIERRA LEONE","Freetown"},
|
|
|
|
{"SOMALIA","Mogadishu"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"SOUTH AFRICA","Pretoria/Cape Town"},
|
|
|
|
{"SUDAN","Khartoum"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"SWAZILAND","Mbabane"},
|
|
|
|
{"TANZANIA","Dodoma"},
|
|
|
|
{"TOGO","Lome"},
|
|
|
|
{"TUNISIA","Tunis"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"UGANDA","Kampala"},
|
|
|
|
{"DEMOCRATIC REPUBLIC OF THE CONGO (ZAIRE)",
|
|
|
|
"Kinshasa"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"ZAMBIA","Lusaka"},
|
|
|
|
{"ZIMBABWE","Harare"},
|
2015-06-15 17:47:35 -07:00
|
|
|
// Asia
|
2016-11-01 11:43:22 -07:00
|
|
|
{"AFGHANISTAN","Kabul"},
|
|
|
|
{"BAHRAIN","Manama"},
|
|
|
|
{"BANGLADESH","Dhaka"},
|
|
|
|
{"BHUTAN","Thimphu"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"BRUNEI","Bandar Seri Begawan"},
|
|
|
|
{"CAMBODIA","Phnom Penh"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"CHINA","Beijing"},
|
|
|
|
{"CYPRUS","Nicosia"},
|
|
|
|
{"INDIA","New Delhi"},
|
|
|
|
{"INDONESIA","Jakarta"},
|
|
|
|
{"IRAN","Tehran"},
|
|
|
|
{"IRAQ","Baghdad"},
|
|
|
|
{"ISRAEL","Jerusalem"},
|
|
|
|
{"JAPAN","Tokyo"},
|
|
|
|
{"JORDAN","Amman"},
|
|
|
|
{"KUWAIT","Kuwait City"},
|
|
|
|
{"LAOS","Vientiane"},
|
|
|
|
{"LEBANON","Beirut"},
|
|
|
|
{"MALAYSIA","Kuala Lumpur"},
|
|
|
|
{"THE MALDIVES","Male"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"MONGOLIA","Ulan Bator"},
|
|
|
|
{"MYANMAR (BURMA)","Rangoon"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"NEPAL","Katmandu"},
|
|
|
|
{"NORTH KOREA","P'yongyang"},
|
|
|
|
{"OMAN","Muscat"},
|
|
|
|
{"PAKISTAN","Islamabad"},
|
|
|
|
{"PHILIPPINES","Manila"},
|
|
|
|
{"QATAR","Doha"},
|
|
|
|
{"SAUDI ARABIA","Riyadh"},
|
|
|
|
{"SINGAPORE","Singapore"},
|
|
|
|
{"SOUTH KOREA","Seoul"},
|
|
|
|
{"SRI LANKA","Colombo"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"SYRIA","Damascus"},
|
|
|
|
{"TAIWAN (REPUBLIC OF CHINA)","Taipei"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"THAILAND","Bangkok"},
|
|
|
|
{"TURKEY","Ankara"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"UNITED ARAB EMIRATES","Abu Dhabi"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"VIETNAM","Hanoi"},
|
|
|
|
{"YEMEN","Sana'a"},
|
2015-06-15 17:47:35 -07:00
|
|
|
// Australia and Oceania
|
2016-11-01 11:43:22 -07:00
|
|
|
{"AUSTRALIA","Canberra"},
|
|
|
|
{"FIJI","Suva"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"KIRIBATI","Bairiki"},
|
|
|
|
{"MARSHALL ISLANDS","Dalap-Uliga-Darrit"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"MICRONESIA","Palikir"},
|
|
|
|
{"NAURU","Yaren"},
|
|
|
|
{"NEW ZEALAND","Wellington"},
|
|
|
|
{"PALAU","Koror"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"PAPUA NEW GUINEA","Port Moresby"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"SOLOMON ISLANDS","Honaira"},
|
|
|
|
{"TONGA","Nuku'alofa"},
|
|
|
|
{"TUVALU","Fongafale"},
|
|
|
|
{"VANUATU","Port Vila"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"WESTERN SAMOA","Apia"},
|
|
|
|
// Eastern Europe and former USSR
|
2016-11-01 11:43:22 -07:00
|
|
|
{"ARMENIA","Yerevan"},
|
|
|
|
{"AZERBAIJAN","Baku"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"BELARUS (BYELORUSSIA)","Minsk"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"BULGARIA","Sofia"},
|
|
|
|
{"GEORGIA","Tbilisi"},
|
|
|
|
{"KAZAKSTAN","Almaty"},
|
|
|
|
{"KYRGYZSTAN","Alma-Ata"},
|
|
|
|
{"MOLDOVA","Chisinau"},
|
|
|
|
{"RUSSIA","Moscow"},
|
2016-01-25 18:05:55 -08:00
|
|
|
{"TAJIKISTAN","Dushanbe"},
|
|
|
|
{"TURKMENISTAN","Ashkabad"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"UKRAINE","Kyiv"},
|
|
|
|
{"UZBEKISTAN","Tashkent"},
|
2015-06-15 17:47:35 -07:00
|
|
|
// Europe
|
2016-11-01 11:43:22 -07:00
|
|
|
{"ALBANIA","Tirana"},
|
|
|
|
{"ANDORRA","Andorra la Vella"},
|
|
|
|
{"AUSTRIA","Vienna"},
|
|
|
|
{"BELGIUM","Brussels"},
|
|
|
|
{"BOSNIA-HERZEGOVINA","Sarajevo"},
|
|
|
|
{"CROATIA","Zagreb"},
|
|
|
|
{"CZECH REPUBLIC","Prague"},
|
|
|
|
{"DENMARK","Copenhagen"},
|
|
|
|
{"ESTONIA","Tallinn"},
|
|
|
|
{"FINLAND","Helsinki"},
|
|
|
|
{"FRANCE","Paris"},
|
|
|
|
{"GERMANY","Berlin"},
|
|
|
|
{"GREECE","Athens"},
|
|
|
|
{"HUNGARY","Budapest"},
|
|
|
|
{"ICELAND","Reykjavik"},
|
|
|
|
{"IRELAND","Dublin"},
|
|
|
|
{"ITALY","Rome"},
|
|
|
|
{"LATVIA","Riga"},
|
|
|
|
{"LIECHTENSTEIN","Vaduz"},
|
|
|
|
{"LITHUANIA","Vilnius"},
|
|
|
|
{"LUXEMBOURG","Luxembourg"},
|
|
|
|
{"MACEDONIA","Skopje"},
|
|
|
|
{"MALTA","Valletta"},
|
|
|
|
{"MONACO","Monaco"},
|
|
|
|
{"MONTENEGRO","Podgorica"},
|
|
|
|
{"THE NETHERLANDS","Amsterdam"},
|
|
|
|
{"NORWAY","Oslo"},
|
|
|
|
{"POLAND","Warsaw"},
|
|
|
|
{"PORTUGAL","Lisbon"},
|
|
|
|
{"ROMANIA","Bucharest"},
|
|
|
|
{"SAN MARINO","San Marino"},
|
|
|
|
{"SERBIA","Belgrade"},
|
|
|
|
{"SLOVAKIA","Bratislava"},
|
|
|
|
{"SLOVENIA","Ljuijana"},
|
|
|
|
{"SPAIN","Madrid"},
|
|
|
|
{"SWEDEN","Stockholm"},
|
|
|
|
{"SWITZERLAND","Berne"},
|
|
|
|
{"UNITED KINGDOM","London"},
|
|
|
|
{"VATICAN CITY","Vatican City"},
|
2015-06-15 17:47:35 -07:00
|
|
|
// North and Central America
|
|
|
|
{"ANTIGUA AND BARBUDA","Saint John's"},
|
|
|
|
{"BAHAMAS","Nassau"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"BARBADOS","Bridgetown"},
|
|
|
|
{"BELIZE","Belmopan"},
|
|
|
|
{"CANADA","Ottawa"},
|
|
|
|
{"COSTA RICA","San Jose"},
|
|
|
|
{"CUBA","Havana"},
|
|
|
|
{"DOMINICA","Roseau"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"DOMINICAN REPUBLIC","Santo Domingo"},
|
|
|
|
{"EL SALVADOR","San Salvador"},
|
|
|
|
{"GRENADA","Saint George's"},
|
|
|
|
{"GUATEMALA","Guatemala City"},
|
|
|
|
{"HAITI","Port-au-Prince"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"HONDURAS","Tegucigalpa"},
|
|
|
|
{"JAMAICA","Kingston"},
|
|
|
|
{"MEXICO","Mexico City"},
|
|
|
|
{"NICARAGUA","Managua"},
|
|
|
|
{"PANAMA","Panama City"},
|
|
|
|
{"ST. KITTS AND NEVIS","Basseterre"},
|
|
|
|
{"ST. LUCIA","Castries"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"ST. VINCENT AND THE GRENADINES","Kingstown"},
|
|
|
|
{"UNITED STATES OF AMERICA","Washington, D.C."},
|
|
|
|
// South America
|
|
|
|
{"ARGENTINA","Buenos Aires"},
|
|
|
|
{"BOLIVIA","Sucre (legal)/La Paz(administrative)"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"BRAZIL","Brasilia"},
|
|
|
|
{"CHILE","Santiago"},
|
|
|
|
{"COLOMBIA","Bogota"},
|
|
|
|
{"ECUADOR","Quito"},
|
|
|
|
{"GUYANA","Georgetown"},
|
|
|
|
{"PARAGUAY","Asuncion"},
|
|
|
|
{"PERU","Lima"},
|
|
|
|
{"SURINAME","Paramaribo"},
|
2015-06-15 17:47:35 -07:00
|
|
|
{"TRINIDAD AND TOBAGO","Port of Spain"},
|
2016-11-01 11:43:22 -07:00
|
|
|
{"URUGUAY","Montevideo"},
|
|
|
|
{"VENEZUELA","Caracas"},
|
2015-06-15 17:47:35 -07:00
|
|
|
};
|
|
|
|
// Use AbstractMap by implementing entrySet()
|
|
|
|
private static class FlyweightMap
|
|
|
|
extends AbstractMap<String,String> {
|
|
|
|
private static class Entry
|
|
|
|
implements Map.Entry<String,String> {
|
|
|
|
int index;
|
|
|
|
Entry(int index) { this.index = index; }
|
|
|
|
@Override
|
|
|
|
public boolean equals(Object o) {
|
2017-01-08 22:55:49 -08:00
|
|
|
return o instanceof FlyweightMap &&
|
|
|
|
Objects.equals(DATA[index][0], o);
|
|
|
|
}
|
|
|
|
@Override
|
|
|
|
public int hashCode() {
|
2017-01-10 14:11:16 -08:00
|
|
|
return Objects.hashCode(DATA[index][0]);
|
2015-06-15 17:47:35 -07:00
|
|
|
}
|
|
|
|
@Override
|
|
|
|
public String getKey() { return DATA[index][0]; }
|
|
|
|
@Override
|
2017-05-10 11:45:39 -06:00
|
|
|
public String getValue() {
|
|
|
|
return DATA[index][1];
|
|
|
|
}
|
2015-06-15 17:47:35 -07:00
|
|
|
@Override
|
|
|
|
public String setValue(String value) {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
}
|
2016-07-05 14:46:09 -06:00
|
|
|
// Implement size() & iterator() for AbstractSet:
|
2015-06-15 17:47:35 -07:00
|
|
|
static class EntrySet
|
|
|
|
extends AbstractSet<Map.Entry<String,String>> {
|
|
|
|
private int size;
|
|
|
|
EntrySet(int size) {
|
|
|
|
if(size < 0)
|
|
|
|
this.size = 0;
|
|
|
|
// Can't be any bigger than the array:
|
|
|
|
else if(size > DATA.length)
|
|
|
|
this.size = DATA.length;
|
|
|
|
else
|
|
|
|
this.size = size;
|
|
|
|
}
|
|
|
|
@Override
|
|
|
|
public int size() { return size; }
|
|
|
|
private class Iter
|
|
|
|
implements Iterator<Map.Entry<String,String>> {
|
|
|
|
// Only one Entry object per Iterator:
|
|
|
|
private Entry entry = new Entry(-1);
|
|
|
|
@Override
|
|
|
|
public boolean hasNext() {
|
|
|
|
return entry.index < size - 1;
|
|
|
|
}
|
|
|
|
@Override
|
|
|
|
public Map.Entry<String,String> next() {
|
|
|
|
entry.index++;
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
@Override
|
|
|
|
public void remove() {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@Override
|
|
|
|
public
|
|
|
|
Iterator<Map.Entry<String,String>> iterator() {
|
|
|
|
return new Iter();
|
|
|
|
}
|
|
|
|
}
|
2017-01-22 16:48:11 -08:00
|
|
|
private static
|
|
|
|
Set<Map.Entry<String,String>> entries =
|
2015-06-15 17:47:35 -07:00
|
|
|
new EntrySet(DATA.length);
|
|
|
|
@Override
|
|
|
|
public Set<Map.Entry<String,String>> entrySet() {
|
|
|
|
return entries;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Create a partial map of 'size' countries:
|
|
|
|
static Map<String,String> select(final int size) {
|
|
|
|
return new FlyweightMap() {
|
|
|
|
@Override
|
|
|
|
public Set<Map.Entry<String,String>> entrySet() {
|
|
|
|
return new EntrySet(size);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
static Map<String,String> map = new FlyweightMap();
|
|
|
|
public static Map<String,String> capitals() {
|
|
|
|
return map; // The entire map
|
|
|
|
}
|
|
|
|
public static Map<String,String> capitals(int size) {
|
|
|
|
return select(size); // A partial map
|
|
|
|
}
|
|
|
|
static List<String> names =
|
|
|
|
new ArrayList<>(map.keySet());
|
|
|
|
// All the names:
|
|
|
|
public static List<String> names() { return names; }
|
|
|
|
// A partial list:
|
|
|
|
public static List<String> names(int size) {
|
|
|
|
return new ArrayList<>(select(size).keySet());
|
|
|
|
}
|
|
|
|
public static void main(String[] args) {
|
2015-11-03 12:00:44 -08:00
|
|
|
System.out.println(capitals(10));
|
|
|
|
System.out.println(names(10));
|
|
|
|
System.out.println(new HashMap<>(capitals(3)));
|
2017-05-10 11:45:39 -06:00
|
|
|
System.out.println(
|
|
|
|
new LinkedHashMap<>(capitals(3)));
|
2015-11-03 12:00:44 -08:00
|
|
|
System.out.println(new TreeMap<>(capitals(3)));
|
|
|
|
System.out.println(new Hashtable<>(capitals(3)));
|
|
|
|
System.out.println(new HashSet<>(names(6)));
|
|
|
|
System.out.println(new LinkedHashSet<>(names(6)));
|
|
|
|
System.out.println(new TreeSet<>(names(6)));
|
|
|
|
System.out.println(new ArrayList<>(names(6)));
|
|
|
|
System.out.println(new LinkedList<>(names(6)));
|
|
|
|
System.out.println(capitals().get("BRAZIL"));
|
2015-06-15 17:47:35 -07:00
|
|
|
}
|
2015-09-07 11:44:36 -06:00
|
|
|
}
|
|
|
|
/* Output:
|
2015-06-15 17:47:35 -07:00
|
|
|
{ALGERIA=Algiers, ANGOLA=Luanda, BENIN=Porto-Novo,
|
|
|
|
BOTSWANA=Gaberone, BURKINA FASO=Ouagadougou,
|
|
|
|
BURUNDI=Bujumbura, CAMEROON=Yaounde, CAPE VERDE=Praia,
|
|
|
|
CENTRAL AFRICAN REPUBLIC=Bangui, CHAD=N'djamena}
|
2017-05-10 11:45:39 -06:00
|
|
|
[ALGERIA, ANGOLA, BENIN, BOTSWANA, BURKINA FASO,
|
|
|
|
BURUNDI, CAMEROON, CAPE VERDE, CENTRAL AFRICAN
|
|
|
|
REPUBLIC, CHAD]
|
2015-06-15 17:47:35 -07:00
|
|
|
{BENIN=Porto-Novo, ANGOLA=Luanda, ALGERIA=Algiers}
|
|
|
|
{ALGERIA=Algiers, ANGOLA=Luanda, BENIN=Porto-Novo}
|
|
|
|
{ALGERIA=Algiers, ANGOLA=Luanda, BENIN=Porto-Novo}
|
|
|
|
{ALGERIA=Algiers, ANGOLA=Luanda, BENIN=Porto-Novo}
|
2017-05-10 11:45:39 -06:00
|
|
|
[BENIN, BOTSWANA, ANGOLA, BURKINA FASO, ALGERIA,
|
|
|
|
BURUNDI]
|
|
|
|
[ALGERIA, ANGOLA, BENIN, BOTSWANA, BURKINA FASO,
|
|
|
|
BURUNDI]
|
|
|
|
[ALGERIA, ANGOLA, BENIN, BOTSWANA, BURKINA FASO,
|
|
|
|
BURUNDI]
|
|
|
|
[ALGERIA, ANGOLA, BENIN, BOTSWANA, BURKINA FASO,
|
|
|
|
BURUNDI]
|
|
|
|
[ALGERIA, ANGOLA, BENIN, BOTSWANA, BURKINA FASO,
|
|
|
|
BURUNDI]
|
2015-06-15 17:47:35 -07:00
|
|
|
Brasilia
|
2015-09-07 11:44:36 -06:00
|
|
|
*/
|