如何在APEX中内省变量的类?

时间:2012-03-03 16:11:30

标签: dynamic salesforce apex-code

我想知道运行时变量/属性的类。例如:

Integer i = 5;

//pseudo-code
if (i.className == 'Integer') {
    System.debug('This is an integer.');
} else {
    System.debug('This is not an integer, but a ' + i.className);
}

我找不到在文档中返回类类型的方法/属性(假设它在那里)。我错过了吗?

7 个答案:

答案 0 :(得分:9)

来自p. 122 of the Apex Developer Guide

  

如果您需要在运行时验证对象是否实际上是   特定类的实例,使用instanceof关键字...

但是,您不能在其子类的实例上使用instanceof关键字,否则您将收到编译错误。例如:

Integer i = 0;
System.debug(i instanceof Integer);

>> COMPILE ERROR: Operation instanceof is always true since an instance of Integer is always an instance of Integer.

您只需要在超类上使用instanceof关键字。例如:

System.debug((Object)i instanceof Integer);

>> true

如果您需要有关类本身类型的信息,请检查System.Type方法(pg 396 of current Apex developer's guide。以下是一些示例:

Type integerType;
integerType = Type.forName('Integer');
integerType = Integer.class;

答案 1 :(得分:8)

今天早些时候有人问过我。这是一个更动态的解决方案。

MyClass mine     = new MyClass();
String myStr     = String.valueOf(mine);
String className = myStr.split(':')[0];
System.assertEquals('MyClass', className);

答案 2 :(得分:2)

Salesforce Apex的通用代码,用于解释:

public class Animal {}
public class Bird extends Animal {}

Animal a = new Bird();                      // Success
System.debug(a);                            // Bird:[]
System.debug((Bird)a);                      // Bird:[]
System.debug((Animal)a);                    // Bird:[]
System.debug(a instanceof Bird);            // true
System.debug((Animal)a instanceof Bird);    // true
System.debug((Bird)a instanceof Bird);      // Operation instanceof is always true since an instance of Bird is always an instance of Bird
System.debug((Bird)a instanceof Animal);    // Operation instanceof is always true since an instance of Bird is always an instance of Animal

答案 3 :(得分:2)

只需添加有关toString()String.valueOf()的更多信息。

不能依赖

String.valueOf()来获取运行时Apex类的名称。我不认为这是记录在案的,但如果您使用override关键字向自定义类添加toString()方法,则String.valueOf()将使用它来确定对象的字符串值。

class MyClassDefaultToString {
}

class MyClassOverrideToString {
    public override String toString() {
        return 'override toString';
    }
}

MyClassDefaultToString c1 = new MyClassDefaultToString();
System.assertEquals('MyClassDefaultToString:[]', String.valueOf(c1));

MyClassOverrideToString c2 = new MyClassOverrideToString();
System.assertNotEquals('MyClassOverrideToString:[]', String.valueOf(c2));
System.assertEquals('override toString', String.valueOf(c2));

答案 4 :(得分:0)

不理想,但这是一种完成这项工作的方法:

public String getClassType(Object o)
{
    String retVal = 'Unknown';

    if(o instanceof Klass1)
        retVal = 'Klass1';
    else if(o instanceof Klass2)
        retVal = 'Klass2';
    else if(o instanceof Klass3)
        retVal = 'Klass3';

    return retVal;
} 

答案 5 :(得分:0)

这对我有用

Map<String, Object> m =  (Map<String, Object>)JSON.deserializeUntyped(JSON.serialize(myObjectInstance));
Map<String, Object> mComplexProperty = (Map<String, Object>)m.get('complexProperty');
String mSimpleProperty = (String)m.get('simpleProperty');

我不确定JSON序列化和反序列化是否会产生开销,但......现在它正在工作

答案 6 :(得分:0)

如果您需要FQCN而不是类名(例如从Java的getName()获得的名称)怎么办?这是您可以做的。

public class Beasts {
  public interface Animal {}
  public virtual class Bird implements Animal {}
  public class Robin extends Bird {}
}

System.assertEquals(
  Robin.class,
  Type.forName(ObjUtil.getName(new Robin(), true))
);
System.assertEquals(
  Robin.class,
  Type.forName(ObjUtil.getName((Animal)new Robin(), true)))
);
System.assertEquals(
  Robin.class,
  Type.forName(ObjUtil.getName((Bird)new Robin(), true)))
);

public class ObjUtil {
  public static String getName(Object obj, Boolean isQualifiedName) {
    // Qualified name
    if (isQualifiedName) {
      try {
        if (0 == (Long) obj) {
        }
      } catch (Exception ex) {
        typeName = ex.getMessage().remove('Invalid conversion from runtime type ').remove(' to Long');
        System.debug('typeName = ' + typeName);
        return typeName;
      }
    } else {
      // Class name
      typeName = String.valueOf(obj);
      if (typeName.indexOf(':[') != -1) {
        typeName = typeName.split(':')[0];
        System.debug('typeName = ' + typeName);
        return typeName;
      }
    }
  }
}