搭建Tomcat(四)---Servlet容器

目录

引入

Servlet容器

一、优化MyTomcat

①先将MyTomcat的main函数搬过来:

②将getClass()函数搬过来

 ③创建容器

 ④连接ServletConfigMapping和MyTomcat

连接:

⑤完整的ServletConfigMapping和MyTomcat方法:

a.ServletConfigMapping

b.MyTomcat

二、优化Server

 方法1:调用(可以用,但是这里咱们不用,过于冗余)

方法二: 直接嵌套

三、获取类对象

处理请求:

实现上述操作后,就可以测试是否正确:

本地的Key值是 “/myFrist”

去客户端(浏览器)输入试试:

结果:


引入

在先前的tomcat搭建学习中,已经对tomcat的雏形做了基本的实现,即如下的过程:

接下来继续tomcat的搭建。

Servlet容器

目前Servlet容器:

 接下来,我们来优化一下MyTomcat:

一、优化MyTomcat

首先在tomcat包下创建一下config包,然后在里面创建文件ServletConfigMapping:

那么ServletConfigMapping里面编写什么呢?就是从MyTomcat中编写的内容,现在需要copy到这个文件中,实现MyTomcat的解放:

【ServletConfigMapping在tomcat中就是为了获取配置信息;而MyTomcat中写的,正是扫描的过程,即未来扫描动态资源映射表的过程,所以现在将它放进合适的地方,即ServletConfigMapping里面】

注意: 不是原封不动的照搬,需要做一些小小的调整。

①先将MyTomcat的main函数搬过来:

注意这里做了个小调整,将MyTomcat的main函数里面的内容放进了代码块中;

static代码块的特点:在main函数执行之前先执行

