MyBatis 操作数据库(入门)

news2024/11/28 5:50:11

一:MyBatis概念

(1)MyBatis

💗MyBatis是一款优秀的持久层框架,用于简化JDBC的开发

(2)持久层

1.持久层

💜持久层:持久化操作的层,通常指数据访问层(dao),是用来操作数据库的

2.持久层的规范

①包规范:一般取名叫mapper


②接口规范:XxxMapper

(3)MyBatis的开发方式

💗①注解

(目录三)


💗②XML

(目录四)

二:MyBatis入门

(1)准备工作

1.核心步骤

💗创建springboot工程,并导入mybatis的起步依赖、mysql的驱动包

2.创建工程并导入依赖

🌟步骤:File➜New Project➜Spring Initializr➜按照下面的选,然后一路next即可


(2)数据准备

1.在数据库创建用户表

①数据以及SQL代码

-- 创建数据库
DROP DATABASE IF EXISTS mybatis_test;
CREATE DATABASE mybatis_test DEFAULT CHARACTER SET utf8mb4;
-- 使⽤数据数据
USE mybatis_test;
-- 创建表[⽤⼾表]
DROP TABLE IF EXISTS userinfo;
CREATE TABLE `userinfo` (
`id` INT ( 11 ) NOT NULL AUTO_INCREMENT,
`username` VARCHAR ( 127 ) NOT NULL,
`password` VARCHAR ( 127 ) NOT NULL,
`age` TINYINT ( 4 ) NOT NULL,
`gender` TINYINT ( 4 ) DEFAULT '0' COMMENT '1-男 2-⼥ 0-默认',
`phone` VARCHAR ( 15 ) DEFAULT NULL,
`delete_flag` TINYINT ( 4 ) DEFAULT 0 COMMENT '0-正常, 1-删除',
`create_time` DATETIME DEFAULT now(),
`update_time` DATETIME DEFAULT now(),
PRIMARY KEY ( `id` )
) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;
-- 添加⽤⼾信息
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'admin', 'admin', 18, 1, '18612340001' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'zhangsan', 'zhangsan', 18, 1, '18612340002' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'lisi', 'lisi', 18, 1, '18612340003' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'wangwu', 'wangwu', 18, 1, '18612340004' );

②在Navicat中创建数据表

2.创建对应的实体类

①先创建一个model包,里面用来放实体类,然后创建对应的实体类UserInfo


②在UserInfo中完善代码

(注意:实体类的属性名与表中的字段名需要⼀⼀对应)

package com.hlizoo.demo.model;
import lombok.Data;
import java.util.Date;

@Data
public class UserInfo {
    private Integer id;
    private String username;
    private String password;
    private Integer age;
    private Integer gender;
    private String phone;
    private Integer deleteFlag;
    private Date createTime;
    private Date updateTime;
}

(3)数据库配置文件

1.相关的配置文件和配置项

💗Mybatis中要连接数据库,需要数据库相关参数配置


①URL

②登录名

③密码

④MySQL驱动类


🌟关于MySQL驱动类的注意事项:

(1)如果使用MySQL是5.x之前版本则MySQL驱动类使用的是"com.mysql.jdbc.Driver"

(2)如果使用MySQL是5.x之后版本则MySQL驱动类使用的是“com.mysql.cj.jdbc.Driver”

2.application.yml配置
# 数据库连接配置
spring:
  datasource:
    url: jdbc:mysql://127.0.0.1:3306/mybatis_test?characterEncoding=utf8&useSSL=false
    username: root
    password: root
    driver-class-name: com.mysql.cj.jdbc.Driver


//用户名和密码根据你的数据库而定
//mybatis_test就是你自己的数据库名,你的可能是别的名
3.application.properties配置
#驱动类名称
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
#数据库连接的url
spring.datasource.url= jdbc:mysql://127.0.0.1:3306/mybatis_test?characterEncoding=utf8&useSSL=false
#连接数据库的⽤⼾名
spring.datasource.username=root
#连接数据库的密码
spring.datasource.password=root

//用户名和密码根据你的数据库而定
//mybatis_test就是你自己的数据库名

(4)@Mapper注解

1.注解含义

🌟表示是MyBatis中的Mapper接口

2.功能

💗程序运行时, 框架会自动生成接口的实现类对象(代理对象),并交给Spring的IOC容器管理


💚与五大注解功能一样;但因为这里用的是MyBatis,所以要用它的注解而不用五大注解

(5)写持久层代码

1.回顾持久层的规范

①包规范:一般取名叫mapper


②接口规范:XxxMapper

2.创建包以及接口代码

①创建的包和接口


②UserInfoMapper接口代码

(@Select注解:代表的就是select查询,也就是注解对应方法的具体实现内容)

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Select;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    @Select("select * from userinfo")
    public List<UserInfo> selectAll();
}

(6)IDEA自动生成测试类

1. 在需要测试的Mapper接口中,右键➜Generate➜Test


2. 选择要测试的方法,点击OK


