我有一个类作为序列的一部分被实例化(一个方法通过一个随机的类列表并创建每个类的3个实例)
我正在编写列表中的一个类(注意:所有类都存在于同一个包中)
我无法修改任何其他类的源文件,我需要停止继我自己类之后的所有类的实例化,或者用空方法覆盖它们的构造函数。
到目前为止,我已经尝试使用反射,并一直在看类加载器,但不知道如何做到这一点
我可以通过反射获得构造函数的引用,但到目前为止还无法干扰它
edit ->我有一个想法(动态重写和重新编译目标类),代码编译,但我得到一个异常,阻止新类编译
private void reWrite(String name) {//throws IOException {
try{
File sourceFile = new File("/temp/" + "name" + ".java");
System.out.println("file defined");
FileWriter writer = new FileWriter(sourceFile);
System.out.println("filewriter defined");
writer.write("public class "+name+" extends Creature {/n"
+ "private static int instanceCount = 0;/n" + "public "+name+"() {/n"
+ " instanceCount++;/n" + "}/n" + "public void doTurn() {}/n"
+ "public final void creatureDestroyed() {/n"
+ " instanceCount--;/n" + "}/n"
+ "public final int getInstanceCount() {/n"
+ " return instanceCount;/n" + "}/n" + "}/n");
writer.close();
System.out.println("written");
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
StandardJavaFileManager fileManager = compiler.getStandardFileManager(null,null,null);
fileManager.setLocation(StandardLocation.CLASS_OUTPUT,
Arrays.asList(new File("/temp")));
// Compile the file
compiler.getTask(
null,
fileManager,
null,
null,
null,
fileManager.getJavaFileObjectsFromFiles(Arrays
.asList(sourceFile))).call();
System.out.println("compiled");
fileManager.close();
}catch(Exception e){
System.out.println("Print");
}
}
edit->以下是我根据
下面的注释提供的代码try {
getW = (World.class.getDeclaredMethod("getWorld"));
getW.setAccessible(true);
theWorld = getW.invoke(World.class);
getK = (Creature.class.getDeclaredMethod("die"));
getK.setAccessible(true);
} catch (NoSuchMethodException exc) {
System.out.println("Method Not Found.");
} catch (Exception e) {
System.out.println("well shit");
}
try {
f = World.class.getDeclaredField("locations");
f.setAccessible(true);
locations = f.get(theWorld);
loc = (ArrayList<Creature>) locations;
} catch (Exception e) {
System.out.println("well poop");
}
if (loc.size() > 0 && !allMine) {
for (int i = 0; i < loc.size(); i++) {
if (loc.get(i).getClass() != Colonial.class
&& loc.get(i).getClass() != Plant.class) {
try {
getK.invoke(loc.get(i));
} catch (Exception e) {
}
loc.set(i, new OpenLand());
}
}
}
您可以使用sun.misc.Unsafe#allocateInstance(MyClass.class)来构造实例,而无需实际调用构造函数。详见http://mishadoff.com/blog/java-magic-part-4-sun-dot-misc-dot-unsafe/
如果允许修改命令行,则应该尝试java代理。代理是一个类,它基本上在类加载之前挂钩,并且可以修改类的字节码。为了在不成为JVM专家的情况下修改字节码,javassist是一个很好的库,它允许您编写一些类似java的代码并在方法之前/之后注入它。所以基本上,你需要
- 写一个代理修改你的敌人类的构造函数
- 使用代理启动JVM
这里有一些关于代理的基本信息:
https://zeroturnaround.com/rebellabs/how-to-make-java-more-dynamic-with-runtime-code-generation/
javassist教程非常清晰:
http://jboss-javassist.github.io/javassist/tutorial/tutorial.html
当然这是一个肮脏的hack,但是你要求它
您不需要来'干扰'构造函数。它是固定的。你只需要调用它,用Constructor<T>.newInstance(...).
。如果你需要设置对象的其他属性,调用setter。如果setter不存在,你就不能这样做