static{
        try {
            // 1. 扫描包路径 (com.wzh.tomcat.myweb)
            String packageName = "com.qcby.tomcat.myweb";
            List<Class<?>> classes = getClasses(packageName);   //通过getClasses()方法获取到了myweb这个包下面的所有类的类对象,并将其放到了类对象中

            // 2. 遍历所有类,检查是否有@WebServlet注解
            for (Class<?> clazz : classes) {
                if (clazz.isAnnotationPresent(WebServlet.class)) {
                    // 3. 获取@WebServlet注解的值
                    WebServlet webServlet = clazz.getAnnotation(WebServlet.class);
                    System.out.println("类名: " + clazz.getName() + " | URL路径: " + webServlet.url());
                    classMap.put(webServlet.url(),(Class<HttpServlet>) clazz);

                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

②将getClass()函数搬过来

 这里直接copy就好,放到刚才的static代码块之下:

/**
     * 获取指定包下的所有类
     *
     * @param packageName 包名,例如 "com.qcby.tomcat.myweb"
     * @return 类对象列表
     * @throws Exception
     */
    //将MyTomcat中的getClass方法也粘贴过来
    private static List<Class<?>> getClasses(String packageName) throws Exception {
        List<Class<?>> classes = new ArrayList<>(); //将类文件封装进List中
        String path = packageName.replace('.', '/'); // 将包名转换为文件路径

        // 通过类加载器获取包的资源路径
        ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
        Enumeration<URL> resources = classLoader.getResources(path);

        while (resources.hasMoreElements()) {
            URL resource = resources.nextElement();
            File directory = new File(resource.toURI());

            // 扫描文件夹下的所有类文件
            if (directory.exists()) {
                for (File file : directory.listFiles()) {
                    if (file.getName().endsWith(".class")) {    //获得.class文件
                        // 获取类的完整类名
                        String className = packageName + "." + file.getName().replace(".class", "");
                        classes.add(Class.forName(className));
                    }
                }
            }
        }
        return classes;
    }

 基于static代码块会在main方法之前执行的特性,想要执行static中的内容,只需要放一个空的main函数运行即可:

public static void main(String[] args) {
        //当然也可以直接将static中的内容放进main函数里面(不常用)
  }

 ③创建容器

真正的servlet容器:

public static Map<String,Class<HttpServlet>> classMap=new HashMap<>();

/*
* 至于这里value的位置为什么写HttpServlet呢?按理说应该写类对象----多态
* 多态--父类的引用指向子类的对象
* 不妨来看一下,能写进来的MyFirstServlet、MySecondServlet、MyThirdServlet等的类对象
* 这些类对象都继承了HttpServlet
* 即:
* MyFirstServlet的类对象--->是HttpServlet的子类
* MySecondServlet的类对象--->是HttpServlet的子类
* MyThirdServlet的类对象--->是HttpServlet的子类
*【多态性--父类的引用指向子类的对象,并且子类的对象可以向上转型为父类的引用】
* */

 ④连接ServletConfigMapping和MyTomcat

我们只是想优化MyTomcat,而不是彻底换掉MyTomcat,所以现在我们要做的就是将创建的ServletConfigMapping和先前的MyTomcat连接起来:

当然由于主要代码已经放进ServletConfigMapping中了,所以原先的MyTomcat可以清空了。

连接:

注意②中的执行方法---放main;那么想要连接ServletConfigMapping和MyTomcat,我们依旧可以采用这种方法:让MyTomcat调用ServletConfigMapping中的方法,迫使ServletConfigMapping中的static代码块运行。

public class MyTomcat {
    public static void main(String[] args) {
        ServletConfigMapping.init();
        //通过调用ServletConfigMapping中的静态方法来促使ServletConfigMapping类的执行,即将这两个类通过这种方式连接
    }
}

当然这个写在ServletConfigMapping中的方法,可以是空的。

⑤完整的ServletConfigMapping和MyTomcat方法:

a.ServletConfigMapping
package com.qcby.tomcat.config;

import com.qcby.tomcat.HttpServlet.HttpServlet;
import com.qcby.tomcat.webservlet.WebServlet;

import java.io.File;
import java.net.URL;
import java.util.*;

/*
* servlet容器
* */


public class ServletConfigMapping {

    //真正的servlet容器
    /*
    * 至于这里value的位置为什么写HttpServlet呢?按理说应该写类对象----多态
    * 多态--父类的引用指向子类的对象
    * 不妨来看一下,能写进来的MyFirstServlet、MySecondServlet、MyThirdServlet等的类对象
    * 这些类对象都继承了HttpServlet
    * 即:
    * MyFirstServlet的类对象--->是HttpServlet的子类
    * MySecondServlet的类对象--->是HttpServlet的子类
    * MyThirdServlet的类对象--->是HttpServlet的子类
    *【多态性--父类的引用指向子类的对象,并且子类的对象可以向上转型为父类的引用】
    * */
    public static Map<String,Class<HttpServlet>> classMap=new HashMap<>();



    //扫描遍历(即MyTomcat之中的内容)
    //static代码块在main方法执行之前执行
    //将MyTomcat中的main方法放进代码块中
    static{
        try {
            // 1. 扫描包路径 (com.wzh.tomcat.myweb)
            String packageName = "com.qcby.tomcat.myweb";
            List<Class<?>> classes = getClasses(packageName);   //通过getClasses()方法获取到了myweb这个包下面的所有类的类对象,并将其放到了类对象中

            // 2. 遍历所有类,检查是否有@WebServlet注解
            for (Class<?> clazz : classes) {
                if (clazz.isAnnotationPresent(WebServlet.class)) {
                    // 3. 获取@WebServlet注解的值
                    WebServlet webServlet = clazz.getAnnotation(WebServlet.class);
                    System.out.println("类名: " + clazz.getName() + " | URL路径: " + webServlet.url());
                    classMap.put(webServlet.url(),(Class<HttpServlet>) clazz);

                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /**
     * 获取指定包下的所有类
     *
     * @param packageName 包名,例如 "com.qcby.tomcat.myweb"
     * @return 类对象列表
     * @throws Exception
     */
    //将MyTomcat中的getClass方法也粘贴过来
    private static List<Class<?>> getClasses(String packageName) throws Exception {
        List<Class<?>> classes = new ArrayList<>(); //将类文件封装进List中
        String path = packageName.replace('.', '/'); // 将包名转换为文件路径

        // 通过类加载器获取包的资源路径
        ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
        Enumeration<URL> resources = classLoader.getResources(path);

        while (resources.hasMoreElements()) {
            URL resource = resources.nextElement();
            File directory = new File(resource.toURI());

            // 扫描文件夹下的所有类文件
            if (directory.exists()) {
                for (File file : directory.listFiles()) {
                    if (file.getName().endsWith(".class")) {    //获得.class文件
                        // 获取类的完整类名
                        String className = packageName + "." + file.getName().replace(".class", "");
                        classes.add(Class.forName(className));
                    }
                }
            }
        }
        return classes;
    }

    //基于static代码块会在main方法之前执行的特性,想要执行static中的内容,只需要放一个空的main函数运行即可
    public static void main(String[] args) {
        //当然也可以直接将static中的内容放进main函数里面(不常用)
    }

    public static void init(){

    }

}
b.MyTomcat
package com.qcby.tomcat;


import com.qcby.tomcat.config.ServletConfigMapping;

public class MyTomcat {
    public static void main(String[] args) {
        ServletConfigMapping.init();//通过调用ServletConfigMapping中的静态方法来促使ServletConfigMapping类的执行,即将这两个类通过这种方式连接

    }
}

二、优化Server

用同样的方法,首先把server的main函数解放掉:

【将main函数中的内容取出,并放进新创的serverInit函数中,删除原来的main函数即可。】

//优化Server,取出main方法
    public static void serverInit() throws Exception {
        // 1.打开通信端口   tomcat:8080   3306  ---------》进行网络通信
        ServerSocket serverSocket = new ServerSocket(8080);
        System.out.println("****************server start.....");

        //2.接受请求数据
        while (true) {
            Socket socket = serverSocket.accept();  //--------------------->注意:此时监听网卡的是:主线程
            System.out.println("有客户进行了链接");
            new Thread(() -> {
                //处理数据---------》数据的处理在于读和写
                try {
                    handler(socket);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }).start();
        }
    }

 方法1:调用(可以用,但是这里咱们不用,过于冗余)

接着用同样的操作,在MyTomcat中调用这新创建的方法:

package com.qcby.tomcat;

import com.qcby.tomcat.config.ServletConfigMapping;
import com.qcby.tomcat.socket.Server;

public class MyTomcat {
    public static void main(String[] args) {
        try {
            //通过调用ServletConfigMapping中的静态方法来促使ServletConfigMapping类的执行,即将这两个类通过这种方式连接
            ServletConfigMapping.init();//初始化servlet容器
            Server.serverInit();//启动server服务
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

【看着代码变多了很多,实际上就加了一行,然后为了安全性,抛出了异常】

方法二: 直接嵌套

这里我们用另一种方法去优化Server,即将Server和MyTomcat连接起来:

即直接将方才做过修改的代码粘贴到MyTomcat中,并且在MyTomcat的main函数中调用Server的serverInit(即过去Server类的main函数)

如下是这一小阶段完成后的MyTomcat代码示意:

package com.qcby.tomcat;

import com.qcby.tomcat.HttpServlet.HttpServlet;
import com.qcby.tomcat.Request.Request;
import com.qcby.tomcat.Response.Response;
import com.qcby.tomcat.config.ServletConfigMapping;
import com.qcby.tomcat.socket.Server;

import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;

public class MyTomcat {
    //实例化Request
    public static Request request = new Request();
    public static void main(String[] args) throws Exception {
        ServletConfigMapping.init();
        serverInit();
    }

    //优化Server,取出main方法
    public static void serverInit() throws Exception {
        // 1.打开通信端口   tomcat:8080   3306  ---------》进行网络通信
        ServerSocket serverSocket = new ServerSocket(8080);
        System.out.println("****************server start.....");

        //2.接受请求数据
        while (true) {
            Socket socket = serverSocket.accept();  //--------------------->注意:此时监听网卡的是:主线程
            System.out.println("有客户进行了链接");
            new Thread(() -> {
                //处理数据---------》数据的处理在于读和写
                try {
                    handler(socket);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }).start();
        }
    }

    public static void handler(Socket socket) throws Exception {
        //读取请求的数据
        InputStream inputStream = socket.getInputStream();
        requestContext(inputStream);
    }

    public static void requestContext(InputStream inputStream) throws IOException { //获取全部信息
        //将bit流转为文字信息
        int count = 0;
        while (count == 0) {
            count = inputStream.available();
        }
        byte[] bytes = new byte[count];
        inputStream.read(bytes);
        String Context = new String(bytes);
        System.out.println(Context);

        //解析数据
        if (Context.equals("")) {
            System.out.println("你输入了一个空请求");
        } else {
            String firstLine = Context.split("\\n")[0];
            String method = firstLine.split("\\s")[0];
            String path = firstLine.split("\\s")[1];
            System.out.println(method + " " + path);
            //任何请求都会被打到这个类中,随后就会被解析
            //将解析后的数据(method和path放进申请的static的Request实例中--再被运输给其他需要的地方)
            request.setMethod(method);
            request.setPath(path);
        }
    }
}

三、获取类对象

前面已经对MyTomcat做了优化,下一步就是去获取类对象;

如何获取类对象呢?简单来说谁存储着类对象的信息呢?---Request

那么就要接着对server类做出优化---因为目前Server和MyTomcat类还是存在问题,两者打不通;

既然想把两者联系起来,那么就要有嵌套调用或是直接嵌套。

在这里,我们依旧是用直接嵌套的方法写:

处理请求:

处理接收到的请求,首先在Map映射中寻找是否能够匹配上key值,即传送的path值

能查到则取获取Map的value值,即类对象: 

 //处理请求
    public static void dis(Request request) throws Exception {
        if(!request.getPath().equals("")){
            if(ServletConfigMapping.classMap.get(request.getPath())!=null){
                //能进到这里,则说明能够获取Map中对应的key值,能够匹配上
                Class<HttpServlet> classServlet=ServletConfigMapping.classMap.get(request.getPath());
                HttpServlet servlet=classServlet.newInstance(); //newInstance动态地创建一个类的新实例(对象)
                servlet.doGet(request,response);

            }
        }
    }

上面的代码最终是要被requestContext(InputStream inputStream)函数调用的。

即最终的MyTomcat代码:

package com.qcby.tomcat;

import com.qcby.tomcat.HttpServlet.HttpServlet;
import com.qcby.tomcat.Request.Request;
import com.qcby.tomcat.Response.Response;
import com.qcby.tomcat.config.ServletConfigMapping;
import com.qcby.tomcat.socket.Server;

import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;

public class MyTomcat {
    //实例化Request
    public static Request request = new Request();
    public static Response response = new Response();
    public static void main(String[] args) throws Exception {
        //两项调用连接
        ServletConfigMapping.init();
        serverInit();
    }


    //处理请求
    public static void dis(Request request) throws Exception {
        if(!request.getPath().equals("")){
            if(ServletConfigMapping.classMap.get(request.getPath())!=null){
                //能进到这里,则说明能够获取Map中对应的key值,能够匹配上
                Class<HttpServlet> classServlet=ServletConfigMapping.classMap.get(request.getPath());
                HttpServlet servlet=classServlet.newInstance(); //newInstance动态地创建一个类的新实例(对象)
                servlet.doGet(request,response);

            }
        }
    }

    //优化Server,取出main方法
    public static void serverInit() throws Exception {
        // 1.打开通信端口   tomcat:8080   3306  ---------》进行网络通信
        ServerSocket serverSocket = new ServerSocket(8080);
        System.out.println("****************server start.....");

        //2.接受请求数据
        while (true) {
            Socket socket = serverSocket.accept();  //--------------------->注意:此时监听网卡的是:主线程
            System.out.println("有客户进行了链接");
            new Thread(() -> {
                //处理数据---------》数据的处理在于读和写
                try {
                    handler(socket);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }).start();
        }
    }

    public static void handler(Socket socket) throws Exception {
        //读取请求的数据
        InputStream inputStream = socket.getInputStream();
        requestContext(inputStream);
    }

    public static void requestContext(InputStream inputStream) throws Exception { //获取全部信息
        //将bit流转为文字信息
        int count = 0;
        while (count == 0) {
            count = inputStream.available();
        }
        byte[] bytes = new byte[count];
        inputStream.read(bytes);
        String Context = new String(bytes);
        System.out.println(Context);

        //解析数据
        if (Context.equals("")) {
            System.out.println("你输入了一个空请求");
        } else {
            String firstLine = Context.split("\\n")[0];
            String method = firstLine.split("\\s")[0];
            String path = firstLine.split("\\s")[1];
            System.out.println(method + " " + path);
            //任何请求都会被打到这个类中,随后就会被解析
            //将解析后的数据(method和path放进申请的static的Request实例中--再被运输给其他需要的地方)
            request.setMethod(method);
            request.setPath(path);
        }
        //调用上面的处理函数
        dis(request);
    }
}

实现上述操作后,就可以测试是否正确:

本地的Key值是 “/myFrist”

去客户端(浏览器)输入试试:

 输入对应key值的:

结果:

取看输出这里:
 可以看出,上述代码被顺利执行了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/939758.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

谁说C比C++快?

看到这个问题&#xff0c;我我得说&#xff1a;这事儿没有那么简单。 1. 先把最大的误区打破 "C永远比C快" —— 某位1990年代的程序员 这种说法就像"自行车永远比汽车省油"一样荒谬。我们来看个例子&#xff1a; // C风格 char* str (char*)malloc(100…

html <a>设置发送邮件链接、打电话链接 <a href=“mailto:></a> <a href=“tel:></a>

1.代码 <ul><li>电话&#xff1a;<a href"tel:18888888888">188-8888-8888</a></li><li>邮箱&#xff1a;<a href"mailto:10000qq.com">10000qq.com</a></li><li>邮箱&#xff1a;<a hre…

Nginx三种安装方式

Nginx安装 可以登录 Nginx 的官方网站&#xff1a;https://www.nginx.com/ 找到安装方式。 查看如何安装开源的版本&#xff1a;https://docs.nginx.com/nginx/admin-guide/installing-nginx/installing-nginx-open-source/ 通过官方的说明&#xff0c;也可以知道安装&#…

Android 10 Launcher3 删除谷歌搜索

命令行获取页面 手机处于launcher首页 adb shell dumpsys window | findstr mCurrentFocus 输出 mCurrentFocusWindow{9afb34d u0 com.android.launcher3/com.android.launcher3.Launcher} 找到源码路径 packages/apps/Launcher3/ Android10源码 搜索控件 grep -r -n Apps…

自动驾驶AVM环视算法--python版本的俯视TOP投影模式

c语言版本和算法原理的可以查看本人的其他文档。《自动驾驶AVM环视算法--全景的俯视图像和原图》本文档进用于展示部分代码的视线&#xff0c;获取方式网盘自行获取&#xff08;非免费介意勿下载&#xff09;&#xff1a;链接: https://pan.baidu.com/s/1MJa8ZCEfNyLc5x0uVegto…

前端OpenAPI根据后端Swagger自动生成前端接口报错

测试之后发现是因为Map<Long,List<CommentVO>>的返回值类型的锅&#xff0c;改成Page<List<CommentVO>>即可解决。 前端使用的umiMAX的openapi&#xff0c;报错如下&#xff1a; originalRef: BaseResponseboolean\n "401&q…

java开发入门学习五-流程控制

流程控制语句 if&#xff0c; if...else&#xff0c; if..else if..else 与前端相同 略 switch case 与前端不同的是case不能使用表达式&#xff0c;使用表达式会报错 class TestSwitch {public static void main(String[] args) {// switch 表达式只能是特定的数据类型…

.NET 技术 | 调用系统API创建Windows服务

01阅读须知 此文所提供的信息只为网络安全人员对自己所负责的网站、服务器等&#xff08;包括但不限于&#xff09;进行检测或维护参考&#xff0c;未经授权请勿利用文章中的技术资料对任何计算机系统进行入侵操作。利用此文所提供的信息而造成的直接或间接后果和损失&#xf…

《PCI密码卡技术规范》题目

单选1 在《PCI密码卡技术规范》中&#xff0c;下列哪项不属于PCI密码卡的功能&#xff08;&#xff09;。 A.密码运算功能 B.密钥管理功能 C.物理随机数产生功能 D.随主计算机可信检测功能 正确答案&#xff1a;D. <font style"color:#DF2A3F;">解析&…

Java操作Redis-Jedis

介绍 前面我们讲解了Redis的常用命令&#xff0c;这些命令是我们操作Redis的基础&#xff0c;那么我们在 java程序中应该如何操作Redis呢&#xff1f;这就需要使用Redis的Java客户端&#xff0c;就如同我们使 用JDBC操作MySQL数据库一样。 Redis 的 Java 客户端很多&#xff0…

QT网络(二):TCP通信

传输层概念 传输控制协议&#xff08;transmission control protocol&#xff0c;TCP&#xff09;是一种被大多数 Internet 网络协议用于数据传输的底层网络协议&#xff0c;它是可靠的、面向流和连接的传输协议&#xff0c;特别适合用于连续数据传输。 应用层在网络模型中的…

Cherno C++学习笔记 P43 对象生存周期

这篇文章我们讲一下对象的生存周期。这个也是一个很重要的问题&#xff0c;因为我们总说&#xff0c;编程其实就是在操纵内存&#xff0c;而知道了变量如何在栈上生存&#xff0c;以及我们如何利用这些特性来让我们的编程更加简单&#xff0c;我们才是真的理解了编程。我们都知…

02、10个富士胶片模拟的设置

二色彩 1、色彩的加减控制全局的饱和度增减&#xff1b; 2、色彩效果只提升暖色系饱和度&#xff1b; 3、FX蓝色大幅度提升蓝色系饱和度&#xff1b; 4、三个参数都不改变颜色的色相。 2.1 色彩 色彩调整的是拍摄画面整体的色彩饱和程度 2.2色彩效果 调整的是画面中暖色…

【Unity3D】ILRuntime学习记录一

Unity 2019.4.0f1 导入ILRuntime 2.1.0版本 项目目录/Packages/manifest.json添加如下代码&#xff1a; {"scopedRegistries":[{"name":"ILRuntime","url":"https://registry.npmjs.org","scopes":["com.ou…

ECharts柱状图-柱图38,附视频讲解与代码下载

引言&#xff1a; 在数据可视化的世界里&#xff0c;ECharts凭借其丰富的图表类型和强大的配置能力&#xff0c;成为了众多开发者的首选。今天&#xff0c;我将带大家一起实现一个柱状图图表&#xff0c;通过该图表我们可以直观地展示和分析数据。此外&#xff0c;我还将提供…

2024年12月19日Github流行趋势

项目名称&#xff1a;ByteByteGoHq / system-design-101 项目维护者&#xff1a;slam, LombardiDaniel, Stephanvs, alastairp, lucasberti 等项目介绍&#xff1a;使用视觉和简单术语解释复杂的系统。帮助你准备系统设计面试。项目star数&#xff1a;66,527项目fork数&#xf…

Elasticsearch:什么是信息检索?

信息检索定义 信息检索 (IR) 是一种有助于从大量非结构化或半结构化数据中有效、高效地检索相关信息的过程。信息&#xff08;IR&#xff09;检索系统有助于搜索、定位和呈现与用户的搜索查询或信息需求相匹配的信息。 作为信息访问的主要形式&#xff0c;信息检索是每天使用…

机械鹦鹉与真正的智能:大语言模型推理能力的迷思

编者按&#xff1a; 大语言模型真的具备推理能力吗&#xff1f;我们是否误解了"智能"的本质&#xff0c;将模式匹配误认为是真正的推理&#xff1f; 本文深入探讨了大语言模型&#xff08;LLMs&#xff09;是否真正具备推理能力这一前沿科学问题&#xff0c;作者的核…

day-21 内核链表以及栈

1.昨日作业 1.删除指定节点 找到删除就完事了&#xff0c;双向可以停在删除处。 /***************************** 功能&#xff1a;删除指定结点&#xff08;通过姓名&#xff09;* 参数&#xff1a;phead&#xff1b;oldname; * 返回&#xff1a;成功0&#xff0c;失-1&…

深入剖析MyBatis的架构原理

架构设计 简要画出 MyBatis 的架构图 >> ​​ Mybatis 的功能架构分为哪三层&#xff1f; API 接口层 提供给外部使用的接口 API&#xff0c;开发人员通过这些本地 API 来操纵数据库。接口层一接收到调用请求就会调用数据处理层来完成具体的数据处理。MyBatis 和数据库的…