得到方法调用者和java中的位操作符

时间:2021-12-13 15:29:29

获取方法的调用者

如果你写了一个超级复杂超级大的项目,在项目后期调试修改的时候,突然想知道到底是哪些类调用了ImportantClass中的Important方法,你会怎么做呢?

首先,你可能说:我用不到!如果这样的话,到这里你就可以return了。
做法一:最常规有效的做法,使用IDE的全目录全文搜索方法名。当然这样是最有效快速的。但是如果有很多别的类中也定义了Important方法,你就会得到很多垃圾搜索结果。同时你只能搜索到目录下的结果,而对于运行时调用Important方法的地方你就无所适从了。

做法二:这就是我要说的,编写程序,获取方法调用者。最直观的例子就是我们每天都在用了log4j。在执行log.debug或者别的输出日至的操作的时候,它都会按照设定的格式输出类名、方法名、甚至所在的行数。如果我们在Important方法中插入这么一段,就可以知道到底是谁在运行期调用了Important方法了。

在demo程序中有详细的注释和一个完整的例子。我们知道把那段代码拷贝到关系的程序中,修改一下fullClassName就可以了。 

得到方法调用者和java中的位操作符得到方法调用者和java中的位操作符/**
得到方法调用者和java中的位操作符 * 
得到方法调用者和java中的位操作符 
*/

得到方法调用者和java中的位操作符
得到方法调用者和java中的位操作符得到方法调用者和java中的位操作符
/**
得到方法调用者和java中的位操作符 * 
@author zangmeng
得到方法调用者和java中的位操作符 * 
得到方法调用者和java中的位操作符 * 
得到方法调用者和java中的位操作符 * 如果我们注意观察一下异常时的输出,我们就知道java调用栈的结构了
得到方法调用者和java中的位操作符 * 最后被调用的方法在栈顶。
得到方法调用者和java中的位操作符 * 我们这个方法的工作过程就是首先得到调用栈,然后从栈顶向下搜索
得到方法调用者和java中的位操作符 * 直到搜索到我们关心的Important类。
得到方法调用者和java中的位操作符 * 然后,在这个类的方法后面的,就是我们关心的Important类的important方法的调用者了
得到方法调用者和java中的位操作符 * 这时候我们检测第一个不是Important类的类,则这个类必定是调用Important类中
得到方法调用者和java中的位操作符 * important方法的类和方法。
得到方法调用者和java中的位操作符 * 然后我们将之打印出来,就可以了。
得到方法调用者和java中的位操作符 * 同样的道理,我们也可以得到更上一层的调用关系。
得到方法调用者和java中的位操作符 * 一直追述到线程的启动方法。
得到方法调用者和java中的位操作符 * 
得到方法调用者和java中的位操作符 
*/

得到方法调用者和java中的位操作符得到方法调用者和java中的位操作符
public class Important {
得到方法调用者和java中的位操作符得到方法调用者和java中的位操作符    
public void important() {
得到方法调用者和java中的位操作符        String fullClassName 
= "Important";
得到方法调用者和java中的位操作符        
// 首先得到调用栈
得到方法调用者和java中的位操作符
        StackTraceElement stack[] = (new Throwable()).getStackTrace();
得到方法调用者和java中的位操作符        
// 然后从栈中向上搜索,直到搜索到我们的Important类。
得到方法调用者和java中的位操作符
        int ix = 0;
得到方法调用者和java中的位操作符得到方法调用者和java中的位操作符        
while (ix < stack.length) {
得到方法调用者和java中的位操作符            StackTraceElement frame 
= stack[ix];
得到方法调用者和java中的位操作符            String cname 
= frame.getClassName();
得到方法调用者和java中的位操作符得到方法调用者和java中的位操作符            
if (cname.equals(fullClassName)) {
得到方法调用者和java中的位操作符                
break;
得到方法调用者和java中的位操作符            }

得到方法调用者和java中的位操作符            ix
++;
得到方法调用者和java中的位操作符        }

得到方法调用者和java中的位操作符        
// 此时ic位置放置的是Important类。
得到方法调用者和java中的位操作符得到方法调用者和java中的位操作符
        while (ix < stack.length) {
得到方法调用者和java中的位操作符            StackTraceElement frame 
= stack[ix];
得到方法调用者和java中的位操作符            String cname 
= frame.getClassName();
得到方法调用者和java中的位操作符得到方法调用者和java中的位操作符            
if (!cname.equals(fullClassName)) {
得到方法调用者和java中的位操作符                
//第一个费Important类的类就是调用者
得到方法调用者和java中的位操作符
                System.out.println("调用者此方法的类名:"+cname);
得到方法调用者和java中的位操作符                System.out.println(
""+cname+"中调用次方法的方法名:"+frame.getMethodName());
得到方法调用者和java中的位操作符            }

得到方法调用者和java中的位操作符            ix
++;
得到方法调用者和java中的位操作符        }

得到方法调用者和java中的位操作符得到方法调用者和java中的位操作符        
/**
得到方法调用者和java中的位操作符         * some important operation is followed :)
得到方法调用者和java中的位操作符         
*/

得到方法调用者和java中的位操作符
得到方法调用者和java中的位操作符        System.out.println(
"__This is an important method,and I care for the infor of the invoker!__");
得到方法调用者和java中的位操作符    }

得到方法调用者和java中的位操作符}

得到方法调用者和java中的位操作符

 

java中的位操作符


因为不常用,所以经常忘记。索性抽时间总结一下,备忘。
对于java这种语言来说,位操作符比较少用到
&:按位与。
|:按位或。
^:按位异或。
~:按位取反。
<<:左移,高位插0,否则插1;
>>:右移,高位插0,否则插1;
>>>:无符号右移,无论正负,高位插0
令人奇怪的是java似乎好像大概说不定没有同或操作符,why?莫非是想让我们把~和^组合一下代替同或操作符?
http://www.matrix.org.cn/resource/article/43/43978_Java_Bitfields_Bitboards.html 这篇文章讲的是相当不错,看了个开头,感觉很有水平,不过篇幅太长,不宜上班时阅读,留作备忘以后慢慢读之。