3.编写代码

(记得加@SpringBootTest注解, 加载Spring运行环境)

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

import java.util.List;

import static org.junit.jupiter.api.Assertions.*;

//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void selectAll() {
        List<UserInfo> userInfoList = userInfoMapper.selectAll();
        System.out.println(userInfoList);
    }
}

4.运行结果

三:MyBatis注解的基础操作

(1)打印日志

1.作用

🌟Mybatis当中我们可以借助日志,查看到sql语句的执行、执行传递的参数以及执行的结果

2.方法

💗在配置文件中进行配置即可


①application.properties配置

#指定mybatis输出⽇志的位置, 输出控制台
mybatis.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl

②application.yml配置

# 配置打印 MyBatis⽇志
mybatis:
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
3.观察效果

💚重新运行程序之后,观察效果


①查询语句

②传递的参数以及类型

③SQL执行结果

(2)参数传递

1.作用

🌟为了解决限制字段条件的语句

(比如查询id=4的用户数据,那么id该怎么传?)

2.方法
💜两个步骤

💗①先在注解修饰的方法形参中加入你需要限制条件的字段名


💗②然后再给注解中查询语句限制条件的字段名加上#{ }


💚此时传过去的限制条件就是动态的了;比如我想查询id=3的用户数据,那么我就传入参数id为3,如果我突然又想查询id=4的用户数据,那就传入参数id为4,就很方便,动态的输入,而不是写死

3.代码及运行结果

①UserInfoMapper代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Select;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    @Select("select * from userinfo where id=#{id}")
    List<UserInfo> selectOne(Integer id);
}

②UserInfoMapperTest测试代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

import java.util.List;

import static org.junit.jupiter.api.Assertions.*;

//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void selectOne() {
        //查询id=4的用户数据
        List<UserInfo> userInfoList1 = userInfoMapper.selectOne(4);
        System.out.println(userInfoList1);
    }
}

③运行结果


④比如我突然想查询id=3的用户数据,那么我只需要将参数改成3即可

4.注意事项

①如果SQL查询只需要一个参数,那么参数字段名是可以任意的

(但还是建议和参数名保持⼀致,两个是一样最好!!!)


②如果SQL查询需要的参数超过一个,那么参数字段名不可以任意


③通过@Param设置参数的别名,如果使用@Param设置别名, #{...}里面的属性名必须和 @Param设置的⼀样

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Select;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    @Select("select * from userinfo where id=#{userid}")
    List<UserInfo> selectOne2(@Param("userid") Integer id);
}
package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

import java.util.List;

import static org.junit.jupiter.api.Assertions.*;

//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void selectOne2() {
        List<UserInfo> userInfoList2 = userInfoMapper.selectOne2(1);
        System.out.println(userInfoList2);
    }
}

(3)增(Insert)

1.方法

💜两个步骤;使用@Insert注解


💗①先在@Insert注解修饰的方法形参中加入需要添加的实体类对象

(方法形参传的是对象)


💗②然后再在@Insert注解中写上insert语句,values后面的参数加上#{}

(这些参数名要与实体类的属性名相一致)


💙@Insert语句返回的是Integer类型,即返回的是有多少行受影响

2.代码及运行结果

①UserInfo实体类的代码

package com.hlizoo.demo.model;
import lombok.Data;
import java.util.Date;

@Data
public class UserInfo {
    private Integer id;
    private String username;
    private String password;
    private Integer age;
    private Integer gender;
    private String phone;
    private Integer deleteFlag;
    private Date createTime;
    private Date updateTime;
}

②UserInfoMapper代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.Insert;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Select;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    @Insert("insert into userinfo(username,password,age,gender,phone) "+
            "values(#{username},#{password},#{age},#{gender},#{phone})")
    Integer insert(UserInfo userInfo);
}

③UserInfoMapperTest测试代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.util.List;
import static org.junit.jupiter.api.Assertions.*;

@Slf4j
//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void insert() {
        UserInfo userInfo = new UserInfo();
        userInfo.setUsername("zhaoliu");
        userInfo.setPassword("zhaoliu");
        userInfo.setAge(18);
        userInfo.setGender(0);
        userInfo.setPhone("123456666");
        Integer result = userInfoMapper.insert(userInfo);
        log.info("inset方法的执行结果:"+result);
    }
}

④观察结果

3.返回主键
1.作用

🌟Insert语句默认返回的是受影响的行数,但有些情况下,数据插插入之后,还需要有后续的关联操作,需要获取到新插入数据的id,即要拿到自增id


2.方法

💗在Mapper接口的方法上添加⼀个Options的注解


@Options注解有两个参数

①useGeneratedKeys:是否使用自动生成的key;默认值为false

②keyProperty:将这个自动生成的key赋值给谁;默认是未设置

3.代码

①UserInfoMapper代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.*;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    @Options(useGeneratedKeys = true,keyProperty = "id")
    @Insert("insert into userinfo(username,password,age,gender,phone) "+
            "values(#{username},#{password},#{age},#{gender},#{phone})")
    Integer insert(UserInfo userInfo);
}

