栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

MyBatis学习笔记

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

MyBatis学习笔记

记录自己的MyBatis学习的笔记,学习课程是通过B站尚硅谷的视频学习。

@Description MyBatis学习
@Author Van
①MyBatis简介
    1.MyBatis历史:
        1.MyBatis最初是Apache的一个开源项目iBatis,2010年6月这个项目由Apache Software Foundation迁移到了Google Code。
          随着开发团队转投Google Code旗下, iBatis3.x正式更名为MyBatis。代码于2013年11月迁移到Github。
        2.是一个基于Java的持久层框架。 iBatis提供的持久层框架包括SQL Maps和Data Access Objects(DAO)。
    2.MyBatis特性
        1) MyBatis 是支持定制化 SQL、存储过程以及高级映射的优秀的持久层框架
        2) MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集
        3) MyBatis可以使用简单的XML或注解用于配置和原始映射,将接口和Java的POJO(Plain Old Java Objects,普通的Java对象)映射成数据库中的记录
        4) MyBatis 是一个 半自动的ORM(Object Relation Mapping)框架
    3.MyBatis下载::https://github.com/mybatis/mybatis-3
    4.和其它持久化层技术对比
        1.JDBC
            SQL 夹杂在Java代码中耦合度高,导致硬编码内伤
            维护不易且实际开发需求中 SQL 有变化,频繁修改的情况多见
            代码冗长,开发效率低
        2.Hibernate 和 JPA
            操作简便,开发效率高
            程序中的长难复杂 SQL 需要绕过框架
            内部自动生产的 SQL,不容易做特殊优化
            基于全映射的全自动框架,大量字段的 POJO 进行部分映射时比较困难。
            反射操作太多,导致数据库性能下降
        3.MyBatis
            轻量级,性能出色
            SQL 和 Java 编码分开,功能边界清晰。Java代码专注业务、SQL语句专注数据
            开发效率稍逊于HIbernate,但是完全能够接受
