|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
|
|
package sun.nio.cs.ext; |
|
|
|
import java.lang.ref.SoftReference; |
|
import java.nio.charset.Charset; |
|
import java.nio.charset.spi.CharsetProvider; |
|
import java.util.ArrayList; |
|
import java.util.TreeMap; |
|
import java.util.Iterator; |
|
import java.util.Locale; |
|
import java.util.Map; |
|
|
|
|
|
/** |
|
* Abstract base class for charset providers. |
|
* |
|
* @author Mark Reinhold |
|
*/ |
|
|
|
public class AbstractCharsetProvider |
|
extends CharsetProvider |
|
{ |
|
|
|
|
|
*/ |
|
private Map<String,String> classMap |
|
= new TreeMap<>(String.CASE_INSENSITIVE_ORDER); |
|
|
|
|
|
*/ |
|
private Map<String,String> aliasMap |
|
= new TreeMap<>(String.CASE_INSENSITIVE_ORDER); |
|
|
|
|
|
*/ |
|
private Map<String,String[]> aliasNameMap |
|
= new TreeMap<>(String.CASE_INSENSITIVE_ORDER); |
|
|
|
|
|
*/ |
|
private Map<String,SoftReference<Charset>> cache |
|
= new TreeMap<>(String.CASE_INSENSITIVE_ORDER); |
|
|
|
private String packagePrefix; |
|
|
|
protected AbstractCharsetProvider() { |
|
packagePrefix = "sun.nio.cs"; |
|
} |
|
|
|
protected AbstractCharsetProvider(String pkgPrefixName) { |
|
packagePrefix = pkgPrefixName; |
|
} |
|
|
|
|
|
|
|
*/ |
|
private static <K,V> void put(Map<K,V> m, K name, V value) { |
|
if (!m.containsKey(name)) |
|
m.put(name, value); |
|
} |
|
|
|
private static <K,V> void remove(Map<K,V> m, K name) { |
|
V x = m.remove(name); |
|
assert (x != null); |
|
} |
|
|
|
|
|
*/ |
|
protected void charset(String name, String className, String[] aliases) { |
|
synchronized (this) { |
|
put(classMap, name, className); |
|
for (int i = 0; i < aliases.length; i++) |
|
put(aliasMap, aliases[i], name); |
|
put(aliasNameMap, name, aliases); |
|
cache.clear(); |
|
} |
|
} |
|
|
|
protected void deleteCharset(String name, String[] aliases) { |
|
synchronized (this) { |
|
remove(classMap, name); |
|
for (int i = 0; i < aliases.length; i++) |
|
remove(aliasMap, aliases[i]); |
|
remove(aliasNameMap, name); |
|
cache.clear(); |
|
} |
|
} |
|
|
|
protected boolean hasCharset(String name) { |
|
synchronized (this) { |
|
return classMap.containsKey(name); |
|
} |
|
} |
|
|
|
|
|
*/ |
|
protected void init() { } |
|
|
|
private String canonicalize(String charsetName) { |
|
String acn = aliasMap.get(charsetName); |
|
return (acn != null) ? acn : charsetName; |
|
} |
|
|
|
private Charset lookup(String csn) { |
|
|
|
|
|
SoftReference<Charset> sr = cache.get(csn); |
|
if (sr != null) { |
|
Charset cs = sr.get(); |
|
if (cs != null) |
|
return cs; |
|
} |
|
|
|
|
|
String cln = classMap.get(csn); |
|
|
|
if (cln == null) |
|
return null; |
|
|
|
|
|
try { |
|
|
|
Class<?> c = Class.forName(packagePrefix + "." + cln, |
|
true, |
|
this.getClass().getClassLoader()); |
|
|
|
@SuppressWarnings("deprecation") |
|
Charset cs = (Charset)c.newInstance(); |
|
cache.put(csn, new SoftReference<Charset>(cs)); |
|
return cs; |
|
} catch (ClassNotFoundException x) { |
|
return null; |
|
} catch (IllegalAccessException x) { |
|
return null; |
|
} catch (InstantiationException x) { |
|
return null; |
|
} |
|
} |
|
|
|
public final Charset charsetForName(String charsetName) { |
|
synchronized (this) { |
|
init(); |
|
return lookup(canonicalize(charsetName)); |
|
} |
|
} |
|
|
|
public final Iterator<Charset> charsets() { |
|
|
|
final ArrayList<String> ks; |
|
synchronized (this) { |
|
init(); |
|
ks = new ArrayList<>(classMap.keySet()); |
|
} |
|
|
|
return new Iterator<Charset>() { |
|
Iterator<String> i = ks.iterator(); |
|
|
|
public boolean hasNext() { |
|
return i.hasNext(); |
|
} |
|
|
|
public Charset next() { |
|
String csn = i.next(); |
|
synchronized (AbstractCharsetProvider.this) { |
|
return lookup(csn); |
|
} |
|
} |
|
|
|
public void remove() { |
|
throw new UnsupportedOperationException(); |
|
} |
|
}; |
|
} |
|
|
|
public final String[] aliases(String charsetName) { |
|
synchronized (this) { |
|
init(); |
|
return aliasNameMap.get(charsetName); |
|
} |
|
} |
|
|
|
} |