我的编程空间,编程开发者的网络收藏夹
学习永远不晚

mybatis项目CRUD步骤实例详解

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

mybatis项目CRUD步骤实例详解

mybatis项目CRUD步骤

1.pom.xml引入相应的依赖

  <?xml version="1.0" encoding="UTF-8"?>
  <project xmlns="http://maven.apache.org/POM/4.0.0"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
      <modelVersion>4.0.0</modelVersion>
  <!--父工程-->
      <groupId>org.example</groupId>
      <artifactId>demo1</artifactId>
      <packaging>pom</packaging>
      <version>1.0-SNAPSHOT</version>
      <modules>
          <module>demo01</module>
          <module>demo02</module>
          <module>demo03</module>
          <module>demo04</module>
      </modules>
      <!--    导入依赖-->
       <dependencies>
           <dependency>
               <groupId>mysql</groupId>
               <artifactId>mysql-connector-java</artifactId>
               <version>8.0.28</version>
           </dependency>
  <!--         导入lombok依赖-->
           <dependency>
               <groupId>org.projectlombok</groupId>
               <artifactId>lombok</artifactId>
               <version>1.18.22</version>
           </dependency>
           <dependency>
               <groupId>org.mybatis</groupId>
               <artifactId>mybatis</artifactId>
               <version>3.5.9</version>
           </dependency>
           <dependency>
               <groupId>junit</groupId>
               <artifactId>junit</artifactId>
               <version>4.13</version>
               <scope>test</scope>
           </dependency>
           <dependency>
               <groupId>org.mybatis.caches</groupId>
               <artifactId>mybatis-ehcache</artifactId>
               <version>1.2.1</version>
           </dependency>
       </dependencies>
  <!--maven资源导出,约定大于配置-->
      <build>
          <resources>
              <resource>
                  <directory>class="lazy" data-src/main/resources</directory>
                  <includes>
                      <include>***.xml</include>
                  </includes>
                  <filtering>false</filtering>
              </resource>
              <resource>
                  <directory>class="lazy" data-src/main/java</directory>
                  <includes>
                      <include>***.xml</include>
                  </includes>
                  <filtering>false</filtering>
              </resource>
          </resources>
      </build>
  
  
  </project>

2. 在resources目录下写配置文件

  • 数据库连接配置
  • db.properties
  driver=com.mysql.cj.jdbc.Driver
  url=jdbc:mysql://localhost:3306/mybatis?useSSL=true&useUnicode=true&characterEncoding=UTF-8
  username=root
  password=123456
  • mybatis配置
  • mybatis-config.xml
  <?xml version="1.0" encoding="UTF-8" ?>
  <!DOCTYPE configuration
          PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
          "http://mybatis.org/dtd/mybatis-3-config.dtd">
  <!--configuration核心配置文件-->
  <configuration>
  <!--    引入外部配置文件-->
      <properties resource="db.properties">
  
      </properties>
  
      <settings>
          <!--    开启日志-->
          <setting name="logImpl" value="STDOUT_LOGGING"/>
          <!--显示的开启缓存-->
          <setting name="cacheEnabled" value="true"/>
      </settings>
  <!--    类型别名:可以给实体类起别名-->
      <typeAliases>
  <!--        固定别名-->
  <!--        <typeAlias type="com.wyc.pojo.User" alias="User"></typeAlias>-->
  
  <!--        扫描包:扫描实体类的包,它的默认别名就为这个类的类名,首字母小写!-->
          <package name="com.wyc.pojo"/>
      </typeAliases>
  
      <environments default="development">
          <environment id="development">
              <transactionManager type="JDBC"/>
              <dataSource type="POOLED">
                  <property name="driver" value="${driver}"/>
                  <property name="url" value="${url}"/>
                  <property name="username" value="${username}"/>
                  <property name="password" value="${password}"/>
              </dataSource>
          </environment>
      </environments>
  <!--    注册mapper-->
      <mappers>
          <mapper class="com.wyc.dao.UserMapper"/>
      </mappers>
  </configuration>

  • 自定义缓存配置
  • ehcache.xml
  <?xml version="1.0" encoding="UTF-8"?>
  <ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd"
           updateCheck="false">
  
      <diskStore path="./tmpdir/Tmp_EhCache"/>
  
      <defaultCache
              eternal="false"
              maxElementsInMemory="10000"
              overflowToDisk="false"
              diskPersistent="false"
              timeToIdleSeconds="1800"
              timeToLiveSeconds="259200"
              memoryStoreEvictionPolicy="LRU"/>
  
      <cache
              name="cloud_user"
              eternal="false"
              maxElementsInMemory="5000"
              overflowToDisk="false"
              diskPersistent="false"
              timeToIdleSeconds="1800"
              timeToLiveSeconds="1800"
              memoryStoreEvictionPolicy="LRU"/>
  </ehcache>

