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

com.feilong.lib.javassist.CtMember Maven / Gradle / Ivy

/*
 * Javassist, a Java-bytecode translator toolkit.
 * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved.
 *
 * The contents of this file are subject to the Mozilla Public License Version
 * 1.1 (the "License"); you may not use this file except in compliance with
 * the License.  Alternatively, the contents of this file may be used under
 * the terms of the GNU Lesser General Public License Version 2.1 or later,
 * or the Apache License Version 2.0.
 *
 * Software distributed under the License is distributed on an "AS IS" basis,
 * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
 * for the specific language governing rights and limitations under the
 * License.
 */

package com.feilong.lib.javassist;

/**
 * An instance of CtMember represents a field, a constructor,
 * or a method.
 */
public abstract class CtMember{

    CtMember          next;          // for internal use

    protected CtClass declaringClass;

    /*
     * Make a circular link of CtMembers declared in the
     * same class so that they are garbage-collected together
     * at the same time.
     */
    static class Cache extends CtMember{

        @Override
        protected void extendToString(StringBuffer buffer){
        }

        @Override
        public boolean hasAnnotation(String clz){
            return false;
        }

        @Override
        public Object getAnnotation(Class clz) throws ClassNotFoundException{
            return null;
        }

        @Override
        public Object[] getAnnotations() throws ClassNotFoundException{
            return null;
        }

        @Override
        public byte[] getAttribute(String name){
            return null;
        }

        @Override
        public Object[] getAvailableAnnotations(){
            return null;
        }

        @Override
        public int getModifiers(){
            return 0;
        }

        @Override
        public String getName(){
            return null;
        }

        @Override
        public String getSignature(){
            return null;
        }

        @Override
        public void setAttribute(String name,byte[] data){
        }

        @Override
        public void setModifiers(int mod){
        }

        @Override
        public String getGenericSignature(){
            return null;
        }

        @Override
        public void setGenericSignature(String sig){
        }

        private CtMember methodTail;

        private CtMember consTail;  // constructor tail

        private CtMember fieldTail;

        Cache(CtClassType decl){
            super(decl);
            methodTail = this;
            consTail = this;
            fieldTail = this;
            fieldTail.next = this;
        }

        CtMember methodHead(){
            return this;
        }

        CtMember lastMethod(){
            return methodTail;
        }

        CtMember consHead(){
            return methodTail;
        } // may include a static initializer

        CtMember lastCons(){
            return consTail;
        }

        CtMember fieldHead(){
            return consTail;
        }

        CtMember lastField(){
            return fieldTail;
        }

        void addMethod(CtMember method){
            method.next = methodTail.next;
            methodTail.next = method;
            if (methodTail == consTail){
                consTail = method;
                if (methodTail == fieldTail){
                    fieldTail = method;
                }
            }

            methodTail = method;
        }

        /*
         * Both constructors and a class initializer.
         */
        void addConstructor(CtMember cons){
            cons.next = consTail.next;
            consTail.next = cons;
            if (consTail == fieldTail){
                fieldTail = cons;
            }

            consTail = cons;
        }

        void addField(CtMember field){
            field.next = this; // or fieldTail.next
            fieldTail.next = field;
            fieldTail = field;
        }

        static int count(CtMember head,CtMember tail){
            int n = 0;
            while (head != tail){
                n++;
                head = head.next;
            }

            return n;
        }

        void remove(CtMember mem){
            CtMember m = this;
            CtMember node;
            while ((node = m.next) != this){
                if (node == mem){
                    m.next = node.next;
                    if (node == methodTail){
                        methodTail = m;
                    }

                    if (node == consTail){
                        consTail = m;
                    }

                    if (node == fieldTail){
                        fieldTail = m;
                    }

                    break;
                }
                m = m.next;
            }
        }
    }

    protected CtMember(CtClass clazz){
        declaringClass = clazz;
        next = null;
    }

    final CtMember next(){
        return next;
    }

    /**
     * This method is invoked when setName() or replaceClassName()
     * in CtClass is called.
     *
     * @see CtMethod#nameReplaced()
     */
    void nameReplaced(){
    }

    @Override
    public String toString(){
        StringBuffer buffer = new StringBuffer(getClass().getName());
        buffer.append("@");
        buffer.append(Integer.toHexString(hashCode()));
        buffer.append("[");
        buffer.append(Modifier.toString(getModifiers()));
        extendToString(buffer);
        buffer.append("]");
        return buffer.toString();
    }

    /**
     * Invoked by {@link #toString()} to add to the buffer and provide the
     * complete value. Subclasses should invoke this method, adding a
     * space before each token. The modifiers for the member are
     * provided first; subclasses should provide additional data such
     * as return type, field or method name, etc.
     */
    protected abstract void extendToString(StringBuffer buffer);

