为什么Mapper不用写实现类就能访问到数据库

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档


前言

提示:这里可以添加本文要记录的大概内容:
例如:随着人工智能的不断发展,机器学习这门技术也越来越重要,很多人都开启了学习机器学习,本文就介绍了机器学习的基础内容。


提示:以下是本篇文章正文内容,下面案例可供参考

一、动态代理

1. jdk动态代理有具体实现类

1.1 实现接口

package jdk_dynamic_proxy.with_target_impl;

public interface IUserDao {
    void saveUser();

    void sayHello();

    String returnValue();
}

1.2 具体实现类

package jdk_dynamic_proxy.with_target_impl;

public class UserDaoImpl implements IUserDao {
    @Override
    public void saveUser() {
        System.out.println("保存用户________");
    }

    @Override
    public void sayHello() {
        System.out.println("hello world!");
    }

    @Override
    public String returnValue() {
        return "I am return value.";
    }
}

1.3 代理有实现类的target

/**
 * 代理有实现类的target
 */
public class ProxyFactory {
    private Object target;
    public ProxyFactory(Object target) {
        this.target = target;
    }

    public Object getProxyInstance() {
        return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {


                // 当一个接口有多个方法时,此时可以通过method.getName()来判断是哪个方法,从而实现不同的代理逻辑
                if (method.getName().equals("sayHello")) {
                    System.out.println("sayHello前_______");
                    method.invoke(target, args);
                    System.out.println("sayHello后_______");
                } else if (method.getName().equals("saveUser")) {
                    System.out.println("saveUser前_______");
                    method.invoke(target, args);
                    System.out.println("saveUser后_______");
                // 动态代理可以修改除了前后增强外,还可以修改原有方法的返回值,利用这个特性可以在过滤器中修改编码,从而纠正中文乱码
                } else if (method.getName().equals("returnValue")) {
                    return "I am return value.-->> I am proxy return value.";
                }
                return null;
            }
        });
    }
}

1.4 测试

public class Main {
    public static void main(String[] args) {
        IUserDao userDaoProxy = (IUserDao) new ProxyFactory(new UserDaoImpl()).getProxyInstance();
        userDaoProxy.saveUser();
        userDaoProxy.sayHello();
        System.out.println(userDaoProxy.returnValue());

    }
}

2. jdk动态代理中无具体实现类

mybatis的mapper接口,还有dubbo的rpc调用使用的就是这种情况。

2.1 定义接口

public interface IUserDao {
    void saveUser();
}

2.2 定义无具体实现类的代理生产类

public class ProxyFactoryWithoutTargetImpl {
    private Class target;

    public ProxyFactoryWithoutTargetImpl(Class target) {
        this.target = target;
    }

    public Object getProxyInstance() {
        return Proxy.newProxyInstance(target.getClassLoader(), new Class[]{target}, new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                System.out.println("我仅仅代理了一个接口的方法,而没有代理有具体实现类的方法");
                return null;
            }
        });
    }
}

2.3 测试

public class Main {
    public static void main(String[] args) {
        IUserDao userDaoProxy = (IUserDao)new ProxyFactoryWithoutTargetImpl(IUserDao.class).getProxyInstance();
        userDaoProxy.saveUser();
    }
}

二、MyBatis中是如何使用的

三、doubbo中是如何使用的


参考文档

Mybaits 源码解析 (五)----- Mapper接口底层原理(为什么Mapper不用写实现类就能访问到数据库?)
为啥mybatis的mapper只有接口没有实现类,但它却能工作?
描述Java动态代理的几种实现方式,分别说出相应的优缺点。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值