3. 创建相应的包

img

各个层的含义

img

4. 在utils层

获取获取sqlSessionFactory对象

  package com.wyc.utils;
  
  import org.apache.ibatis.io.Resources;
  import org.apache.ibatis.session.SqlSession;
  import org.apache.ibatis.session.SqlSessionFactory;
  import org.apache.ibatis.session.SqlSessionFactoryBuilder;
  
  import java.io.IOException;
  import java.io.InputStream;
  
  //sqlSessionFactory -->sqlSession
  public class MybayisUtils {
      private static SqlSessionFactory sqlSessionFactory;
      static {
          try{
              //使用mybatis第一步: 获取sqlSessionFactory对象
                  String resource = "mybatis-config.xml";
                  InputStream inputStream = Resources.getResourceAsStream(resource);
                  sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
              } catch (IOException e) {
                  e.printStackTrace();
              }
      }
      public static SqlSession getSqlSession(){
          return sqlSessionFactory.openSession(true);
  
  
      }
  }

5. 在pojo层创建实体类,类名与数据库表一致

  package com.wyc.pojo;
  
  import lombok.Data;
  
  @Data   //自动生成get set 等方法,详细点击 右边的Structure
  public class User {
      private int id;
      private String name;
      private String pwd;
  }

6.在dao层编写Mapper接口,和Mapper.xml sql语法

UserMapper

  package com.wyc.utils;
  
  import org.apache.ibatis.io.Resources;
  import org.apache.ibatis.session.SqlSession;
  import org.apache.ibatis.session.SqlSessionFactory;
  import org.apache.ibatis.session.SqlSessionFactoryBuilder;
  
  import java.io.IOException;
  import java.io.InputStream;
  
  //sqlSessionFactory -->sqlSession
  public class MybayisUtils {
      private static SqlSessionFactory sqlSessionFactory;
      static {
          try{
              //使用mybatis第一步: 获取sqlSessionFactory对象
                  String resource = "mybatis-config.xml";
                  InputStream inputStream = Resources.getResourceAsStream(resource);
                  sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
              } catch (IOException e) {
                  e.printStackTrace();
              }
      }
      public static SqlSession getSqlSession(){
          return sqlSessionFactory.openSession(true);
  
  
      }
  }

