当前位置: 首页>编程语言>正文

mapper没有注入到ioc容器 spring如何注入mapper接口

(三)Mybatis------引入Mapper接口

引入Mapper接口并测试,新建接口。

package dao;

import POJO.ACicpTest1;

//这个接口只是为了,更加方便上层代码调用Mybatis的具体功能
// ACicpTest1Dao的路径和文件名,要和对应xml文件中的namespace一致
// 方法名要和xml文件中sql的id一致,xml中id只能存在一个,因此接口中的方法不能重载


public interface ACicpTest1Dao {

    public ACicpTest1 findOneById(String requestNo);
}

新建测试类:

import POJO.ACicpTest1;
import dao.ACicpTest1Dao;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;

import java.io.IOException;

public class MybatisTest {

    private SqlSession sqlSession;


    // junit会在每一个@Test方法前执行@Before方法
    @Before
    public void init() throws IOException {

//        1.使用Mybatis的Resources类读取Mybatis全局配置文件
//        InputStream stream = Resources.getResourceAsStream("mybatis-config.xml");
//        2.使用InputStream创建sqlSessionFactoryBuilder对象
//        SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
//        3.调用builder对象的build方法创建SqlSessionFactory对象
//        SqlSessionFactory build = builder.build(stream);
//        4.通过SqlSessionFactory对象开启一个从Java程序到数据库的会话
//        SqlSession sqlSession = build.openSession();
        //5.在通过SqlSession对象找到Mapper配置文件中可以执行的sql语句
        //两个参数,分别是statement、parameter
        // statement参数格式:Mapper配置文件namespace属性.SQL标签的id属性
        //parameter参数;给SQL语句传入的参数

         sqlSession = new SqlSessionFactoryBuilder().
                build(Resources.getResourceAsStream("mybatis-config.xml")).openSession();
    }

    @Test
    public void testMapperInterFace(){
        //根据ACicpTest1Dao接口的class对象获取Mapper接口类型的对象
        ACicpTest1Dao mapper = sqlSession.getMapper(ACicpTest1Dao.class);
        ACicpTest1 s = mapper.findOneById("40978");
        System.out.println(s);

    }

    @After
    public void clear(){
        sqlSession.commit();
        sqlSession.close();
    }
}

Mysql插入insert方法、删除delete、更新update时,会返回受影响的行数。