②UserInfoMapperTest测试代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.util.List;
import static org.junit.jupiter.api.Assertions.*;

@Slf4j
//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void insert() {
        UserInfo userInfo = new UserInfo();
        userInfo.setUsername("zhaoliu");
        userInfo.setPassword("zhaoliu");
        userInfo.setAge(18);
        userInfo.setGender(0);
        userInfo.setPhone("123456666");
        Integer result = userInfoMapper.insert(userInfo);
        log.info("inset方法的执行结果:"+result);
    }
}

③测试效果

4.重命名特殊情况

🌟@Insert注解修饰的方法同样可以用@Param进行重命名


💗如果使用@Param将对象进行重命名,#{}需要使用重命名的对象名字.属性来获取


(4)删(Delete)

1.方法

💜两个步骤;使用@Delete注解


💗①先在@Delete注解修饰的方法形参中加入你需要删除哪些限制条件的字段名数据


💗②然后再给@Delete注解中删除语句限制条件的字段名加上#{ }

2.代码及运行结果

①UserInfoMapper代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.*;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    @Delete("delete from userinfo where id=#{id}")
    Integer delete(Integer id); 
}

②UserInfoMapperTest测试代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.util.List;
import static org.junit.jupiter.api.Assertions.*;

@Slf4j
//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void delete() {
        userInfoMapper.delete(6);
        log.info("删除完毕");
    }
}

③运行结果

(5)改(Update)

1.方法

💜两个步骤;使用@Update注解


💗①先在@Update注解修饰的方法形参中加入你需要修改哪些限制条件的字段名数据


💗②然后再给@Update注解中修改语句限制条件的字段名加上#{ }

2.代码及运行结果

①UserInfoMapper代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.*;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    @Update("update usernifo set age=#{age} where id=#{id}")
    Integer update(UserInfo userInfo);
}

②UserInfoMapperTest测试代码

import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.util.List;
import static org.junit.jupiter.api.Assertions.*;

@Slf4j
//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void update() {
        UserInfo userInfo = new UserInfo();
        userInfo.setAge(55);
        userInfo.setId(5);
        Integer result = userInfoMapper.update(userInfo);
        if(result>0){
            log.info("数据修改成功!");
        }
    }
}

③运行结果

(6)查(Select)

1.方法

💜两个步骤;使用@Select注解


💗①先在@Select注解修饰的方法形参中加入你需要修改哪些限制条件的字段名数据


💗②然后再给@Select注解中修改语句限制条件的字段名加上#{ }

2.字段起别名方式

🌟在SQL语句中,给列名起别名,保持别名和实体类属性名一样


💚原因:结果映射;MyBatis会自动的根据数据库的字段名和Java对象的属性名进行映射;如果名称一样就进行赋值


①比如我现在UserInfo实体类的属性名如下

②那我在@Select注解中的查询语句字段保持跟上述一样

③运行结果

3.使用@Results注解方式

💗方法:使用@Results和@Result注解完成起别名,实现结果映射

(1)column:列名;表示要给哪一个字段起别名

(数据库字段)

(2)property:表示起的别名叫什么

(java属性字段)


①UserInfoMapper代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.*;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    @Results({
         @Result(column = "delete_flag",property = "deleteFlag"),
         @Result(column = "create_time",property = "createTime"),
         @Result(column = "update_time",property = "updateTime")
    })
    @Select("select * from userinfo")
    List<UserInfo> selectAll2();
}

②UserInfoMapperTest测试代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.util.List;

@Slf4j
//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void selectAll2() {
        List<UserInfo> userInfoList = userInfoMapper.selectAll2();
        System.out.println(userInfoList);
    }
}

③运行结果

🌟如果其他SQL,也希望可以复用这个映射关系,可以给这个Results定义一个名称

(好处:避免写了太多的@Results,代码可读性差)


💗方法

①给@Results加上一个id属性,表示定义一个名称;再加个value

②再在需要复用Results的方法加个@ResultsMapper注解,然后写value属性为定义的名称


4.驼峰命名(推荐)

💗方法:使用配置文件;使得这两种方式自动进行映射,而不用进行重命名

(原因:通常数据库列使用蛇形命名法进行命名(下划线分割各个单词),而Java属性⼀般遵循驼峰命名法约定)


①application.yml文件

mybatis:
 configuration:
     map-underscore-to-camel-case: true

②application.properties文件

mybatis.configuration.map-underscore-to-camel-case=true

四:MyBatis XML配置文件

(1)准备工作

1.配置数据库

🌟参考《目录二MyBatis入门的(3)数据库配置文件》

2.配置xml的文件路径

💗classpath后面跟的就是resources里所在的目录

比如下面我放的是mapper,那么我就要在resources目录下创建一个mapper目录


①application.yml文件