②搭建MyBatis
    1.开发环境
        IDE:idea 2021.1.3
        构建工具:maven 3.8.1
        MySQL版本:MySQL 8.0.17
        MyBatis版本:MyBatis 3.5.7
    2.创建maven工程
        a>打包方式:jar
        b>引入依赖
            jar
            
               
               
                    org.mybatis
                    mybatis
                    3.5.7
                

                
                
                    junit
                    junit
                    4.12
                    test
                

                
                
                    mysql
                    mysql-connector-java
                    8.0.26
                

            

    3.创建MyBatis的核心配置文件
        》习惯上命名为mybatis-config.xml,这个文件名仅仅只是建议,并非强制要求。将来整合Spring之后,这个配置文件可以省略,所以大家操作时可以直接复制、粘贴。
        》核心配置文件主要用于配置连接数据库的环境以及MyBatis的全局配置信息
        》核心配置文件存放的位置是src/main/resources目录下
        
        
        
            
                
                    
                    
                        
                        
                        
                        
                    

                

            

            
            
                
            

        

    4.创建mapper接口
        1.MyBatis中的mapper接口相当于以前的dao。但是区别在于,mapper仅仅是接口,我们不需要提供实现类。
        2.一般会有一个和bean同一层的mapper包下创建一个接口。
    5.创建MyBatis的映射文件
        1.相关概念:ORM(Object Relationship Mapping)对象关系映射。
            对象:Java的实体类对象
            关系:关系型数据库
            映射:二者之间的对应关系
            java      数据库
             类            表
            属性       字段/列
            对象       记录/行
        2.映射文件的命名规则:
            1.表所对应的实体类的类名+Mapper.xml 例如:表t_user,映射的实体类为User,所对应的映射文件为UserMapper.xml
            2.因此一个映射文件对应一个实体类,对应一张表的操作
            3.MyBatis映射文件用于编写SQL,访问以及操作表中的数据
            4.MyBatis映射文件存放的位置是src/main/resources/mappers目录下
        3.MyBatis中可以面向接口操作数据,要保证两个一致:
            a>mapper接口的全类名和映射文件的命名空间(namespace)保持一致
            b>mapper接口中方法的方法名和映射文件中编写SQL的标签的id属性保持一致
            要有4个东西:表---实体类---mapper接口---映射文件
            mybatis-config.xml:
                
                
                    
                

            UserMapper.xml:
                
                    
                    
                        insert into t_user values(null,'admin','123456',23,'男','maomao@qq.com')
                    

                

    6.执行语句---JUnit测试
        //sqlSession默认不自动提交事务,需要打开传值true就行
        @Test
        public void testMyBatis() throws IOException {
            //1.mybatis提供了一个SQLSession,需要先加载核心配置文件
            InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
            //2.创建SqlSessionFactoryBuilder对象
            SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();
            //3.通过核心配置文件所对应的字节输入流创建工厂类SqlSessionFactory,生产SqlSession对象
            SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(is);
            //4.创建SqlSession对象,此时通过SqlSession对象所操作的sql都必须手动提交或回滚事务
            SqlSession sqlSession = sqlSessionFactory.openSession();
            //SqlSession sqlSession = sqlSessionFactory.openSession(true); 代表自动提交
            //5.通过代理模式创建UserMapper接口的代理实现类对象
            UserMapper mapper = sqlSession.getMapper(UserMapper.class);
            //6./调用UserMapper接口中的方法,就可以根据UserMapper的全类名匹配元素文件,通过调用的方法名匹配映射文件中的SQL标签,并执行标签中的SQL语句
            int result = mapper.insertUser();
            //7.提交事务,mybatis遵循原生jdbc事务,在执行之后必须提交事务或者回滚才能生效
            sqlSession.commit();
            System.out.println(result);
        }
        SqlSession:代表Java程序和数据库之间的会话。(HttpSession是Java程序和浏览器之间的会话)
        SqlSessionFactory:是“生产”SqlSession的“工厂”。
        工厂模式:如果创建某一个对象,使用的过程基本固定,那么我们就可以把创建这个对象的相关代码封装到一个“工厂类”中,以后都使用这个工厂类来“生产”我们需要的对象。
    7.加入log4j日志文件
        1.加入依赖
            
            
                log4j
                log4j
                1.2.17
            

        2.加入log4j的配置文件
            log4j的配置文件名为log4j.xml,存放的位置是src/main/resources目录下
        3.日志的级别
            FATAL(致命)>ERROR(错误)>WARN(警告)>INFO(信息)>DEBUG(调试)
            从左到右打印的内容越来越详细
    8.mybatis的增删改查
        1.添加
            
            
                insert into t_user values(null,'admin','123456',23,'男')
            

        2.删除
            
            
                delete from t_user where id = 7
            
        3.修改
            同上
        4.查询一个实体类对象
            
            
                select * from t_user
            
            ----返回的list直接遍历
            list.forEach(user -> System.out.println(user));
        --------注意
        1、查询的标签select必须设置属性resultType或resultMap,用于设置实体类和数据库表的映射关系
            resultType:自动映射,用于属性名和表中字段名一致的情况
            resultMap:自定义映射,用于一对多或多对一或字段名和属性名不一致的情况
        2、当查询的数据为多条时,不能使用实体类作为返回值,只能使用集合,否则会抛出异常TooManyResultsException;但是若查询的数据只有一条,可以使用实体类或集合作为返回值
③核心配置文件讲解
    --------------Ecvironments标签
    1.environments:设置多个连接数据库的环境
        属性:
            default:设置默认使用的环境的id
    2.environment:设置具体的连接数据库的环境信息
        属性:
            id:设置环境的唯一标识,可通过environments标签中的default设置某一个环境的id,表示默认使用的环境
    3.transactionManager:设置事务管理方式
        属性:
            type:设置事务管理方式,type="JDBC|MANAGED"
            type="JDBC":设置当前环境的事务管理都必须手动处理
            type="MANAGED":设置事务被管理,例如spring中的AOP
    4.dataSource:设置数据源
        属性:
            type:设置数据源的类型,type="POOLED|UNPOOLED|JNDI"
            type="POOLED":使用数据库连接池,即会将创建的连接进行缓存,下次使用可以从缓存中直接获取,不需要重新创建
            type="UNPOOLED":不使用数据库连接池,即每次使用连接都需要重新创建
            type="JNDI":调用上下文中的数据源
    -------------properties标签
        
        
    -------------typeAlias标签
    1.typeAlias:设置某个具体的类型的别名
        属性:
            type:需要设置别名的类型的全类名
            alias:设置此类型的别名,若不设置此属性,该类型拥有默认的别名,即类名且不区分大小写
                    若设置此属性,此时该类型的别名只能使用alias所设置的值
    2.
        
    ------------mapper标签
        
        
④MyBatis获取参数值的两种方式(重点)-----特别重要
    1.概念
        >MyBatis获取参数值的两种方式:${}和#{}
            1.${}的本质就是字符串拼接,#{}的本质就是占位符赋值
            2.${}使用字符串拼接的方式拼接sql,若为字符串类型或日期类型的字段进行赋值时,需要手动加单引号;
               但是#{}使用占位符赋值的方式拼接sql,此时为字符串类型或日期类型的字段进行赋值时,可以自动添加单引号
    2.单个字面量类型的参数
        1.若mapper接口中的方法参数为单个的字面量类型:
            此时可以使用${}和#{}以任意的名称获取参数的值,注意${}需要手动加单引号
    3.多个字面量类型的参数
        1.若mapper接口中的方法参数为多个时:
            此时MyBatis会自动将这些参数放在一个map集合中,以arg0,arg1...为键,以参数为值;以param1,param2...为键,以参数为值;
            因此只需要通过${}和#{}访问map集合的键就可以获取相对应的值,注意${}需要手动加单引号
    4.map集合类型的参数
        >若mapper接口中的方法需要的参数为多个时,此时可以手动创建map集合,将这些数据放在map中
        >只需要通过${}和#{}访问map集合的键就可以获取相对应的值,注意${}需要手动加单引号
    5.实体类类型的参数
        >若mapper接口中的方法参数为实体类对象时
        >此时可以使用${}和#{},通过访问实体类对象中的属性名获取属性值,注意${}需要手动加单引
    6.使用@Param标识参数
        >可以通过@Param注解标识mapper接口中的方法参数
        >此时,会将这些参数放在map集合中,以@Param注解的value属性值为键,以参数为值;以param1,param2...为键,以参数为值;
        只需要通过${}和#{}访问map集合的键就可以获取相对应的值,注意${}需要手动加单引号
    7.@Param注解源码分析
        1.里面有一个map,在底层会获取每一个参数和参数的注解放在一个二维数组里面,遍历如果注解有param就把索引为键参数为值放在叫names的map里面。
        2.然后会把传进来的真实值放在args里面
        3.此时会循环names,names的值为0=username,1=password,然后会把args和names的值以names的值为键、args的值为值,重新放入一个新的map里面---新的map名:param
          注意:放入新的map的时候 args是一个列表索引为0.1.2....  遍历names,用names的值当param的键,用args[names.getKey()]的值当param的值
        4.然后有一个常量gernericParamName ---》就是param+(i+1) i从0开始
        5.param.put(gernericParamName,args[names.getKey()])有把args的值 按照param1=args[0],param2=args[1]的形式放入param
        6.因此最后的map才会有两个一个是自定义的参数名=值 一个是param1=值,并且顺序是参数名1=值1,param1=值1,参数名2=值2,param2=值2
        ----解释:因为是for循环names在第一个put两次之后才会进入下一个循环
    -------理解
        参数获取分为两类:
            1.实体类,通过实体类的属性名来访问值
            2.其他都用@Param注解,包括数组和list等,都会自动生成来标识