<insert id="insertById">
        <!-- 本质上#{requestNo}、#{loancurrentstatus}还是走了实体类的get、set方法,通过打断点可以看到-->
        insert into a_cicp_test_1 values  (#{requestNo},#{loancurrentstatus})
       </insert>

#{}会被换成?占位符

${}不会被换成?占位符,而是会根据?占位符进行字符串拼接,这样会引起SQL注入的问题,存在安全问题 至于为什么#{}能够防止SQL注入
原因:#{}匹配的是一个占位符,相当于 JDBC 中的一个?,底层采用的是 PreparedStatement,会对一些敏感字符进行过滤,编译过后会对传递的值加上双引号,因此可以防止 SQL 注入问题。

注意:如果表名是动态的,需要从参数中传进来,此时只能用拼接,不能用#{},因为数据库不允许表名位置使用?占位符。
写、更新操作:返回的是受影响的条数。

传入参数类型分为基本类型和复杂类型

简单类型:一个值的参数,int、String、Double等等
负载类型:实体类、集合、数组类型、复合类型,Student、Lits、int[]、List[Student]

传入对象

int insertEmployee(Employee employee);

<insert id="insertEmployee">
    insert into t_emp(emp_name,emp_salary) values(#{empName},#{empSalary})
</insert>

当传入的是对象时,Mybatis会根据#{}中传入的数据,加工成getXxx()方法,通过反射在实体类对象中调用这个方法,从而获取到对应的数据。填充到#{}这个位置。

传入多个简单的数据类型

int updateEmployee(@Param("empId") Integer empId,@Param("empSalary") Double empSalary);

 <update id="updateEmployee">
        update t_emp set emp_salary=#{empSalary} where emp_id=#{empId}
    </update>

当传入多个数据类型时,使用@Param(”xxx“),其中的字段名要和SQL中的字段名一致。

传入Map

Map<String, Object> paramMap = new HashMap<>();
    
    paramMap.put("id", 1);
    paramMap.put("name", "zhangsan");
     paramMap.put("age", 5);

int updateEmployeeByMap(Map<String, Object> paramMap);


    <update id="updateEmployeeByMap">
        update t_emp set name=#{name},age=#{age} where id=#{id}
    </update>

传入map时,#{}中写map中对应的key值即可。

出参各种类型

返回参数时对象

<select id="selectEmployee" resultType="com.atguigu.mybatis.entity.Employee">
    select emp_id empId,emp_name empName,emp_salary empSalary from t_emp where emp_id=#{maomi}
</select>

如果配置文件中没有做字段映射时,需要给查询出的字段起别名,使字段别名和对象中的属性值一致。

增加全局配置自动识别对应关系

要求比较高,格式必须为:数据库字段命名为student_id,而属性命名为studentId,如果开发过程中规定好了,并且都能遵守,就可以用,否则慎用。

<!-- 在全局范围内对Mybatis进行配置 -->
<settings>
    <!-- 具体配置 -->
    <!-- 从org.apache.ibatis.session.Configuration类中可以查看能使用的配置项 -->
    <!-- 将mapUnderscoreToCamelCase属性配置为true,表示开启自动映射驼峰式命名规则 -->
    <!-- 规则要求数据库表字段命名方式:单词_单词 -->
    <!-- 规则要求Java实体类属性名命名方式:首字母小写的驼峰式命名 -->
    <setting name="mapUnderscoreToCamelCase" value="true"/>
</settings>

返回Map类型

Map<String,Object> selectEmpNameAndMaxSalary(); 

<!-- 返回工资最高的员工的姓名和他的工资 -->
<select id="selectEmpNameAndMaxSalary" resultType="map">
        SELECT
            emp_name 员工姓名,
            emp_salary 员工工资,
            (SELECT AVG(emp_salary) FROM t_emp) 部门平均工资
        FROM t_emp WHERE emp_salary=(
            SELECT MAX(emp_salary) FROM t_emp
        )
</select>

Map<String, Object> resultMap = employeeMapper.selectEmpNameAndMaxSalary();
    
        Set<Map.Entry<String, Object>> entrySet = resultMap.entrySet();
    
        for (Map.Entry<String, Object> entry : entrySet) {
            String key = entry.getKey();
            Object value = entry.getValue();
            System.out.println(key + "=" + value);
        }

返回Map类型时,最好是不能封装到一个对象中,如果能封装到一个对象中,则使用对象返回,其中字段名作为key,获取值作为value,循环取值。

设置自增主键

比如:规则引擎中,规则调用日志表,和规则触发表,此时规则触发表保存时,需要有外键关联到规则调用日志表,才能够关联查询到,该笔数据触发了哪些规则。
此时在保存规则调用日志数据时就需要新增useGeneratedKeys属性设为true,开启自增主键。keyProperty代表着将自增值赋给类的属性,但并不是自增主键设置为返回值,而是传入的对象中,该字段会赋值,使用时通过get属性获得自增主键值

<!-- int insertEmployee(Employee employee); -->
<!-- useGeneratedKeys属性字面意思就是“使用生成的主键” -->
<!-- keyProperty属性可以指定主键在实体类对象中对应的属性名,Mybatis会将拿到的主键值存入这个属性 -->
<insert id="insertEmployee" useGeneratedKeys="true" keyProperty="empId">
    insert into t_emp(emp_name,emp_salary)
    values(#{empName},#{empSalary})
</insert>

 employeeMapper.insertEmployee(employee);
    //获取自增主键。
 System.out.println("employee.getEmpId() = " + employee.getEmpId());

Mybatis是将自增主键的值设置到实体类对象中,而不是以Mapper接口方法返回值的形式返回。

对于不支持自增主键的数据库,比如oracle

写法1:

<insert id="insertEmployee" 
        parameterType="com.atguigu.mybatis.beans.Employee"  
            databaseId="oracle">
        <selectKey order="BEFORE" keyProperty="id" 
                                       resultType="integer">
            select employee_seq.nextval from dual 
        </selectKey>    
        insert into orcl_employee(id,last_name,email,gender) values(#{id},#{lastName},#{email},#{gender})
</insert>

写法2:

<insert id="insertEmployee" 
        parameterType="com.atguigu.mybatis.beans.Employee"  
            databaseId="oracle">
        <selectKey order="AFTER" keyProperty="id" 
                                         resultType="integer">
            select employee_seq.currval from dual 
        </selectKey>    
    insert into orcl_employee(id,last_name,email,gender) values(employee_seq.nextval,#{lastName},#{email},#{gender})
</insert>

可以使用 selectKey 子元素:selectKey 元素将会首先运行,id 会被设置,然后插入语句会被调用

使用resultMap标签,定义类、和数据表之前的对应关系

此时返回值类型就不能用resultType,而使用resultMap,并且值为resultMap标签中的id值

<!-- 专门声明一个resultMap设定column到property之间的对应关系 -->
<resultMap id="selectEmployeeByRMResultMap" type="com.atguigu.mybatis.entity.Employee">
    
    <!-- 使用id标签设置主键列和主键属性之间的对应关系 -->
    <!-- column属性用于指定字段名;property属性用于指定Java实体类属性名 -->
    <id column="emp_id" property="empId"/>
    
    <!-- 使用result标签设置普通字段和Java实体类属性之间的关系 -->
    <result column="emp_name" property="empName"/>
    <result column="emp_salary" property="empSalary"/>
</resultMap>
    
<!-- Employee selectEmployeeByRM(Integer empId); -->
<select id="selectEmployeeByRM" resultMap="selectEmployeeByRMResultMap">
    select emp_id,emp_name,emp_salary from t_emp where emp_id=#{empId}
</select>



https://www.xamrdz.com/lan/5gk1963872.html

相关文章: