All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.springframework.core.ControlFlowFactory Maven / Gradle / Ivy

There is a newer version: 5.3.34
Show newest version
/*
 * Copyright 2002-2006 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.springframework.core;

import java.io.PrintWriter;
import java.io.StringWriter;

import org.springframework.util.Assert;

/**
 * Static factory to conceal automatic choice of Java 1.4 or 1.3 ControlFlow
 * implementation class.
 *
 * 

We want to use the more efficient Java 1.4 StackTraceElement if we can, * but we don't want to impose a runtime dependency on JDK 1.4. * * @author Rod Johnson * @author Juergen Hoeller * @since 02.02.2004 */ public abstract class ControlFlowFactory { /** * Returns an appropriate {@link ControlFlow} instance. * @return an appropriate {@link ControlFlow} instance */ public static ControlFlow createControlFlow() { return JdkVersion.isAtLeastJava14() ? (ControlFlow) new Jdk14ControlFlow() : (ControlFlow) new Jdk13ControlFlow(); } /** * Utilities for cflow-style pointcuts. Note that such pointcuts are * 5-10 times more expensive to evaluate than other pointcuts, as they require * analysis of the stack trace (through constructing a new throwable). * However, they are useful in some cases. *

This implementation uses the StackTraceElement class introduced in Java 1.4. * @see java.lang.StackTraceElement */ static class Jdk14ControlFlow implements ControlFlow { private StackTraceElement[] stack; public Jdk14ControlFlow() { this.stack = new Throwable().getStackTrace(); } /** * Searches for class name match in a StackTraceElement. */ public boolean under(Class clazz) { Assert.notNull(clazz, "Class must not be null"); String className = clazz.getName(); for (int i = 0; i < stack.length; i++) { if (this.stack[i].getClassName().equals(className)) { return true; } } return false; } /** * Searches for class name match plus method name match * in a StackTraceElement. */ public boolean under(Class clazz, String methodName) { Assert.notNull(clazz, "Class must not be null"); Assert.notNull(methodName, "Method name must not be null"); String className = clazz.getName(); for (int i = 0; i < this.stack.length; i++) { if (this.stack[i].getClassName().equals(className) && this.stack[i].getMethodName().equals(methodName)) { return true; } } return false; } /** * Leave it up to the caller to decide what matches. * Caller must understand stack trace format, so there's less abstraction. */ public boolean underToken(String token) { if (token == null) { return false; } StringWriter sw = new StringWriter(); new Throwable().printStackTrace(new PrintWriter(sw)); String stackTrace = sw.toString(); return stackTrace.indexOf(token) != -1; } public String toString() { StringBuffer sb = new StringBuffer("Jdk14ControlFlow: "); for (int i = 0; i < this.stack.length; i++) { if (i > 0) { sb.append("\n\t@"); } sb.append(this.stack[i]); } return sb.toString(); } } /** * Java 1.3 version of utilities for cflow-style pointcuts. We can't * rely on the Java 1.4 StackTraceElement class. *

Note that such pointcuts are 10-15 times more expensive to evaluate under * JDK 1.3 than other pointcuts, as they require analysis of the stack trace * (through constructing a new throwable). However, they are useful in some cases. */ static class Jdk13ControlFlow implements ControlFlow { private final String stackTrace; private final int stackTraceLength; public Jdk13ControlFlow() { StringWriter sw = new StringWriter(); new Throwable().printStackTrace(new PrintWriter(sw)); this.stackTrace = sw.toString(); this.stackTraceLength = this.stackTrace.length(); } /** * Searches for class name match in the stringified stacktrace. */ public boolean under(Class clazz) { Assert.notNull(clazz, "Class must not be null"); return this.stackTrace.indexOf(clazz.getName()) != -1; } /** * Searches for class name + "." + method name match in the stringified * stacktrace. Checks the character right after the method name for '(' * or whitespace, to only match the exact method name (and not a method * with a longer name that happens to start with the same characters). *

The whitespace check has been introduced for compatibility with * GNU ClassPath, as of Spring 1.2.7. Sun JDKs (and JDKs with licensed * Sun core libraries) always append '(' right after the method name. */ public boolean under(Class clazz, String methodName) { Assert.notNull(clazz, "Class must not be null"); Assert.notNull(methodName, "Method name must not be null"); String searchPattern = clazz.getName() + "." + methodName; int patternLength = searchPattern.length(); int index = 0; do { index = this.stackTrace.indexOf(searchPattern, index); if (index != -1) { int endIndex = index + patternLength; if (endIndex == this.stackTraceLength) { return true; } char afterPattern = this.stackTrace.charAt(endIndex); if (afterPattern == '(' || Character.isWhitespace(afterPattern)) { return true; } index = endIndex; } } while (index != -1); return false; } /** * Leave it up to the caller to decide what matches. * Caller must understand stack trace format, so there's less abstraction. */ public boolean underToken(String token) { return (token != null && this.stackTrace.indexOf(token) != -1); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy