Non mi è piaciuta nessuna delle implementazioni (perché usano un Regex che è un'operazione costosa, o una libreria che è eccessiva se hai solo bisogno di un metodo), quindi ho finito per usare la classe java.net.URI con alcuni controlli extra e limitando i protocolli a: http, https, file, ftp, mailto, news, urn.
E sì, catturare le eccezioni può essere un'operazione costosa, ma probabilmente non così grave come le espressioni regolari:
final static Set<String> protocols, protocolsWithHost;
static {
protocolsWithHost = new HashSet<String>(
Arrays.asList( new String[]{ "file", "ftp", "http", "https" } )
);
protocols = new HashSet<String>(
Arrays.asList( new String[]{ "mailto", "news", "urn" } )
);
protocols.addAll(protocolsWithHost);
}
public static boolean isURI(String str) {
int colon = str.indexOf(':');
if (colon < 3) return false;
String proto = str.substring(0, colon).toLowerCase();
if (!protocols.contains(proto)) return false;
try {
URI uri = new URI(str);
if (protocolsWithHost.contains(proto)) {
if (uri.getHost() == null) return false;
String path = uri.getPath();
if (path != null) {
for (int i=path.length()-1; i >= 0; i--) {
if ("?<>:*|\"".indexOf( path.charAt(i) ) > -1)
return false;
}
}
}
return true;
} catch ( Exception ex ) {}
return false;
}