⑤MyBatis的各种查询功能
    1.查询一个实体类对象
        User getUserById(@Param("id") int id)    
        
            select count(id) from t_user
        
    4.查询一条数据为map集合
        Map getUserToMap(@Param("id") int id);
        
                select * from t_user
            
        2.方式二:
            >将表中的数据以map集合的方式查询,一条数据对应一个map;若有多条数据,就会产生多个map集合,并且最终要以一个map的方式返回数据,
              此时需要通过@MapKey注解设置map集合的键,值是每条数据所对应的map集合
            @MapKey("id")
            Map getAllUserToMap();
            
            
            
            select * from t_user where username like "%"#{mohu}"%"
        
        //最好使用第三种
    2.批量删除---只能使用${}
        int deleteMore(@Param("ids") String ids);
        
            delete from t_user where id in (${ids})
        
    3.动态设置表名---只能使用${}
        List getAllUser(@Param("tableName") String tableName);
        
     *         select * from t_emp
     *      
     */
     >若字段名和实体类中的属性名不一致,但是字段名符合数据库的规则(使用_),实体类中的属性名符合Java的规则(使用驼峰)
        此时也可通过以下两种方式处理字段名和实体类中的属性的映射关系
            a>可以通过为字段起别名的方式,保证和实体类中的属性名保持一致
            b>可以在MyBatis的核心配置文件中设置一个全局配置信息mapUnderscoreToCamelCase,可以在查询表中数据时,自动将_类型的字段名转换为驼峰
            例如:字段名user_name,设置了mapUnderscoreToCamelCase,此时字段名就会转换为userName
    2.多对一映射处理
        ---》多对一,在多的类创建一的对象,例:在emp类中创建private Dept dept;设置getset、toString方法    
        
    3.一对多映射处理
        ---》一对多,在一中创建一个集合标识所有的信息
        
        
       
       
       
           
           
           
           
           
       

        

        
                select * from t_dept where did = #{did}
            
