• 面试官:private 方法可以被代理吗?别答错了。。


    JDK动态代理中:抽象类不可以被代理 没有接口不可以被代理

    • 在 Java 8之前,接口可以有常量变量和抽象方法。我们不能在接口中提供方法实现。如果我们要提供抽象方法和非抽象方法(方法与实现)的组合,那么我们就得使用抽象类。
    • 在 Java 8 接口引入了一些新功能——默认方法和静态方法。
    • Java 9 不仅像 Java 8 一样支持接口默认方法,同时还支持私有方法
    • jdk1.9接口中可以有static、default、private方法、private static方法 同时他们都必须有方法体

    接口的静态方法能被重写或实现吗?

    不可被重写 不可被代理

    静态方法是属于类的,不属于对象,类哪有重写或实现,所以在其实现类里当然不能重写或实现 在代理中也不会有此方法,userProxy里面都没有,更别说被代理了,那是不可能的

    defaultMethod可以被代理吗?

    可以被重写(不强制),也可以被代理

    privateMethod可以被代理吗?

    不可被重写 不可被代理

    只能在接口自己内部调用,,都没有权限访问它当然不能被代理了,所以动态代理中根本不会有这个私有方法

    当然为了证明这一点可以 使用接口的公共方法来间接调用它,看它到底有没有被代理, 而接口中可以有方法体的只有static、default、private方法,而static又不可以(static不可以调用非static方法),别无选择,只能用default测了 发现private并没有被代理

    protected 、package下面方法可以被代理吗?

    这些修饰符在接口中都是不允许的,接口中都没有,哪来的代理啊

    总结

    综上所述只有public和default方法可以被代理,private和static都不可以

    • public可重写可被代理(在实现类中重写是强制的)
    • private和static不可以被重写也不可以被代理
    • default可重写可代理(在实现类中不强制必须重写,不过不管重写不重写都会自动被代理)
    public interface MapperInterface {
        static void staticMethod() {
            System.out.println("static bb");
        }
     
        default void defaultMethod() {
            privateMethod();
        }
     
        private static void privateMethod() {
            System.out.println("MapperInterface private");
        }
     
        void publicMethod();
    }
    public class DynamicProxy {
     
        public static <T> T getProxy(T t) {
            return (T) Proxy.newProxyInstance(t.getClass().getClassLoader(), t.getClass().getInterfaces(), new InvocationHandler() {
                @Override
                public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
     
                    //调用目标方法之前做的公共逻辑
                    System.out.println("建立连接");
                    //调用目标方法,这句话必须要有的,就是调用被传入的对象的方法(里面写了核心代码)
                    Object object = method.invoke(t, args);
                    //调用目标方法之后做的公共逻辑
                    System.out.println("提交事务");
                    System.out.println("关闭连接");
                    return object;
                }
            });
        }
    }
    public class User implements MapperInterface {
     
     
        @Override
        public void publicMethod() {
            System.out.println("user public");
        }
     
    // 可以被重写 不强制,不管有没有重写 都可以被代理
    //    @Override
    //    public void defaultMethod() {
    //        System.out.println("user default");
    //    }
    }
    public class Test {
        public static void main(String[] args) {
            User user = new User();
            MapperInterface userProxy = DynamicProxy.getProxy(user);
            userProxy.defaultMethod();
        }
    }
    

    来源:blog.csdn.net/weixin_39143647/article/details/118485683

    近期热文推荐:

    1.1,000+ 道 Java面试题及答案整理(2022最新版)

    2.劲爆!Java 协程要来了。。。

    3.Spring Boot 2.x 教程,太全了!

    4.别再写满屏的爆爆爆炸类了,试试装饰器模式,这才是优雅的方式!!

    5.《Java开发手册(嵩山版)》最新发布,速速下载!

    觉得不错,别忘了随手点赞+转发哦!

  • 相关阅读:
    node起本地服务器以及实现代理,前端接口转发
    一键前端代理,一行命令开启nginx容器,代理前端页面
    go语言学习笔记
    patch需要数据格式前端算法,patch算法基础,两个对象对比取差异属性
    react-native中使用Echarts,自己使用WebView封装Echarts经验
    如何用js自己实现Animate运动函数
    vue中的表单异步校验方法封装
    Entity Framework6使用SQL Server Compact免安装部署
    WCF异常传播
    解决.net的堆碎片化带来的内存占用过大的问题
  • 原文地址:https://www.cnblogs.com/javastack/p/16169813.html
Copyright © 2020-2023  润新知