# 配置 mybatis xml 的⽂件路径,在 resources/mapper 创建所有表的 xml ⽂件
mybatis:
 mapper-locations: classpath:mapper/**Mapper.xml

②application.properties文件

# 配置 mybatis xml 的⽂件路径,在 resources/mapper 创建所有表的 xml ⽂件
mybatis.mapper-locations=classpath:mapper/**Mapper.xml
3.XML的实现

🌟注意:XML和注解是可以共存的

第一步:创建接口

💗如图,创建一个名为UserInfoXMLMapper的接口,然后声明一个方法


package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.Mapper;
import java.util.List;

@Mapper
public interface UserInfoXMLMapper {
    List<UserInfo> selectAll();
}
第二步:创建一个XML文件

💗在classpath后面跟的目录底下创建一个XML文件,在这里写XML代码

(建议这个XML文件名字跟你的接口名字是一样的)


💙比如我的classpath后面是mapper,那我就在mapper目录下添加一个XML文件

第三步:编写XML声明代码

💗注:<mapper namespace>指的是你要实现的是哪个接口

①接口必须是全限定类名

②全限定类名:包+类名

<?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="  ">

</mapper>

🌟比如我的包是package com.hlizoo.demo.mapper,类名是UserInfoXMLMapper,那么我的mapper namespace如下图所示

4.查询所有用户的实现

🌟代码的实现都在mapper标签里

(1)id:接口定义的方法名

(2)resultType:定义的实体类;实体类也必须是全限定类名


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.hlizoo.demo.mapper.UserInfoXMLMapper">

    <select id="selectAll" resultType="com.hlizoo.demo.model.UserInfo">
        select * from userinfo
    </select>

</mapper>

测试代码:

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.web.servlet.handler.UserRoleAuthorizationInterceptor;
import java.util.List;
import static org.junit.jupiter.api.Assertions.*;

@Slf4j
@SpringBootTest
class UserInfoXMLMapperTest {
    @Autowired
    private UserInfoXMLMapper userInfoXMLMapper;

    @Test
    void selectAll() {
        List<UserInfo> userInfoList = userInfoXMLMapper.selectAll();
        log.info(userInfoList.toString());
    }
}

结果:

(2)增(Insert)

1.方法

💗使用<insert></insert>标签;id依旧是接口定义的方法名


💚属性声明和注解一样,使用#{ }


🌟除了select查询需要resultType,其他的增删改查只需要写id即可

2.代码

①UserInfo接口的方法代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.Mapper;
import java.util.List;

@Mapper
public interface UserInfoXMLMapper {
    Integer insert(UserInfo userInfo);
}

②UserInfoXMLMapper的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.hlizoo.demo.mapper.UserInfoXMLMapper">

    <insert id="insert">
        insert into userinfo(username,password,age,gender,phone)
        values(#{username},#{password},#{age},#{gender},#{phone})
    </insert>

</mapper>

③UserInfoXMLMapperTest测试代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.web.servlet.handler.UserRoleAuthorizationInterceptor;
import java.util.List;
import static org.junit.jupiter.api.Assertions.*;

@Slf4j
@SpringBootTest
class UserInfoXMLMapperTest {
    @Autowired
    private UserInfoXMLMapper userInfoXMLMapper;

    @Test
    void insert() {
        UserInfo userInfo = new UserInfo();
        userInfo.setUsername("love");
        userInfo.setPassword("java");
        userInfo.setAge(8);
        userInfo.setGender(1);
        userInfo.setPhone("1589666666");
        Integer result = userInfoXMLMapper.insert(userInfo);
        log.info("影响的行数:"+result);
    }
}

④运行结果

3.返回主键

💗在insert标签中设置useGeneratedKeys和keyProperty属性,和注解一模一样


4.关于重命名

如果在接口方法中形参中对象使用@Param重命名,那么#{}需要使用重命名的名字.属性来获取


💙和注解一模一样

(3)删(Delete)

1.方法

💗使用<delete></delete>标签;id依旧是接口定义的方法名


💚属性声明和注解一样,使用#{ }


🌟除了select查询需要resultType,其他的增删改查只需要写id即可

2.代码

①UserInfo接口的方法代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.Mapper;
import java.util.List;

@Mapper
public interface UserInfoXMLMapper {
     Integer delete(Integer id);
}

②UserInfoXMLMapper的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.hlizoo.demo.mapper.UserInfoXMLMapper">

    <delete id="delete">
        delete from userinfo where id=#{id}
    </delete>

</mapper>

③UserInfoXMLMapperTest测试代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.web.servlet.handler.UserRoleAuthorizationInterceptor;
import java.util.List;
import static org.junit.jupiter.api.Assertions.*;

@Slf4j
@SpringBootTest
class UserInfoXMLMapperTest {
    @Autowired
    private UserInfoXMLMapper userInfoXMLMapper;

    @Test
    void delete() {
        Integer result = userInfoXMLMapper.delete(8);
        log.info("成功影响的行数:"+result);
    }
}

④运行结果

(4)改(Update)

1.方法

💗使用<update></update>标签;id依旧是接口定义的方法名


💚属性声明和注解一样,使用#{ }


🌟除了select查询需要resultType,其他的增删改查只需要写id即可

2.代码

①UserInfo接口的方法代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.Mapper;
import java.util.List;

@Mapper
public interface UserInfoXMLMapper {
     Integer update(UserInfo userInfo);
}

②UserInfoXMLMapper的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.hlizoo.demo.mapper.UserInfoXMLMapper">

    <update id="update">
        update userinfo set gendar=#{} where id=#{id}
    </update>

</mapper>

③UserInfoXMLMapperTest测试代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.web.servlet.handler.UserRoleAuthorizationInterceptor;
import java.util.List;
import static org.junit.jupiter.api.Assertions.*;

@Slf4j
@SpringBootTest
class UserInfoXMLMapperTest {
    @Autowired
    private UserInfoXMLMapper userInfoXMLMapper;

    @Test
    void update() {
        UserInfo userInfo = new UserInfo();
        userInfo.setGender(555);
        userInfo.setId(9);
        userInfoXMLMapper.update(userInfo);
    }
}

④运行结果

(5)查(Select)

1.方法

💗使用<select></select>标签;id是接口定义的方法名,resultType是实体类的全限定类名


💚属性声明和注解一样,使用#{ }

2.结果映射

💗和注解的三种方法一样,还是比较推荐驼峰命名

(只有和第二种的略微不一样,下面我们详细说第二种)

3.使用ResultMap和Result标签

五:多表查询

(1)准备工作

1.思路

①创建一个数据库mybatis_test,在库中创建一张用户表,一张文章表

(用户表的id对应文章表的uid)


②创建一个model包,包中创建两个实体类,一个表示用户实体类,一个表示文章实体类

2.用户表的SQL代码和实体类

①SQL代码

-- 创建数据库
DROP DATABASE IF EXISTS mybatis_test;
CREATE DATABASE mybatis_test DEFAULT CHARACTER SET utf8mb4;
-- 使⽤数据数据
USE mybatis_test;
-- 创建表[⽤⼾表]
DROP TABLE IF EXISTS userinfo;
CREATE TABLE `userinfo` (
`id` INT ( 11 ) NOT NULL AUTO_INCREMENT,
`username` VARCHAR ( 127 ) NOT NULL,
`password` VARCHAR ( 127 ) NOT NULL,
`age` TINYINT ( 4 ) NOT NULL,
`gender` TINYINT ( 4 ) DEFAULT '0' COMMENT '1-男 2-⼥ 0-默认',
`phone` VARCHAR ( 15 ) DEFAULT NULL,
`delete_flag` TINYINT ( 4 ) DEFAULT 0 COMMENT '0-正常, 1-删除',
`create_time` DATETIME DEFAULT now(),
`update_time` DATETIME DEFAULT now(),
PRIMARY KEY ( `id` )
) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;
-- 添加⽤⼾信息
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'admin', 'admin', 18, 1, '18612340001' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'zhangsan', 'zhangsan', 18, 1, '18612340002' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'lisi', 'lisi', 18, 1, '18612340003' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'wangwu', 'wangwu', 18, 1, '18612340004' );

②实体类

package com.hlizoo.demo.model;
import lombok.Data;
import java.util.Date;

@Data
public class UserInfo {
    private Integer id;
    private String username;
    private String password;
    private Integer age;
    private Integer gender;
    private String phone;
    private Integer deleteFlag;
    private Date createTime;
    private Date updateTime;
}
3.文章表的SQL代码和实体类

①SQL代码

-- 创建文章表
DROP TABLE IF EXISTS articleinfo;

CREATE TABLE articleinfo (
 id INT PRIMARY KEY auto_increment,
 title VARCHAR ( 100 ) NOT NULL,
 content TEXT NOT NULL,
 uid INT NOT NULL,
 delete_flag TINYINT ( 4 ) DEFAULT 0 COMMENT '0-正常, 1-删除',
 create_time DATETIME DEFAULT now(),
 update_time DATETIME DEFAULT now() 
) DEFAULT charset 'utf8mb4';

-- 插入测试数据
INSERT INTO articleinfo ( title, content, uid ) VALUES ( 'Java', 'Java正文', 1);

②实体类

package com.hlizoo.demo.model;
import lombok.Data;
import java.util.Date;

@Data
public class ArticleInfo {
    private Integer id;
    private String title;
    private String content;
    private Integer uid;
    private Integer deleteFlag;
    private Date createTime;
    private Date updateTime;
}

(2)根据文章的uid查找用户信息

💗SQL语句:

select ta.*,tb.username,tb.age from articleinfo ta left join userinfo tb on ta.uid=tb.id where ta.uid=1;

(3)MyBatis实现多表查询

🌟关键点:和普通查询一样,只不过需要在你映射对象的实体类加多几个属性

①先在mapper目录下创建一个ArticleInfoMapper接口


②在ArticleInfoMapper接口中实现对应的方法

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.ArticleInfo;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Select;

@Mapper
public interface ArticleInfoMapper {
    @Select("select ta.*,tb.username,tb.age from articleinfo ta left join userinfo tb on ta.uid=tb.id where ta.uid=1")
    ArticleInfo selectArticleAndUserById(Integer articleId);
}

③在ArticleInfoMapperTest完成测试代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.ArticleInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.util.List;
import static org.junit.jupiter.api.Assertions.*;

@Slf4j
@SpringBootTest
class ArticleInfoMapperTest {
    @Autowired
    private ArticleInfoMapper articleInfoMapper;

    @Test
    void selectArticleAndUserById() {
        ArticleInfo articleInfo = articleInfoMapper.selectArticleAndUserById(1);
        log.info(articleInfo.toString());
    }
}

④补充ArticleInfo实体类信息

(因为select查询语句还用到username和age,如果ArticleInfo没有写就无法显示)

package com.hlizoo.demo.model;
import lombok.Data;
import java.util.Date;

@Data
public class ArticleInfo {
    private Integer id;
    private String title;
    private String content;
    private Integer uid;
    private Integer deleteFlag;
    private Date createTime;
    private Date updateTime;

    //补充用户相关信息
    private String username;
    private Integer age;
}

⑤运行结果

六:#{} 和 ${}

(1)#{}的使用

1.Integer类型的变量

🌟观察日志可得:id的值是使用?进行占位的


2.String类型的变量

🌟①#{username}没有使用拼接引号' '


🌟②观察日志可得:username的值是使用?进行占位的


(2)${}的使用

1.Integer类型的变量

🌟观察日志可得:id的值是直接拼接在字符串当中


2.String类型的变量

🌟①'${username}'必须要使用拼接引号' '


🌟②观察日志可得:username的值是直接拼接在字符串当中


(3)#{} 和 ${}的共同之处

💗#{} 和 ${}都是为了去获取变量的值

(4)#{} 和 ${}的区别

1.SQL名称不同

①#{}:使用的是预编译SQL

(用?作为占用符,提前对SQL进行编译, 然后把参数填充到SQL语句中)


②${}:使用的是即时SQL

(直接把变量拼接上去,一起对SQL进行编译)

2.String类型变量的处理不同

①#{}:如果参数类型为String,会自动的加上引号' '


②${}:如果参数类型为String,必须手动加上引号' '

3.性能高低不同

①#{}:属于预编译SQL;性能比较高


②${}:属于即时SQL;性能比较低

4.SQL注入不同

①#{}:不存在SQL注入的安全漏洞问题


②${}:存在SQL注入的安全漏洞问题

(由于${}是手动添加引号,在用户输⼊参数时,在参数中添加⼀些 SQL关键字,达到改变SQL运行结果的目的,也可以完成恶意攻击)


🌟例如下图这个代码,当你的查询条件加入这个,就会查询到所有的数据

5.使用场景

①#{}:实际开发中,能使用#{}就尽量使用


②${}:用于排序查询和模糊查询;除了这两个,其他查询如果用${},必须考虑SQL注入安全问题

(5)排序功能

1.使用场景

💗当使用到SQL语句排序的时候,使用${}


🌟原因:当我使用#{}的时候,它会自动的帮我加上引号,但降序和升序是不需要引号的

2.排序使用#{}的弊端

💗前面说过,#{}会自动的添加上引号,通过下面的例子,来看看为什么排序不建议用#{}


①UserInfoMapper代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.*;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    //sort表示是按升序还是降序排,一会我们给sort参数传入desc或者asc
    @Select("select * from userinfo order by id #{sort}")
    List<UserInfo> selectUserBySort(String sort);
}

②UserInfoMapperTest代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.util.List;

@Slf4j
//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void selectUserBySort() {
        log.info(userInfoMapper.selectUserBySort("asc").toString());
    }
}

③运行结果

💚注意:asc和desc在SQL语句中是不需要添加引号的

3.排序使用${}的好处

 ①UserInfoMapper代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.ibatis.annotations.*;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    //sort表示是按升序还是降序排,一会我们给sort参数传入desc或者asc
    @Select("select * from userinfo order by id ${sort}")
    List<UserInfo> selectUserBySort(String sort);
}

②UserInfoMapperTest代码
package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.util.List;

@Slf4j
//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void selectUserBySort() {
        log.info(userInfoMapper.selectUserBySort("asc").toString());
    }
}

③运行结果

(6)模糊查询

1.使用场景

💗当使用到SQL语句模糊查询的时候,使用${}


🌟原因:当我使用#{}的时候,它会自动的帮我加上引号

2.模糊查询使用#{}的弊端

 ①UserInfoMapper代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.catalina.User;
import org.apache.ibatis.annotations.*;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    @Select("select * from userinfo where username like '%#{username}%'")
    List<UserInfo> selectUserByLike(String username);
}

②UserInfoMapperTest代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.util.List;

@Slf4j
//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void selectUserByLike() {
        log.info(userInfoMapper.selectUserByLike("java").toString());
    }
}

③运行结果

3.模糊查询使用${}的好处

 ①UserInfoMapper代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import org.apache.catalina.User;
import org.apache.ibatis.annotations.*;
import java.util.List;

@Mapper
public interface UserInfoMapper {
    @Select("select * from userinfo where username like '%${username}%'")
    List<UserInfo> selectUserByLike(String username);
}

②UserInfoMapperTest代码

package com.hlizoo.demo.mapper;
import com.hlizoo.demo.model.UserInfo;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.util.List;

@Slf4j
//切记一定要加@SpringBootTest
@SpringBootTest
class UserInfoMapperTest {
    @Autowired
    UserInfoMapper userInfoMapper;

    @Test
    void selectUserByLike() {
        log.info(userInfoMapper.selectUserByLike("java").toString());
    }
}

③运行结果

七:数据库连接池

(1)介绍

💗概念:数据库连接池负责分配、管理和释放数据库连接


💗特点:它允许应用程序重复使用⼀个现有的数据库连接, 而不是再重新创立一个


💗优点:

①减少了网络开销

②资源重用

③提升了系统的性能

(2)无连接池和有连接池的区别

💙①无连接池:每次执行SQL语句,要先创建⼀个新的连接对象,然后执行SQL语句,SQL语句执行完,再关闭连接对象释放资源;这种重复的创建连接,销毁连接比较消耗资源


💙②有连接池:程序启动时,会在数据库连接池中创建⼀定数量的Connection对象,当客户请求数据库连接池,会从数据库连接池中获取Connection对象,然后执行SQL,SQL语句执行完,再把Connection归还给连接池


(3)常见的数据库连接池

①C3P0

②DBCP

③Druid

④Hikari


💜目前比较流行的是Hikari和Druid

(4)HiKari

💗Hikari : SpringBoot默认使用的数据库连接池


(5)Druid

💛如果我们想把默认的数据库连接池切换为Druid数据库连接池,只需要引入相关依赖即可


dependency>
 <groupId>com.alibaba</groupId>
 <artifactId>druid-spring-boot-starter</artifactId>
 <version>1.1.17</version>
</dependency>

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Web】Ctfshow Thinkphp3.2.3代码审计(1)

目录 ①web569 ②web570 ③web571 ④web572 ①web569 基础考察 /index.php/Admin/Login/ctfshowLogin ②web570 提示找路由 查看附件源码 (config.php) 发现定义了一个可执行命令的路由规则 /index.php/ctfshow/assert/eval($_POST[1]) 1system(tac /f*); ③web571 提…

The module to import is incompatible with the current project【鸿蒙开发-BUG已解决】

文章目录 项目场景:问题描述原因分析:解决方案:心得体会:知识点OpenHarmony:HarmonyOS:项目场景: 报错: The module to import is incompatible with the current project 问题描述 希望通过 import module 将该模块引入到我的项目。 导入后出现错误,因为项目和模块…

Scannet v2 数据集介绍以及子集下载展示

Scannet v2 数据集介绍以及子集下载展示 文章目录 Scannet v2 数据集介绍以及子集下载展示参考数据集简介子集scannet_frames_25kscannet_frames_test 下载脚本 download_scannetv2.py 参考 scannet数据集简介和下载-CSDN博客 scannet v2 数据集下载_scannetv2数据集_蓝羽飞鸟的…

pandas分组选中最大值并且新增列

题目 根据每个session_id分组&#xff0c;将popular最大的值设为这个session中所有popular的值 category item_id label popular session_id 0 4729 True 53.0 4069 0 4729 True 53.0 4069 0 4729 True 53.0 4069 0…

C++ PCL点云dscan密度分割三维

程序示例精选 C PCL点云dscan密度分割三维 如需安装运行环境或远程调试&#xff0c;见文章底部个人QQ名片&#xff0c;由专业技术人员远程协助&#xff01; 前言 这篇博客针对《C PCL点云dscan密度分割三维》编写代码&#xff0c;代码整洁&#xff0c;规则&#xff0c;易读。…

hdlbits系列verilog解答(7420 chip)-49

文章目录 一、问题描述二、verilog源码三、仿真结果一、问题描述 本次将实现7420逻辑芯片,它内部有2个4输入的与非门电路,外部有8个输入和2个输出管脚,功能框图如下所示: 二、verilog源码 module top_module ( input p1a, p1b, p1c, p1d,output p1y,input p2a, p2b, p2c…

Proteus仿真--基于数码管显示的频率计设计

本文介绍基于数码管的频率计设计&#xff08;完整仿真源文件及代码见文末链接&#xff09; 仿真图如下 本设计中80C51单片机作为主控&#xff0c;用数码管作为显示模块&#xff0c;按下按键K1后可进行频率测量并显示 仿真运行视频 Proteus仿真--数码管显示的频率计 附完整Pro…

如何使用nginx部署静态资源

Nginx可以作为静态web服务器来部署静态资源&#xff0c;这个静态资源是指在服务端真实存在&#xff0c;并且能够直接展示的一些文件数据&#xff0c;比如常见的静态资源有html页面、css文件、js文件、图片、视频、音频等资源相对于Tomcat服务器来说&#xff0c;Nginx处理静态资…

学习.NET验证模块FluentValidation的基本用法(续3:ASP.NET Core中的调用方式)

FluentValidation模块支持在ASP.NET Core项目中进行手工或自动验证&#xff0c;主要验证方式包括以下三种&#xff1a;   1&#xff09;手工注册验证类&#xff0c;并在控制器或其它模块中调用验证&#xff1b;   2&#xff09;基于ASP.NET验证管道&#xff08;validation …

【版本管理 | Git】Git rebase 命令最佳实践!确定不来看看?

&#x1f935;‍♂️ 个人主页: AI_magician &#x1f4e1;主页地址&#xff1a; 作者简介&#xff1a;CSDN内容合伙人&#xff0c;全栈领域优质创作者。 &#x1f468;‍&#x1f4bb;景愿&#xff1a;旨在于能和更多的热爱计算机的伙伴一起成长&#xff01;&#xff01;&…

webshell之内置函数免杀

原始webshell 查杀的点在于Runtime.getRuntime().exec非常明显的特征 利用ProcessBuilder替换Runtime.getRuntime().exec(cmd) Runtime.getRuntime().exec(cmd)其实最终调用的是ProcessBuilder这个函数&#xff0c;因此我们可以直接利用ProcessBuilder来替换Runtime.getRunti…

css优化滚动条样式

css代码&#xff1a; ::-webkit-scrollbar {width: 6px;height: 6px; }::-webkit-scrollbar-track {background-color: #f1f1f1; }::-webkit-scrollbar-thumb {background-color: #c0c0c0;border-radius: 3px; }最终样式&#xff1a;

大数据面试大厂真题【附答案详细解析】

1.Java基础篇&#xff08;阿里、蚂蚁、字节、携程、快手、杭州银行等&#xff09; 问题&#xff1a;HashMap的底层实现原理 答案&#xff1a; 在jdk1.8之前&#xff0c;hashmap由 数组-链表数据结构组成&#xff0c;在jdk1.8之后hashmap由 数组-链表-红黑树数据结构组成&…

Java抽象类和接口(1)

&#x1f435;本篇文章将对抽象类和接口相关知识进行讲解 一、抽象类 先来看下面的代码&#xff1a; class Shape {public void draw() {System.out.println("画");} } class Cycle extends Shape {public void draw() {System.out.println("圆形");} } …

Leetcode—18.四数之和【中等】

2023每日刷题&#xff08;四十一&#xff09; Leetcode—18.四数之和 实现代码 class Solution { public:vector<vector<int>> fourSum(vector<int>& nums, int target) {vector<vector<int>> ans;sort(nums.begin(), nums.end());int n …

C/C++ 常用加密与解密算法

计算机安全和数据隐私是现代应用程序设计中至关重要的方面。为了确保数据的机密性和完整性&#xff0c;常常需要使用加密和解密算法。C是一种广泛使用的编程语言&#xff0c;提供了许多加密和解密算法的实现。本文将介绍一些在C中常用的加密与解密算法&#xff0c;这其中包括Xo…

「C++」入门

&#x1f387;个人主页&#xff1a;Ice_Sugar_7 &#x1f387;所属专栏&#xff1a;C启航 &#x1f387;欢迎点赞收藏加关注哦&#xff01; 文章目录 &#x1f349;前言&#x1f349;命名空间&#x1f34c;访问命名空间中的元素&#x1f34c;同名命名空间&#x1f34c;展开&…

企业编码生成程序Python毕业设计

&#xff08;1&#xff09;生成6位数字防伪编码。当用户在主程序界面中输入数字“1”菜单项时&#xff0c;将进入“生成6位数字防伪编码 &#xff08;213563型&#xff09;”的功能执行任务。此时要求输入生成防伪码的数量&#xff0c;可以根据需要输入生成防伪码的数量。按下&…

京东数据采集(京东数据运营):怎样快速获取京东市场大数据?

相信京东平台的很多品牌方们都有做数据分析的需求&#xff0c;但面对多而杂的市场数据&#xff0c;很多运营者都没有思路。单依靠肉眼来看&#xff0c;很多商品的类目、销售成绩、价格分布等运营者也未必清楚。 其实对于京东平台上市场数据的获取&#xff0c;品牌可以直接借助一…

使用VC++设计程序:实现常见的三种图像插值算法:最近邻插值,双线性插值,立方卷积插值

图像放大的三种插值算法 获取源工程可访问gitee可在此工程的基础上进行学习。 该工程的其他文章&#xff1a; 01- 一元熵值、二维熵值 02- 图像平移变换&#xff0c;图像缩放、图像裁剪、图像对角线镜像以及图像的旋转 03-邻域平均平滑算法、中值滤波算法、K近邻均值滤波器 04-…