⑧动态SQL
    》Mybatis框架的动态SQL技术是一种根据特定条件动态拼装SQL语句的功能,它存在的意义是为了解决拼接SQL语句字符串时的痛点问题。
    1.if
        if标签可通过test属性的表达式进行判断,若表达式的结果为true,则标签中的内容会执行;反之标签中的内容不会执行
        !--List getEmpByCondition(Emp emp);-->
        
            select * from t_emp
            
                
                    emp_name = #{empName}
                

                
                    and age = #{age}
                

                
                    and sex = #{sex}
                

                
                    and email = #{email}
                

            

        
        where和if一般结合使用:
            a>若where标签中的if条件都不满足,则where标签没有任何功能,即不会添加where关键字
            b>若where标签中的if条件满足,则where标签会自动添加where关键字,并将条件最前方多余的 and 或者 or 去掉
            注意:where标签不能去掉条件最后多余的and
    3.trim
        
            select * from t_emp
            
                   //注意when等于eles if
                    
                        emp_name = #{empName}
                    

                    
                        age = #{age}
                    

                    
                        sex = #{sex}
                    

                    
                        email = #{email}
                    

                    
                        did = 1 //举例,都不满足就执行这个 otherwise不一样要有
                    

                

            

        
    5.foreach    
        主要用于批量增加和批量删除
        
        
            insert into t_emp values
            
                (null,#{emp.ename},#{emp.age},#{emp.sex},#{emp.email},null)
            

        

        
        
            delete from t_emp where
            
                eid = #{eid}
            

        
        
        
            delete from t_emp where eid in
            
                #{eid}
            

        
        属性:
            collection:设置要循环的数组或集合
            item:表示集合或数组中的每一个数据
            separator:设置循环体之间的分隔符
            open:设置foreach标签中的内容的开始符
            close:设置foreach标签中的内容的结束符
    6.SQL片段
        sql片段,可以记录一段公共sql片段,在使用的地方通过include标签进行引入
        
            eid,ename,age,sex,did
        

        select from t_emp
⑨MyBatis的缓存
    1.MyBatis的一级缓存
        1.一级缓存是SqlSession级别的,通过同一个SqlSession查询的数据会被缓存,下次查询相同的数据,就会从缓存中直接获取,不会从数据库重新访问
        2.使一级缓存失效的四种情况:
            1) 不同的SqlSession对应不同的一级缓存
            2) 同一个SqlSession但是查询条件不同
            3) 同一个SqlSession两次查询期间执行了任何一次增删改操作
            4) 同一个SqlSession两次查询期间手动清空了缓存  ---》sqlSession.clearCache()
    2.MyBatis的二级缓存
        1.二级缓存是SqlSessionFactory级别,通过同一个SqlSessionFactory创建的SqlSession查询的结果会被缓存;此后若再次执行相同的查询语句,结果就会从缓存中获取
            二级缓存开启的条件:
            a>在核心配置文件中,设置全局配置属性cacheEnabled="true",默认为true,不需要设置
            b>在映射文件中设置标签
            c>二级缓存必须在SqlSession关闭或提交之后有效
            d>查询的数据所转换的实体类类型必须实现序列化的接口 bean类去 implements Serializable接口
            注意:不能使用自己写的SqlSessionUtils,因为每创建一个SqlSession都创建了一个新的Factory
        2.使二级缓存失效的情况:
            两次查询之间执行了任意的增删改,会使一级和二级缓存同时失效
    3.二级缓存的相关配置
        在mapper配置文件中添加的cache标签可以设置一些属性:
            1.eviction属性:缓存回收策略
                LRU(Least Recently Used) – 最近最少使用的:移除最长时间不被使用的对象。
                FIFO(First in First out) – 先进先出:按对象进入缓存的顺序来移除它们。
                SOFT – 软引用:移除基于垃圾回收器状态和软引用规则的对象。
                WEAK – 弱引用:更积极地移除基于垃圾收集器状态和弱引用规则的对象。
                默认的是 LRU。
            2.flushInterval属性:刷新间隔,单位毫秒
                默认情况是不设置,也就是没有刷新间隔,缓存仅仅调用语句时刷新
                size属性:引用数目,正整数代表缓存最多可以存储多少个对象,太大容易导致内存溢出
            3.readOnly属性:只读,true/false
                true:只读缓存;会给所有调用者返回缓存对象的相同实例。因此这些对象不能被修改。这提供了很重要的性能优势。
                false:读写缓存;会返回缓存对象的拷贝(通过序列化)。这会慢一些,但是安全,因此默认是false。
    4.MyBatis缓存查询的顺序
        1.先查询二级缓存,因为二级缓存中可能会有其他程序已经查出来的数据,可以拿来直接使用。
        2.如果二级缓存没有命中,再查询一级缓存
        3.如果一级缓存也没有命中,则查询数据库
        4.SqlSession关闭之后,一级缓存中的数据会写入二级缓存
    5.整合第三方缓存EHCache----个人觉得了解就是
        a>添加依赖
            
            
                org.mybatis.caches
                mybatis-ehcache
                1.2.1
            

            
            
                ch.qos.logback
                logback-classic
                1.2.3
            

        b>各jar包功能
            jar包名称                             作用
            mybatis-ehcache          Mybatis和EHCache的整合包
            ehcache                       EHCache核心包
            slf4j-api                      SLF4J日志门面包
            logback-classic         支持SLF4J门面接口的一个具体实现
        c>创建EHCache的配置文件ehcache.xml
            
            
                
                
                                     maxElementsInMemory="1000"
                    maxElementsonDisk="10000000"
                    eternal="false"
                    overflowToDisk="true"
                    timeToIdleSeconds="120"
                    timeToLiveSeconds="120"
                    diskExpiryThreadIntervalSeconds="120"
                    memoryStoreEvictionPolicy="LRU">
                
            

        d>设置二级缓存的类型
                     e>加入logback日志
            存在SLF4J时,作为简易日志的log4j将失效,此时我们需要借助SLF4J的具体实现logback来打印日志。创建logback的配置文件logback.xml
            
            
            
                                     >
                    
                        
                        
                        [%d{HH:mm:ss.SSS}] [%-5level] [%thread] [%logger][%msg]%n
                    

                
                
                
                
                    
                    
                

                
                
            