    /**
     * Returns the class that declares this member.
     */
    public CtClass getDeclaringClass(){
        return declaringClass;
    }

    /**
     * Returns true if this member is accessible from the given class.
     */
    public boolean visibleFrom(CtClass clazz){
        int mod = getModifiers();
        if (Modifier.isPublic(mod)){
            return true;
        }else if (Modifier.isPrivate(mod)){
            return clazz == declaringClass;
        }else{ // package or protected
            String declName = declaringClass.getPackageName();
            String fromName = clazz.getPackageName();
            boolean visible;
            if (declName == null){
                visible = fromName == null;
            }else{
                visible = declName.equals(fromName);
            }

            if (!visible && Modifier.isProtected(mod)){
                return clazz.subclassOf(declaringClass);
            }

            return visible;
        }
    }

    /**
     * Obtains the modifiers of the member.
     *
     * @return modifiers encoded with
     *         javassist.Modifier.
     * @see Modifier
     */
    public abstract int getModifiers();

    /**
     * Sets the encoded modifiers of the member.
     *
     * @see Modifier
     */
    public abstract void setModifiers(int mod);

    /**
     * Returns true if the class has the specified annotation type.
     *
     * @param clz
     *            the annotation type.
     * @return true if the annotation is found, otherwise false.
     * @since 3.11
     */
    public boolean hasAnnotation(Class clz){
        return hasAnnotation(clz.getName());
    }

    /**
     * Returns true if the class has the specified annotation type.
     *
     * @param annotationTypeName
     *            the name of annotation type.
     * @return true if the annotation is found, otherwise false.
     * @since 3.21
     */
    public abstract boolean hasAnnotation(String annotationTypeName);

    /**
     * Returns the annotation if the class has the specified annotation type.
     * For example, if an annotation @Author is associated
     * with this member, an Author object is returned.
     * The member values can be obtained by calling methods on
     * the Author object.
     *
     * @param annotationType
     *            the annotation type.
     * @return the annotation if found, otherwise null.
     * @since 3.11
     */
    public abstract Object getAnnotation(Class annotationType) throws ClassNotFoundException;

    /**
     * Returns the annotations associated with this member.
     * For example, if an annotation @Author is associated
     * with this member, the returned array contains an Author
     * object. The member values can be obtained by calling methods on
     * the Author object.
     *
     * @return an array of annotation-type objects.
     * @see CtClass#getAnnotations()
     */
    public abstract Object[] getAnnotations() throws ClassNotFoundException;

    /**
     * Returns the annotations associated with this member.
     * This method is equivalent to getAnnotations()
     * except that, if any annotations are not on the classpath,
     * they are not included in the returned array.
     *
     * @return an array of annotation-type objects.
     * @see #getAnnotations()
     * @see CtClass#getAvailableAnnotations()
     * @since 3.3
     */
    public abstract Object[] getAvailableAnnotations();

    /**
     * Obtains the name of the member.
     *
     * 

* As for constructor names, see getName() * in CtConstructor. * * @see CtConstructor#getName() */ public abstract String getName(); /** * Returns the character string representing the signature of the member. * If two members have the same signature (parameter types etc.), * getSignature() returns the same string. */ public abstract String getSignature(); /** * Returns the generic signature of the member. * * @see com.feilong.lib.javassist.bytecode.SignatureAttribute#toFieldSignature(String) * @see com.feilong.lib.javassist.bytecode.SignatureAttribute#toMethodSignature(String) * @see CtClass#getGenericSignature() * @since 3.17 */ public abstract String getGenericSignature(); /** * Sets the generic signature of the member. * * @param sig * a new generic signature. * @see com.feilong.lib.javassist.bytecode.SignatureAttribute.ObjectType#encode() * @see com.feilong.lib.javassist.bytecode.SignatureAttribute.MethodSignature#encode() * @see CtClass#setGenericSignature(String) * @since 3.17 */ public abstract void setGenericSignature(String sig); /** * Obtains a user-defined attribute with the given name. * If that attribute is not found in the class file, this * method returns null. * *

* Note that an attribute is a data block specified by * the class file format. * See {@link com.feilong.lib.javassist.bytecode.AttributeInfo}. * * @param name * attribute name */ public abstract byte[] getAttribute(String name); /** * Adds a user-defined attribute. The attribute is saved in the class file. * *

* Note that an attribute is a data block specified by * the class file format. * See {@link com.feilong.lib.javassist.bytecode.AttributeInfo}. * * @param name * attribute name * @param data * attribute value */ public abstract void setAttribute(String name,byte[] data); }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy