求教Class.forName()的用法

wyh15 2003-07-02 08:14:36
如下是一段forName()的使用代码:

public void print(){

try{
Object _obj =Class.forName("application.test.javaclass.PrintJavaClassHello").newInstance();
System.out.println(_obj.getClass().getName());

//还必须要进行显示地强制类型转换,不能满足工作流的要求 ((PrintJavaClassHello)_obj).printHello();

}catch (ClassNotFoundException e1){}
catch (InstantiationException e2){}
catch (IllegalAccessException e3){}
}
问题一:正如我在代码中所注解的,如果我不用PrintJavaClassHello进行强制类型转换,还能调用PrintJavaClassHello中的方法printHello()吗?如果这样不能,那采用什么样的解决方案能实现这一点呢?

...全文
257 4 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
4 条回复
切换为时间正序
请发表友善的回复…
发表回复
aemon 2003-07-03
  • 打赏
  • 举报
回复
问题一:正如我在代码中所注解的,如果我不用PrintJavaClassHello进行强制类型转换,还能调用PrintJavaClassHello中的方法printHello()吗?如果这样不能,那采用什么样的解决方案能实现这一点呢?

使用rtti!详细看thinking in java 2nd 的11章介绍!
在Object _obj =Class.forName(...)").newInstance();得到对象以后
调用isInstance(),判断得到的obj是否为能进行某项操作的对象。
抄thinking in java中的例子:
//: PetCount3.java
// Using Java 1.1 isInstance()
package c11.petcount3;
import java.util.*;

class Pet {}
class Dog extends Pet {}
class Pug extends Dog {}
class Cat extends Pet {}
class Rodent extends Pet {}
class Gerbil extends Rodent {}
class Hamster extends Rodent {}

class Counter { int i; }

public class PetCount3 {
public static void main(String[] args) {
Vector pets = new Vector();
Class[] petTypes = {
Pet.class,
Dog.class,
Pug.class,
Cat.class,
Rodent.class,
Gerbil.class,
Hamster.class,
};
try {
for(int i = 0; i < 15; i++) {
// Offset by one to eliminate Pet.class:
int rnd = 1 + (int)(
Math.random() * (petTypes.length - 1));
pets.addElement(
petTypes[rnd].newInstance());
}
} catch(InstantiationException e) {}
catch(IllegalAccessException e) {}
Hashtable h = new Hashtable();
for(int i = 0; i < petTypes.length; i++)
h.put(petTypes[i].toString(),
new Counter());
for(int i = 0; i < pets.size(); i++) {
Object o = pets.elementAt(i);
// Using isInstance to eliminate individual
// instanceof expressions:
for (int j = 0; j < petTypes.length; ++j)
if (petTypes[j].isInstance(o)) {
String key = petTypes[j].toString();
((Counter)h.get(key)).i++;
}
}
for(int i = 0; i < pets.size(); i++)
System.out.println(
pets.elementAt(i).getClass().toString());
Enumeration keys = h.keys();
while(keys.hasMoreElements()) {
String nm = (String)keys.nextElement();
Counter cnt = (Counter)h.get(nm);
System.out.println(
nm.substring(nm.lastIndexOf('.') + 1) +
" quantity: " + cnt.i);
}
}
} ///:~

foronlyyou 2003-07-02
  • 打赏
  • 举报
回复
mark
kangta 2003-07-02
  • 打赏
  • 举报
回复
关注!帮着UP
liujiboy 2003-07-02
  • 打赏
  • 举报
回复
如果是动态加载就不能采用强制转换类型这种方法。因为我们并不知道这个类到底是Class到底是什么class。也就说如果你用这些代码:
String className="MyClass";
Clsss iClass = Class.forName(className);
Object iObj =iClass.newInstance();
生成了iObj,你确实不能用classname来动态转换。
如果要使用这个生成的object应该怎么办呢?
我假设你的MyClass如下定义。
public class MyClass
{
public MyClass(){
}
public void sayHello(String who)
{
System.out.println("hello world "+who);
}
}
如果要动态的使用sayHello,可以用如下代码
import java.lang.reflect.*;
class TestClass
{
public static void main(String[] args)
{
//这是sayHello的参数类型为String
Class[] parameterTypes={String.class};
//准备把you这次字符串传进去
Object[] parameter={"you"};
//以下是动态装入的类名和需要调用的函数名
String className="MyClass";
String methodName="sayHello";
try{
Class iClass=Class.forName(className);
//产生一个Method对象(sayHello)
Method method=iClass.getMethod(methodName,parameterTypes);
//Class.forname只是动态的装入了类,我们还需要产生对象,因此
//需要调用它的构造函数,用以下方法调用
Constructor con=iClass.getConstructor(new Class[]{});
Object myclass=con.newInstance(new Object[]{});
//对象生成啦,现在调用它
method.invoke(myclass,parameter);

}
catch(Exception e)
{
e.printStackTrace();
}
}
}
上面的代码答应hello world you
这段代码等同于
MyClass myclass=new MyClass();
myclass.sayHello("you");
由此可以看出动态的调用比静态的复杂得多。具体的api函数请参考java api

62,635

社区成员

发帖
与我相关
我的任务
社区描述
Java 2 Standard Edition
社区管理员
  • Java SE
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