-------------个人理解:整合第三方缓存了解就是------------------------------------------------------------
⑩MyBatis逆向工程
    正向工程:先创建Java实体类,由框架负责根据实体类生成数据库表。Hibernate是支持正向工程的。
    逆向工程:先创建数据库表,由框架负责根据数据库表,反向生成如下资源:
        Java实体类
        Mapper接口
        Mapper映射文件
    1.创建逆向工程的步骤
        a>添加依赖和插件
            jar
            
            
                
                    org.mybatis
                    mybatis
                    3.5.7
                

                
                    log4j
                    log4j
                    1.2.17
                

                
                
                    junit
                    junit
                    4.12
                    test
                

                
                    mysql
                    mysql-connector-java
                    8.0.26
                

            

            
            
                
                
                    
                    
                        org.mybatis.generator
                        mybatis-generator-maven-plugin
                        1.3.0
                        
                        
                            
                            
                                org.mybatis.generator
                                mybatis-generator-core
                                1.3.2
                            

                            
                            
                                com.mchange
                                c3p0
                                0.9.2
                            

                            
                            
                                mysql
                                mysql-connector-java
                                8.0.26
                            

        
        
                        

                    

                

            

        b>创建MyBatis的核心配置文件
        c>创建逆向工程的配置文件    
            文件名必须是:generatorConfig.xml
            
                                 PUBLIC "-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN"
                    "http://mybatis.org/dtd/mybatis-generator-config_1_0.dtd">
            
                
                
                    
                                                         connectionURL="jdbc:mysql://localhost:3306/mybatis"
                                    userId="root"
                                    password="123456">
                    

                    
                                                             targetProject=".srcmainjava">
                        
                        
                    

                    
                                                         targetProject=".srcmainresources">
                        
                    

                    
                                                             targetPackage="com.van.mybatis.mapper" targetProject=".srcmainjava">
                        
                    

                    
                    
                    
                    


                    

                
            
    2、QBC查询        
        @Test
        public void testMBG() throws IOException {
            InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
            SqlSession sqlSession = new
            SqlSessionFactoryBuilder().build(is).openSession(true);
            EmpMapper mapper = sqlSession.getMapper(EmpMapper.class);
            EmpExample empExample = new EmpExample();
            //创建条件对象,通过andXXX方法为SQL添加查询添加,每个条件之间是and关系
            empExample.createCriteria().andEnameLike("a").andAgeGreaterThan(20).andDidIsNotNull();
            //将之前添加的条件通过or拼接其他条件
            empExample.or().andSexEqualTo("男");
            List list = mapper.selectByExample(empExample);
            for (Emp emp : list) {
                System.out.println(emp);
            }
        }
①①分页插件----Pagehelper
    1.分页插件使用步骤
        a>添加依赖
            
            
                com.github.pagehelper
                pagehelper
                5.2.0
            

        b>配置分页插件
            在MyBatis的核心配置文件中配置插件
            
            
            
            

    2.分页插件的使用
        1.sql语句中分页是用limit index,pageSize
            index:当前页的起始索引
            pageSize:每页显示的条数
            pageNum:当前页的页码
            index=(pageNum-1)*pageSize
        2.a>在查询功能之前使用PageHelper.startPage(int pageNum, int pageSize)开启分页功能
            pageNum:当前页的页码
            pageSize:每页显示的条数
            b>在查询获取list集合之后,使用PageInfo pageInfo = new PageInfo<>(List list, intnavigatePages)获取分页相关数据
                list:分页之后的数据
                navigatePages:导航分页的页码数
            c>分页相关数据-----
                PageInfo{
                    pageNum=8, pageSize=4, size=2, startRow=29, endRow=30, total=30, pages=8,
                    list=Page{count=true, pageNum=8, pageSize=4, startRow=28, endRow=32, total=30,
                    pages=8, reasonable=false, pageSizeZero=false},
                    prePage=7, nextPage=0, isFirstPage=false, isLastPage=true, hasPreviousPage=true,
                    hasNextPage=false, navigatePages=5, navigateFirstPage4, navigateLastPage8,
                    navigatepageNums=[4, 5, 6, 7, 8]
                }
                常用数据:
                    pageNum:当前页的页码
                    pageSize:每页显示的条数
                    size:当前页显示的真实条数
                    total:总记录数
                    pages:总页数
                    prePage:上一页的页码
                    nextPage:下一页的页码
                    isFirstPage/isLastPage:是否为第一页/最后一页
                    hasPreviousPage/hasNextPage:是否存在上一页/下一页
                    navigatePages:导航分页的页码数
                    navigatepageNums:导航分页的页码,[1,2,3,4,5]
----------------------------------------------------------------------------------
    正式完结MyBatis,开始SpringBoot阶段的学习。不知道这样到底是否是对的,总之希望有个好的结果。
        
        
        
        
        
        
        
        
        

转载请注明:文章转载自 www.mshxw.com
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号