| package com.googlecode.prolog_cafe.lang; |
| import java.util.regex.Matcher; |
| import java.util.regex.Pattern; |
| /** |
| * The <code>PredicateEncoder</code> class contains static methods for encoding predicate names.<br> |
| * The predicate with <code>hoge:f/n</code> is encoded to <code>hoge.PRED_f_n</code>, where |
| * <code>hoge</code> is package name, |
| * <code>f</code> is predicate name, and |
| * <code>n</code> is arity.<br> |
| * |
| * When encoding a predicate name, we apply the following rules:<br> |
| *<ul> |
| *<li>The alphanumeric characters |
| * “<code>a</code>” through “<code>z</code>”, |
| * “<code>A</code>” through “<code>Z</code>” and |
| * “<code>0</code>” through “<code>9</code>” remain the same. |
| *<li>The special characters “<code>_</code>” and “<code>$</code>” remain the same. |
| *<li>All other characters are first converted into a list of character codes. |
| * Then each character code is represented by the 5-character string “<code>$XYZW</code>”, |
| * where <code>XYZW</code> is the four-digit hexadecimal representation of the character code. |
| *</ul> |
| * |
| * For example, |
| * a predicate with <code>hoge:(=..)/2</code> is encoded to <code>hoge.PRED_$003D$002E$002E_2</code>. |
| * |
| * @author Mutsunori Banbara (banbara@kobe-u.ac.jp) |
| * @author Naoyuki Tamura (tamura@kobe-u.ac.jp) |
| * @version 1.1 |
| */ |
| public class PredicateEncoder { |
| |
| /** |
| * Returns a string representation of class for |
| * the predicate with the given arguments. |
| * @param pkg package name |
| * @param functor predicate name |
| * @param arity predicate arity |
| * @return a string representation of class for |
| * the predicate that corresponds to <code>pkg:functor/arity</code>. |
| */ |
| public static String encode(String pkg, String functor, int arity) { |
| String x = functor; |
| Pattern p = Pattern.compile("([^a-zA-Z0-9_'$'])"); |
| Matcher m = p.matcher(x); |
| StringBuffer sb = new StringBuffer(); |
| boolean result = m.find(); |
| while (result) { |
| // m.appendReplacement(sb, String.format("\\$%2X", (int)(m.group().charAt(0)))); |
| m.appendReplacement(sb, String.format("\\$%04X", (int)(m.group().charAt(0)))); |
| result = m.find(); |
| } |
| m.appendTail(sb); |
| x = sb.toString(); |
| return pkg + ".PRED_" + x + "_" + arity; |
| } |
| |
| public static String decodeFunctor(String className) { |
| // Remove the Java package name, if present. |
| int dot = className.lastIndexOf('.'); |
| if (0 < dot) |
| className = className.substring(dot + 1); |
| |
| // Trim the common PRED_ prefix. |
| if (className.startsWith("PRED_")) |
| className = className.substring("PRED_".length()); |
| |
| // Drop the arity from the end. |
| int us = className.lastIndexOf('_'); |
| if (0 < us) |
| className = className.substring(0, us); |
| |
| Pattern p = Pattern.compile("\\$([0-9A-F]{4})"); |
| Matcher m = p.matcher(className); |
| StringBuffer r = new StringBuffer(); |
| while (m.find()) { |
| char c = (char) Integer.parseInt(m.group()); |
| m.appendReplacement(r, Character.toString(c)); |
| } |
| m.appendTail(r); |
| return r.toString(); |
| } |
| |
| public static void main(String argv[]) { |
| String p = argv[0]; |
| String f = argv[1]; |
| int n = (Integer.valueOf(argv[2])).intValue(); |
| System.out.println(p + ":" + f + "/" + n); |
| System.out.println(PredicateEncoder.encode(p,f,n)); |
| } |
| } |