3

我是 Java 新手,我在这里看到了一个问答部分,其中包含两个删除可变性的示例。在测试MutableString.java 时

import java.lang.reflect.Field; 

public class MutableString {

    public static void main(String[] args) { 
        String s = "Immutable"; 
        String t = "Notreally"; 

        mutate(s, t);
        StdOut.println(t); 

        // strings are interned so this doesn't even print "Immutable" (!)
        StdOut.println("Immutable");
    } 

    // change the first min(|s|, |t|) characters of s to t
    public static void mutate(String s, String t) {
        try {
            Field val = String.class.getDeclaredField("value"); 
            Field off = String.class.getDeclaredField("offset"); 
            val.setAccessible(true); 
            off.setAccessible(true); 
            int offset   = off.getInt(s); 
            char[] value = (char[]) val.get(s); 
            for (int i = 0; i < Math.min(s.length(), t.length()); i++)
                value[offset + i] = t.charAt(i); 
        } 
        catch (Exception e) { e.printStackTrace(); }
    } 

} 

我收到以下错误:

java.lang.NoSuchFieldException: offset

任何关于以下内容的输入将不胜感激:

a)为什么会出现此异常
b)如何检查类中存在哪些字段(特别是 Java 字符串)

4

1 回答 1

10

免责声明:这些黑客是有趣的学习课程和有趣的琐事。但它们绝对不是您想在任何生产代码中使用的东西。它导致疼痛。

就其本质而言,这样的黑客攻击总是取决于被黑客攻击的类的实现细节。

在您的情况下,您似乎正在使用一个String没有offset字段的实现,但使用了一些其他机制(或者可能只是一个不同的名称!)。

例如,我刚刚查看了 Oracle Java 7 String 类,它不再具有offset字段(在 Java 6 及更早版本中用于共享char[]子字符串)!*

您可以使用Class.getDeclaredFields()来检查此实现确实定义了哪些字段:

for (Field f : String.class.getDeclaredFields()) {
  System.out.println(f);
}

对于适用于 Java 7 的该 hack 版本,您可以这样做:

public static void mutate(String s, String t) {
    try {
        Field val = String.class.getDeclaredField("value"); 
        val.setAccessible(true); 
        char[] value = (char[]) val.get(s); 
        for (int i = 0; i < Math.min(s.length(), t.length()); i++)
            value[i] = t.charAt(i); 
    } 
    catch (Exception e) { e.printStackTrace(); }
} 

当然,如果内部结构String再次发生变化,这也会中断。

*这是一封谈论该更改的电子邮件char[],似乎在少数特殊情况下,共享唯一可以提高性能。

于 2013-07-05T12:58:55.470 回答