Java: Как получить объект класса текущего класса из статического контекста?

У меня есть функция регистрации, которая принимает вызывающий объект в качестве параметра. Затем я вызываю getClass(). GetSimpleName(), чтобы я мог легко получить имя класса для добавления в мою запись в журнале для удобства. Проблема в том, что когда я вызываю свою функцию журнала из статического метода, я не могу пройти в "this". Моя функция журнала выглядит примерно так:

public static void log(Object o, String msg){
 do_log(o.getClass().getSimpleName()+" "+msg);
}
public void do_something(){
 log(this, "Some message");
}

Но позвольте сказать, что я хочу войти в статическую функцию:

public static void do_something_static(){
 log(this, "Some message from static");
}

Очевидно, do_something_static() не будет работать, потому что он статичен, а "this" не находится в статическом контексте. Как я могу обойти это? И могу ли я сделать это без использования рефлексии (поскольку я понимаю, что есть много накладных расходов, и это может повлиять на производительность, так как я регистрирую много данных)

Я знаю, что могу, возможно, каким-то образом жестко закодировать текущий класс в вызове, но я уверен, что когда я переведу функцию в другой класс, я забуду обновить жестко закодированную ссылку, и она больше не будет правильный.

Спасибо!

8 ответов

Вы можете добавить "Класс" в качестве первого параметра и перегрузить метод журнала:

public class SomeClass {
 // Usage test:
 public static void main( String [] args ) {
 log( SomeClass.class, "Hola" );
 log( new java.util.Date(), "Hola" );
 }
 // Object version would call Class method... 
 public static void log( Object o , String msg ) {
 log( o.getClass(), msg );
 }
 public static void log( Class c , String message ) {
 System.out.println( c.getSimpleName() + " " + message );
 }
}

Вывод:

$ java SomeClass
SomeClass Hola
Date Hola

Но просто плохо, если вызывающий класс передан в качестве первого аргумента. Здесь, где объектно-ориентированная модель вступает в игру как противоположность "процедурного" стиля.

Вы можете получить вызывающий класс с помощью stacktrace, но, как вы уже сказали, накладные расходы будут вызваны тысячами раз.

Но, если вы создаете это как переменную класса, тогда будет только один экземпляр класса, если у вас будет 1000 классов, использующих эту утилиту, у вас будет не более 1000 вызовов.

Что-то вроде этого было бы лучше (тонкая вариация этого другого ответа):

public class LogUtility {
 private final String loggingFrom;
 public static LogUtility getLogger() {
 StackTraceElement [] s = new RuntimeException().getStackTrace();
 return new LogUtility( s[1].getClassName() );
 }
 private LogUtility( String loggingClassName ) {
 this.loggingFrom = "("+loggingClassName+") ";
 }
 public void log( String message ) {
 System.out.println( loggingFrom + message );
 }
}

Тест использования:

class UsageClass {
 private static final LogUtility logger = LogUtility.getLogger();
 public static void main( String [] args ) {
 UsageClass usageClass = new UsageClass();
 usageClass.methodOne();
 usageClass.methodTwo();
 usageClass.methodThree();
 }
 private void methodOne() {
 logger.log("One");
 }
 private void methodTwo() {
 logger.log("Two");
 }
 private void methodThree() {
 logger.log("Three");
 }
}

Вывод

$ java UsageClass
(UsageClass) One
(UsageClass) Two
(UsageClass) Three

Обратите внимание на объявление:

....
class UsageClass {
 // This is invoked only once. When the class is first loaded.
 private static final LogUtility logger = LogUtility.getLogger();
....

Таким образом, не имеет значения, если вы используете "регистратор" из объектов A, objectB, objectC или из метода класса (например, main), у всех их будет один экземпляр регистратора.


запрос к stacktrace может быть суровым для вашей проблемы. У вас есть 3 возможных подхода.

Exception # getStackTrace()

Просто создайте экземпляр исключения и получите первый кадр:

static String className() {
 return new RuntimeException().getStackTrace()[0].getClassName();
}

Тема

Использование Thread еще проще:

static String className2() {
 return Thread.currentThread().getStackTrace()[1].getClassName();
}

эти два подхода имеют тот недостаток, что вы не можете их повторно использовать. Таким образом, вы можете определить для него класс-помощник:

class Helper {
 public static String getClassName() {
 return Thread.currentThread().getStackTrace()[2].getClassName();
 }
}

теперь вы можете получить свое имя класса программно, без жесткого кодирования имени класса:

public static void log(Object o, String msg){
 do_log(Helper.getCClassName() + " " + msg);
}


Изменить метод log на:

public static void log(Class c, String msg){
 do_log(c.getSimpleName()+" "+msg);
}

и если do_something_static находится в классе MyClassWithStatics, то do_something_static станет:

public static void do_something_static(){
 log(MyClassWithStatics.class, "Some message from static");
}


Вместо "this" используйте "MyClass.class" и пусть ваш метод журнала обрабатывает объекты класса без getClass().

Но вместо того, чтобы делать это самостоятельно, подумайте над тем, чтобы позволить фреймворку log сделать это.


В краткосрочной перспективе вы должны жестко закодировать класс в вызове. Если вы считаете, что это понадобится вам в разных местах вашего проекта, так как это static, вы должны инкапсулировать его и заставить класс воспринимать как параметр.

public static void do_something_static(Class callingClass){
 log(callingClass, "Some message from static");
}


Не выполнил бы какой-то Singleton делать то, что вам нужно?

public class LogUtility {
 private final String loggingFrom;
 private static LogUtility instance;
 public static LogUtility getLogger() {
 if(instance == null)
 this.instance = new LogUtility();
 StackTraceElement [] s = new RuntimeException().getStackTrace();
 this.instance.setLoggingFrom(s[1].getClassName())
 return this.instance;
 }
 private LogUtility() {}
 private void setLoggingFrom(String loggingClassName){
 this.loggingFrom = loggingClassName;
 }
 public void log( String message ) {
 System.out.println( loggingFrom + message );
 }
}

Использование (в любом месте вашего проекта):

LogUtility.getLogger().log("Message");


Ну, если вы действительно не хотите, чтобы hardcode что-то вроде ClassName.class, то вы можете попытаться вывести класс, пройдя стек. К счастью, кто-то уже сделал это:). Кроме того, рассмотрите возможность использования регистратора, который позволяет вам регистрировать что-либо без указания вызывающего объекта.


Замените свои статические функции на нестатические. Вместо

Utils.do_something(...)

делать

new Utils().do_something(...)

licensed under cc by-sa 3.0 with attribution.