UserMapper.xml

  <?xml version="1.0" encoding="UTF-8" ?>
  <!DOCTYPE mapper
          PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
          "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  <mapper namespace="com.wyc.dao.UserMapper">
      <insert id="add">
  
      </insert>
      <insert id="addUser" parameterType="com.wyc.pojo.User">
      insert into mybatis.user (id,name,pwd) value (#{id},#{name},#{pwd})
      </insert>
      <insert id="addUser2" parameterType="map">
      insert into mybatis.user (id,name,pwd) value (#{userid},#{username},#{userpwd})
      </insert>
      <update id="update" parameterType="com.wyc.pojo.User">
      update mybatis.user set name =#{name},pwd=#{pwd} where id = #{id} ;
      </update>
      <delete id="deleteUser">
          delete from mybatis.user where id = #{id}
      </delete>
      <select id="getUserList" resultType="com.wyc.pojo.User">
      select * from mybatis.user
    </select>
      <select id="getUserById" resultType="com.wyc.pojo.User" parameterType="int">
      select * from mybatis.user where id = #{id}
      </select>
      <select id="getUserById2" resultType="com.wyc.pojo.User" parameterType="map">
      select * from mybatis.user where id = #{id} and name = #{name};
      </select>
      <select id="getUserLike" resultType="com.wyc.pojo.User">
      select * from mybatis.user where name like "%"#{value}"%"
  
      </select>
          
          
  <!--    在当前mapper.xml中开启二级缓存-->
      <cache eviction="FIFO"
             flushInterval="60000"
             size="512"
             readOnly="true"
      ></cache>
  
  <!--自定义缓存-->
  <cache type="org.mybatis.caches.ehcache.EhcacheCache"/>
  <!--useCache="true":使用开启缓存-->
      <select id="queryUserById" parameterType="int" resultType="user" useCache="true">
          select * from mybatis.user where id = #{id}
      </select>
  </mapper>

BlogMapper

img

BlogMapper.xml

  <?xml version="1.0" encoding="UTF-8" ?>
  <!DOCTYPE mapper
          PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
          "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  <mapper namespace="com.wyc.dao.BlogMapper">
      <insert id="addBlog" parameterType="blog">
          insert into mybatis.blog (id,title,author,create_time,views)
          value (#{id},#{title},#{author},#{createTime},#{views});
      </insert>
      <sql id="if-title-author">                     //if语句
          <if test="title != null">
              and title = #{title}
          </if>
          <if test="author != null">
              and author = #{author}
          </if>
      </sql>
      <select id="queryBlogIf" parameterType="map" resultType="blog">
      select * from mybatis.blog where 1=1
      <where>
          <include refid="if-title-author"></include>
      </where>
      </select>
  
  
  
      <select id="queryBlogChoose" parameterType="map" resultType="blog">   //choose语句
      select * from mybatis.blog
      <where>
          <choose>
              <when test="title != null">
                   title = #{title}
              </when>
              <when test="title != null">
                   and author = #{author}
              </when>
              <otherwise>
                  and views = #{views}
              </otherwise>
          </choose>
      </where>
      </select>
  <!--我们现在传入一个万能的map,这个map可以存一个集合
            select * from mybatis.blog where 1=1 and (id=1 or id=2 or id=3)
  -->
      <select id="queryBlogForeach" parameterType="map" resultType="blog">   //foreach语句
      select * from mybatis.blog
       <where>
              <foreach collection="ids" item="id" open="and (" close=")" separator="or">  separator  拼接sql
                  id = #{id}
              </foreach>
       </where>
      </select>
  
  
      <update id="updateBlog" parameterType="map">
       update mybatis.blog
       <set>
           <if test="title != null">
               title = #{title},
           </if>
           <if test="author != null">
               author = #{author}
           </if>
       </set>
       where id = #{id}
      </update>
  </mapper>

7.编写测试类test,进行测试

1. UserTest

  package com.wyc.dao;
  
  import com.wyc.pojo.User;
  import com.wyc.utils.MybayisUtils;
  import org.apache.ibatis.session.SqlSession;
  import org.junit.Test;
  
  import java.util.HashMap;
  import java.util.List;
  import java.util.Map;
  
  public class UserDaoTest {
      @Test
      //查询所有用户
      public void test(){
          //第一步:获得SqlSession对象
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          //执行sql  方式一:getMapper
          UserMapper mapper = sqlSession.getMapper(UserMapper.class);
          List<User> userList = mapper.getUserList();
          for (User user : userList) {
              System.out.println(user);
  
          }
  
      }
      @Test
      public void testlike(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          UserMapper mapper = sqlSession.getMapper(UserMapper.class);
          List<User> userLike = mapper.getUserLike("zhang%");  //%最好在sql中拼接
          for (User user : userLike) {
              System.out.println("______________________");
              System.out.println(user);
          }
      }
      @Test
      //根据id查询
      public void test1(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
              UserMapper mapper = sqlSession.getMapper(UserMapper.class);
          User userById = mapper.getUserById(1);
          System.out.println(userById);
  
      }
      @Test
      //添加用户 增删改查需要提交事物
      public void test2(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          UserMapper mapper = sqlSession.getMapper(UserMapper.class);
          int addUser = mapper.addUser(new User(5,"HH","123"));
          sqlSession.commit();
  
      }
      @Test
      public void add(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          UserMapper mapper = sqlSession.getMapper(UserMapper.class);
          Map<String, Object> map = new HashMap<String, Object>();
          map.put("userid",7);
          map.put("username","zhangsan");
          mapper.addUser2(map);
          sqlSession.commit();
  
      }
      @Test
      //修改用户
      public void test3(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          UserMapper mapper = sqlSession.getMapper(UserMapper.class);
          int update = mapper.update(new User(3, "rr", "1234"));
          sqlSession.commit();
      }
      @Test
      //删除用户
      public void test4(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          UserMapper mapper = sqlSession.getMapper(UserMapper.class);
          int user = mapper.deleteUser(2);
          sqlSession.commit();
  
      }
  }
 

2. BlogTest

  import com.wyc.dao.BlogMapper;
  import com.wyc.dao.UserMapper;
  import com.wyc.pojo.Blog;
  import com.wyc.pojo.User;
  import com.wyc.utils.IDutils;
  import com.wyc.utils.MybayisUtils;
  import org.apache.ibatis.session.SqlSession;
  import org.junit.Test;
  
  import java.util.ArrayList;
  import java.util.Date;
  import java.util.HashMap;
  import java.util.List;
  
  public class MyTest {
      @Test
      public void test(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
          Blog blog = new Blog();
          blog.setId(IDutils.getId());
          blog.setTitle("mybatis如此简单");
          blog.setAuthor("狂神说");
          blog.setCreateTime(new Date());
          blog.setViews(9999);
          mapper.addBlog(blog);
  
          blog.setId(IDutils.getId());
          blog.setTitle("java如此简单");
          mapper.addBlog(blog);
  
          blog.setId(IDutils.getId());
          blog.setTitle("spring如此简单");
          mapper.addBlog(blog);
  
          blog.setId(IDutils.getId());
          blog.setTitle("微服务如此简单");
          mapper.addBlog(blog);
  
          sqlSession.commit();
  
      }
      @Test
      public void test2(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
          HashMap map = new HashMap();
          map.put("author","狂神说");
          List<Blog> blogs = mapper.queryBlogIf(map);
          for (Blog blog : blogs) {
              System.out.println(blog);
          }
  
      }
      @Test
      public void test3(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
          HashMap map = new HashMap();
          map.put("views",9999);
          List<Blog> blogs = mapper.queryBlogChoose(map);
          for (Blog blog : blogs) {
              System.out.println(blog);
          }
  
      }
      @Test
      public void test4(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
          HashMap map = new HashMap();
  //        map.put("views",9999);
          map.put("id","aaee3fee53d041ef93f51ff42d432d");
          map.put("author","超哥说");
          map.put("title","编程如此简单");
          mapper.updateBlog(map);
          sqlSession.commit();
  
      }
      @Test
      public void test5(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
          HashMap map = new HashMap();
          ArrayList<Integer> ids = new ArrayList<Integer>();
          ids.add(1);
          map.put("ids",ids);
          List<Blog> blogs = mapper.queryBlogForeach(map);
          for (Blog blog : blogs) {
              System.out.println(blog);
          }
          sqlSession.commit();
  
      }
      @Test
      public void queryUserById(){
          SqlSession sqlSession = MybayisUtils.getSqlSession();
          SqlSession sqlSession2 = MybayisUtils.getSqlSession();
          UserMapper mapper = sqlSession.getMapper(UserMapper.class);
          User user = mapper.queryUserById(1);
          System.out.println(user);
          sqlSession.close();
          UserMapper mapper2 = sqlSession2.getMapper(UserMapper.class);
          User user2 = mapper2.queryUserById(1);
          System.out.println(user2);
      }
  }

到此这篇关于mybatis项目CRUD步骤的文章就介绍到这了,更多相关mybatis CRUD步骤内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

mybatis项目CRUD步骤实例详解

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

详解gitee上传项目的步骤

Gitee 作为国内最大的 Git 代码托管平台之一,是众多开发者常用的工具之一。在这里,您可以上传自己的项目代码并进行管理。下面将为您介绍一下,在 Gitee 上上传项目的具体步骤。1.注册及登录 Gitee首先,您需要注册并登录 Git
2023-10-22

详解github中上传项目的步骤

GitHub是全球最大的开源托管平台之一,拥有强大的版本控制功能和在线协作工具,为开发者提供了更为便捷的团队合作方式和版本管理方法。因此,GitHub的使用已成为大量开发者的首选,而上传项目则是Git与GitHub最基本的操作之一。本文将介
2023-10-22

PyCharm教程:详解项目创建步骤

PyCharm是一款功能强大的Python集成开发环境(IDE),在Python开发中被广泛使用。本文将详细介绍PyCharm中如何创建一个新项目的步骤,包括具体的代码示例。第一步,打开PyCharm。首先,我们需要确保已经安装了PyCh
PyCharm教程:详解项目创建步骤
2024-02-23

GitLab上怎么创建项目(步骤详解)

在如今高度数字化的世界里,git已经成为了所有开发者的必备工具。为了更好的管理和维护,GitLab作为一个非常典型的Git代码托管平台,已经被广泛应用。那么,如何在GitLab上面创建一个项目呢?以下是详细的步骤:第一步:创建一个GitLa
2023-10-22

使用vue-cli创建vue2项目的实战步骤详解

相信大部分Vue开发者都使用过vue-cli来构建项目,它的确很方便,但对于很多初级开发者来说,还是要踩不少坑的,下面这篇文章主要给大家介绍了关于使用vue-cli创建vue2项目的实战步骤,需要的朋友可以参考下
2023-01-28

SpringBoot 项目的创建与启动步骤详解

这篇文章主要介绍了SpringBoot 项目的创建与启动,本文分步骤给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
2023-03-23

SpringBoot项目使用mybatis-plus代码生成的实例详解

mybatis-plus是mybatis的增强,不对mybatis做任何改变,涵盖了代码生成,自定义ID生成器,快速实现CRUD,自动分页,逻辑删除等功能。本文就来讲讲SpringBoot项目如何使用mybatis-plus实现代码生成,需要的可以了解一下
2022-11-13

Android PhotoView使用步骤实例详解

Android PhotoView使用步骤实例详解1、步骤一:在布局文件中的代码:
2023-05-31

opencv实践项目之图像拼接详细步骤

OpenCV的应用领域非常广泛,包括图像拼接、图像降噪、产品质检、人机交互、人脸识别、动作识别、动作跟踪、无人驾驶等,下面这篇文章主要给大家介绍了关于opencv实践项目之图像拼接的相关资料,需要的朋友可以参考下
2023-05-19

阿里云服务器项目重启步骤详解

阿里云服务器是阿里云提供的一种高性能计算服务,其稳定性和可靠性受到许多企业和个人的信赖。然而,由于各种原因,阿里云服务器可能会出现宕机或重启的情况,这时就需要重新启动服务器。本文将详细介绍阿里云服务器项目重启的步骤。步骤一:登录阿里云控制台首先,你需要在电脑上登录阿里云控制台,可以通过阿里云官网或者通过阿里云手机
阿里云服务器项目重启步骤详解
2023-10-31

ios 使用xcode11 新建项目工程的步骤详解

xcode11新建项目工程,新增了scenedelegate这个类,转而将原Appdelegate负责的对UI生命周期的处理担子接了过来。故此可以理解为:ios 13以后,Appdelegate负责处理App生命周期,scenedelega
2022-05-18

java Spring MVC4环境搭建实例详解(步骤)

spring WEB MVC框架提供了一个MVC(model-view-controller)模型-视图-控制器的结构和组件,利用它可以开发更灵活、松耦合的web应用。MVC模式使得整个服务应用的各部分(控制逻辑、业务逻辑、UI界面展示)分
2023-05-31

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录