首页
壁纸
统计
友链
Search
1
Vue2详细笔记
37 阅读
2
Nestjs概述-中文
19 阅读
3
ExpressAPI
17 阅读
4
答辩
14 阅读
5
JavaScript企业数据处理实用指南
13 阅读
Nodejs
Vue
Java
Msql
登录
Search
Wasnl
累计撰写
36
篇文章
累计收到
1
条评论
首页
栏目
Nodejs
Vue
Java
Msql
页面
壁纸
统计
友链
搜索到
36
篇与
的结果
2026-04-03
SpringBoot
Spring Boot 2.7 入门到精通笔记(苍穹外卖实战版)一、Spring Boot 概述1.1 什么是 Spring BootSpring Boot 是 Spring 框架的“脚手架”,核心设计理念是 约定优于配置。与传统 Spring 的区别:对比项传统 SpringSpring Boot配置方式繁琐的 XML 配置文件注解 + 自动配置依赖管理手动管理版本、处理冲突起步依赖(Starter)自动引入部署方式打包成 WAR 部署到 Tomcat内嵌 Tomcat,直接 java -jar开发效率需要大量样板代码只需关心业务逻辑1.2 Spring Boot 核心特性自动配置:根据类路径中的依赖自动配置 Bean起步依赖:通过 Starter 机制简化依赖管理内嵌容器:默认集成 Tomcat/Jetty/Undertow生产就绪:Actuator 提供监控、健康检查等能力1.3 Spring Boot 2.7 新特性(重点)Spring Boot 2.7 是一个重要版本,2.5 已停止维护[reference:0]。自动配置变更:自动配置注册文件从 META-INF/spring.factories 迁移到 META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports新增 @AutoConfiguration 注解,用于标识自动配置类[reference:1]支持 after/before 属性进行配置排序其他新特性:支持 GraphQL(spring-boot-starter-graphql)支持 Podman 容器引擎Web Server SSL 增强[reference:2]1.4 苍穹外卖项目概述苍穹外卖是一个企业级 O2O 外卖平台,分为管理端和用户端(微信小程序):角色功能管理端员工管理、菜品/套餐管理、订单管理、数据统计用户端登录授权、浏览点餐、购物车结算、订单追踪技术栈:Spring Boot 2.7 + Spring MVC + MyBatis-Plus + MySQL + Redis + OSS(阿里云)+ AOP + Swagger(Knife4j)+ WebSocket[reference:3]二、快速开始2.1 项目结构(苍穹外卖)苍穹外卖采用 Maven 多模块结构,各模块职责明确,便于维护和扩展:sky-take-out(父工程) ├── pom.xml # 统一依赖版本管理 ├── sky-common/ # 公共模块:工具类、常量、异常类 │ ├── constant/ # 常量定义(如JwtClaimsConstant) │ ├── context/ # ThreadLocal上下文 │ ├── exception/ # 自定义异常 │ ├── properties/ # 配置属性类 │ └── utils/ # 工具类(JWT、阿里OSS、微信支付等) ├── sky-pojo/ # 实体模块:Entity、DTO、VO │ ├── entity/ # 数据库实体类(与表一一对应) │ ├── dto/ # 数据传输对象(层间传递) │ └── vo/ # 视图对象(返回给前端) └── sky-server/ # 后端服务模块(核心业务代码) ├── config/ # 配置类(Redis、Swagger、WebMvc等) ├── controller/ # 控制器层 ├── service/ # 业务逻辑层 ├── mapper/ # 数据访问层 ├── interceptor/ # 拦截器(如JWT认证拦截器) ├── aspect/ # AOP切面(公共字段填充) └── SkyApplication.java # 启动类POJO 分类说明:Entity:实体类,与数据库表一一对应DTO(Data Transfer Object):数据传输对象,用于层间传递数据VO(View Object):视图对象,专门返回给前端展示[reference:4]2.2 父工程 POM 配置(版本管理)<!-- sky-take-out/pom.xml --> <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.7.0</version> <relativePath/> </parent> <groupId>com.sky</groupId> <artifactId>sky-take-out</artifactId> <version>1.0-SNAPSHOT</version> <packaging>pom</packaging> <properties> <java.version>1.8</java.version> <mybatis-plus.version>3.5.2</mybatis-plus.version> </properties> <dependencies> <!-- 统一管理的依赖在这里声明版本 --> </dependencies>2.3 子模块依赖配置<!-- sky-server/pom.xml --> <dependencies> <!-- Spring Boot Web Starter --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <!-- MyBatis-Plus Starter --> <dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-boot-starter</artifactId> <version>3.5.2</version> </dependency> <!-- MySQL 驱动 --> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <scope>runtime</scope> </dependency> <!-- Redis --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency> <!-- 引用公共模块 --> <dependency> <groupId>com.sky</groupId> <artifactId>sky-common</artifactId> <version>1.0-SNAPSHOT</version> </dependency> </dependencies>2.4 启动类package com.sky; import lombok.extern.slf4j.Slf4j; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.transaction.annotation.EnableTransactionManagement; @Slf4j @SpringBootApplication @EnableTransactionManagement // 开启事务管理 public class SkyApplication { public static void main(String[] args) { SpringApplication.run(SkyApplication.class, args); log.info("苍穹外卖后端服务启动成功!"); } }@SpringBootApplication 复合注解解析:注解作用@SpringBootConfiguration标识为配置类(@Configuration 的派生)@ComponentScan开启组件扫描,扫描启动类所在包及子包@EnableAutoConfiguration开启自动配置(核心)自动配置原理(5 步必背):@SpringBootApplication → @EnableAutoConfiguration@Import(AutoConfigurationImportSelector.class) 导入选择器加载 META-INF/spring.factories(或 2.7 新路径 AutoConfiguration.imports)读取自动配置类的全限定类名通过 @ConditionalOnXxx 条件注解判断是否加载三、配置文件3.1 application.yml(苍穹外卖示例)server: port: 8080 spring: datasource: url: jdbc:mysql://localhost:3306/sky_take_out?serverTimezone=Asia/Shanghai&useUnicode=true&characterEncoding=utf-8 username: root password: 123456 driver-class-name: com.mysql.cj.jdbc.Driver redis: host: localhost port: 6379 database: 0 main: allow-circular-references: true # 允许循环依赖 mybatis-plus: configuration: log-impl: org.apache.ibatis.logging.stdout.StdOutImpl # 打印 SQL 日志 global-config: db-config: id-type: auto # 主键自增 mapper-locations: classpath:mapper/*.xml sky: jwt: admin-secret-key: "itcast" admin-ttl: 7200000 admin-token-name: "token"3.2 多环境配置(必背)创建多个配置文件,通过 spring.profiles.active 激活:文件环境application.yml默认配置(公共)application-dev.yml开发环境application-prod.yml生产环境# application.yml(主配置) spring: profiles: active: dev # 激活开发环境 # application-dev.yml server: port: 8081 spring: config: activate: on-profile: dev # application-prod.yml server: port: 80 spring: config: activate: on-profile: prod激活方式:# 命令行激活 java -jar sky-server.jar --spring.profiles.active=prod # IDEA VM options -Dspring.profiles.active=prod四、分层架构(苍穹外卖实战)4.1 分层架构图解(外卖公司比喻)层级角色苍穹外卖示例Controller客服经理(接电话)接收 HTTP 请求,返回 JSON 响应Service业务主管(管流程)实现业务逻辑,调用 MapperMapper仓库管理员(管库存)执行 SQL,操作数据库Entity货物清单与数据库表一一对应的实体4.2 Entity 层(实体类)package com.sky.entity; import lombok.AllArgsConstructor; import lombok.Builder; import lombok.Data; import lombok.NoArgsConstructor; import java.io.Serializable; import java.time.LocalDateTime; @Data @Builder @NoArgsConstructor @AllArgsConstructor public class Employee implements Serializable { private static final long serialVersionUID = 1L; private Long id; private String username; private String name; private String password; private String phone; private String sex; private String idNumber; private Integer status; private LocalDateTime createTime; private LocalDateTime updateTime; private Long createUser; private Long updateUser; }4.3 Mapper 层(数据访问层)package com.sky.mapper; import com.sky.entity.Employee; import org.apache.ibatis.annotations.Mapper; import org.apache.ibatis.annotations.Select; @Mapper public interface EmployeeMapper { @Select("select * from employee where username = #{username}") Employee getByUsername(String username); // 分页查询员工 Page<Employee> pageQuery(EmployeePageQueryDTO pageQueryDTO); // 更新员工信息 void update(Employee employee); }MyBatis-Plus 增强版(推荐) :// 继承 BaseMapper,自动拥有 CRUD 方法 @Mapper public interface DishMapper extends BaseMapper<Dish> { // 无需写任何基础 CRUD,BaseMapper 已提供: // insert(), deleteById(), updateById(), selectById(), selectList() 等 } // Service 层还可以继承 IService @Service public class DishService extends ServiceImpl<DishMapper, Dish> { // 自动拥有 saveBatch(), updateBatchById() 等批量方法 }4.4 Service 层(业务逻辑层)package com.sky.service.impl; import com.sky.context.BaseContext; import com.sky.dto.EmployeeDTO; import com.sky.entity.Employee; import com.sky.mapper.EmployeeMapper; import com.sky.service.EmployeeService; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.BeanUtils; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import org.springframework.util.DigestUtils; @Slf4j @Service public class EmployeeServiceImpl implements EmployeeService { @Autowired private EmployeeMapper employeeMapper; @Override @Transactional // 开启事务 public void save(EmployeeDTO employeeDTO) { Employee employee = new Employee(); // 拷贝属性(DTO → Entity) BeanUtils.copyProperties(employeeDTO, employee); // 设置默认密码(MD5 加密) employee.setPassword(DigestUtils.md5DigestAsHex("123456".getBytes())); employee.setStatus(1); employee.setCreateTime(LocalDateTime.now()); employee.setUpdateTime(LocalDateTime.now()); // 从 ThreadLocal 获取当前操作用户 ID employee.setCreateUser(BaseContext.getCurrentId()); employee.setUpdateUser(BaseContext.getCurrentId()); employeeMapper.insert(employee); } }4.5 Controller 层(控制层)package com.sky.controller.admin; import com.sky.dto.EmployeeDTO; import com.sky.dto.EmployeeLoginDTO; import com.sky.dto.EmployeePageQueryDTO; import com.sky.result.PageResult; import com.sky.result.Result; import com.sky.service.EmployeeService; import com.sky.vo.EmployeeLoginVO; import io.swagger.annotations.Api; import io.swagger.annotations.ApiOperation; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.*; @RestController @RequestMapping("/admin/employee") @Api(tags = "员工管理接口") @Slf4j public class EmployeeController { @Autowired private EmployeeService employeeService; @PostMapping("/login") @ApiOperation("员工登录") public Result<EmployeeLoginVO> login(@RequestBody EmployeeLoginDTO loginDTO) { log.info("员工登录:{}", loginDTO.getUsername()); EmployeeLoginVO employeeLoginVO = employeeService.login(loginDTO); return Result.success(employeeLoginVO); } @PostMapping @ApiOperation("新增员工") public Result save(@RequestBody EmployeeDTO employeeDTO) { log.info("新增员工:{}", employeeDTO); employeeService.save(employeeDTO); return Result.success(); } @GetMapping("/page") @ApiOperation("员工分页查询") public Result<PageResult> page(EmployeePageQueryDTO pageQueryDTO) { log.info("员工分页查询:{}", pageQueryDTO); PageResult pageResult = employeeService.pageQuery(pageQueryDTO); return Result.success(pageResult); } }4.6 分层调用链路总结HTTP 请求 ↓ Controller(接收参数,调用 Service) ↓ Service(业务逻辑,调用 Mapper) ↓ Mapper(执行 SQL,操作数据库) ↓ Entity(数据载体) ↓ 返回结果 → VO → JSON 响应⚠️ 常见错误:❌ Controller 直接调用 Mapper(越级)❌ Mapper 层写业务判断逻辑❌ Service 层写 SQL 拼接五、核心注解速查表5.1 组件注解注解用途层级@Component通用组件,交给 Spring 管理任意@ControllerWeb 控制器(配合 @ResponseBody 返回 JSON)表现层@RestController@Controller + @ResponseBody 的合并表现层@Service业务逻辑层组件业务层@Repository数据访问层组件,会转换持久层异常数据层@Repository 的特殊功能:将数据库操作抛出的异常自动转换为 Spring 统一的数据访问异常体系[reference:5]。5.2 请求映射注解注解作用@RequestMapping通用请求映射,可指定 method@GetMapping处理 GET 请求@PostMapping处理 POST 请求@PutMapping处理 PUT 请求@DeleteMapping处理 DELETE 请求5.3 参数绑定注解注解作用示例@PathVariable获取 URL 路径参数/user/{id} → @PathVariable Long id@RequestParam获取 URL 查询参数?name=张三 → @RequestParam String name@RequestBody获取请求体(JSON → 对象)@RequestBody EmployeeDTO dto@RequestHeader获取请求头@RequestHeader("token") String token5.4 依赖注入注解注解作用@Autowired按类型自动注入(Spring 原生)@Resource按名称自动注入(JSR-250)@Qualifier配合 @Autowired,指定 Bean 名称@Value注入配置文件中的值(如 ${server.port})@ConfigurationProperties批量注入配置属性5.5 配置与条件注解注解作用@Configuration标识配置类@Bean在配置类中声明 Bean@ConditionalOnClass类路径存在指定类时才生效@ConditionalOnMissingBean容器中没有指定 Bean 时才生效@ConditionalOnProperty配置文件存在指定属性时才生效六、全局异常处理6.1 苍穹外卖全局异常处理器package com.sky.handler; import com.sky.constant.MessageConstant; import com.sky.exception.BaseException; import com.sky.result.Result; import lombok.extern.slf4j.Slf4j; import org.springframework.web.bind.annotation.ExceptionHandler; import org.springframework.web.bind.annotation.RestControllerAdvice; @RestControllerAdvice @Slf4j public class GlobalExceptionHandler { // 捕获业务异常 @ExceptionHandler(BaseException.class) public Result exceptionHandler(BaseException ex) { log.error("业务异常:{}", ex.getMessage()); return Result.error(ex.getMessage()); } // 捕获 SQL 异常(如用户名重复) @ExceptionHandler(DuplicateKeyException.class) public Result handleDuplicateKeyException(DuplicateKeyException ex) { log.error("SQL 异常:{}", ex.getMessage()); return Result.error(MessageConstant.ALREADY_EXISTS); } // 兜底异常处理 @ExceptionHandler(Exception.class) public Result exceptionHandler(Exception ex) { log.error("系统异常:{}", ex.getMessage()); return Result.error(MessageConstant.UNKNOWN_ERROR); } }6.2 统一响应对象package com.sky.result; import lombok.Data; import java.io.Serializable; @Data public class Result<T> implements Serializable { private Integer code; // 1 成功,0 失败 private String msg; private T data; public static <T> Result<T> success() { Result<T> result = new Result<>(); result.code = 1; return result; } public static <T> Result<T> success(T data) { Result<T> result = new Result<>(); result.code = 1; result.data = data; return result; } public static <T> Result<T> error(String msg) { Result<T> result = new Result<>(); result.code = 0; result.msg = msg; return result; } }七、JWT 认证与拦截器7.1 JWT 工具类package com.sky.utils; import io.jsonwebtoken.*; import lombok.Data; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.stereotype.Component; import java.util.Date; import java.util.Map; @Data @Component @ConfigurationProperties(prefix = "sky.jwt") public class JwtUtil { private String adminSecretKey; private Long adminTtl; private String adminTokenName; // 生成 JWT 令牌 public String createJWT(Map<String, Object> claims, Long ttlMillis) { long nowMillis = System.currentTimeMillis(); Date now = new Date(nowMillis); JwtBuilder builder = Jwts.builder() .setClaims(claims) .setIssuedAt(now) .signWith(SignatureAlgorithm.HS256, adminSecretKey); if (ttlMillis > 0) { builder.setExpiration(new Date(nowMillis + ttlMillis)); } return builder.compact(); } // 解析 JWT 令牌 public Claims parseJWT(String token) { return Jwts.parser() .setSigningKey(adminSecretKey) .parseClaimsJws(token) .getBody(); } }7.2 JWT 认证拦截器package com.sky.interceptor; import com.sky.constant.JwtClaimsConstant; import com.sky.context.BaseContext; import com.sky.properties.JwtProperties; import com.sky.utils.JwtUtil; import io.jsonwebtoken.Claims; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import org.springframework.web.method.HandlerMethod; import org.springframework.web.servlet.HandlerInterceptor; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; @Component @Slf4j public class JwtTokenAdminInterceptor implements HandlerInterceptor { @Autowired private JwtUtil jwtUtil; @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) { // 判断当前拦截的是否为 Controller 方法 if (!(handler instanceof HandlerMethod)) { return true; } // 从请求头中获取令牌 String token = request.getHeader(jwtUtil.getAdminTokenName()); // 解析令牌 try { Claims claims = jwtUtil.parseJWT(token); Long empId = Long.valueOf(claims.get(JwtClaimsConstant.EMP_ID).toString()); // 将员工 ID 存入 ThreadLocal(供后续业务使用) BaseContext.setCurrentId(empId); log.info("当前员工 ID:{}", empId); return true; } catch (Exception ex) { log.error("JWT 解析失败:{}", ex.getMessage()); response.setStatus(401); return false; } } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { // 请求结束后清除 ThreadLocal,防止内存泄漏 BaseContext.removeCurrentId(); } }7.3 ThreadLocal 工具类(苍穹外卖)package com.sky.context; public class BaseContext { private static ThreadLocal<Long> threadLocal = new ThreadLocal<>(); public static void setCurrentId(Long id) { threadLocal.set(id); } public static Long getCurrentId() { return threadLocal.get(); } public static void removeCurrentId() { threadLocal.remove(); } }ThreadLocal 原理:每个线程拥有自己的变量副本,线程隔离。用完必须 remove(),否则在线程池环境下会导致内存泄漏[reference:6]。7.4 注册拦截器package com.sky.config; import com.sky.interceptor.JwtTokenAdminInterceptor; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Configuration; import org.springframework.web.servlet.config.annotation.InterceptorRegistry; import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport; @Configuration public class WebMvcConfig extends WebMvcConfigurationSupport { @Autowired private JwtTokenAdminInterceptor jwtTokenAdminInterceptor; @Override protected void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(jwtTokenAdminInterceptor) .addPathPatterns("/admin/**") // 拦截所有 /admin 开头的请求 .excludePathPatterns("/admin/employee/login") // 排除登录接口 .excludePathPatterns("/admin/employee/logout"); } }八、AOP 切面编程(公共字段自动填充)8.1 自定义注解package com.sky.annotation; import com.sky.enumeration.OperationType; import java.lang.annotation.*; @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) public @interface AutoFill { OperationType value(); // UPDATE 或 INSERT }8.2 切面类实现package com.sky.aspect; import com.sky.annotation.AutoFill; import com.sky.constant.AutoFillConstant; import com.sky.context.BaseContext; import com.sky.enumeration.OperationType; import lombok.extern.slf4j.Slf4j; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut; import org.aspectj.lang.reflect.MethodSignature; import org.springframework.stereotype.Component; import java.lang.reflect.Method; import java.time.LocalDateTime; @Aspect @Component @Slf4j public class AutoFillAspect { // 切点:标注了 @AutoFill 注解的方法 @Pointcut("@annotation(com.sky.annotation.AutoFill)") public void autoFillPointCut() {} @Before("autoFillPointCut()") public void autoFill(JoinPoint joinPoint) { // 获取当前被拦截的方法上的注解 MethodSignature signature = (MethodSignature) joinPoint.getSignature(); AutoFill autoFill = signature.getMethod().getAnnotation(AutoFill.class); OperationType operationType = autoFill.value(); // 获取方法的参数(实体对象) Object[] args = joinPoint.getArgs(); if (args == null || args.length == 0) return; Object entity = args[0]; // 准备要赋的值 LocalDateTime now = LocalDateTime.now(); Long currentId = BaseContext.getCurrentId(); // 根据操作类型设置不同字段 if (operationType == OperationType.INSERT) { // 插入操作:设置 createTime, updateTime, createUser, updateUser try { Method setCreateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_CREATE_TIME, LocalDateTime.class); Method setUpdateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_TIME, LocalDateTime.class); Method setCreateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_CREATE_USER, Long.class); Method setUpdateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_USER, Long.class); setCreateTime.invoke(entity, now); setUpdateTime.invoke(entity, now); setCreateUser.invoke(entity, currentId); setUpdateUser.invoke(entity, currentId); } catch (Exception e) { log.error("公共字段填充失败:{}", e.getMessage()); } } else if (operationType == OperationType.UPDATE) { // 更新操作:设置 updateTime, updateUser try { Method setUpdateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_TIME, LocalDateTime.class); Method setUpdateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_USER, Long.class); setUpdateTime.invoke(entity, now); setUpdateUser.invoke(entity, currentId); } catch (Exception e) { log.error("公共字段填充失败:{}", e.getMessage()); } } } }8.3 使用示例@Mapper public interface EmployeeMapper { @AutoFill(OperationType.INSERT) void insert(Employee employee); @AutoFill(OperationType.UPDATE) void update(Employee employee); }九、缓存与 Redis9.1 Spring Cache 配置package com.sky.config; import org.springframework.cache.annotation.EnableCaching; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.data.redis.cache.RedisCacheConfiguration; import org.springframework.data.redis.serializer.RedisSerializationContext; import org.springframework.data.redis.serializer.StringRedisSerializer; import java.time.Duration; @Configuration @EnableCaching public class RedisConfig { @Bean public RedisCacheConfiguration cacheConfiguration() { return RedisCacheConfiguration.defaultCacheConfig() .entryTtl(Duration.ofMinutes(30)) // 缓存过期时间 30 分钟 .disableCachingNullValues() // 不缓存 null 值 .serializeKeysWith(RedisSerializationContext.SerializationPair .fromSerializer(new StringRedisSerializer())); } }9.2 缓存注解使用@Service @Slf4j public class DishServiceImpl implements DishService { @Autowired private DishMapper dishMapper; // 查询时缓存:key = "dish:categoryId:" + categoryId @Cacheable(value = "dish", key = "'dish:categoryId:' + #categoryId") public List<Dish> getByCategoryId(Long categoryId) { return dishMapper.getByCategoryId(categoryId); } // 更新时清空缓存 @CacheEvict(value = "dish", key = "'dish:categoryId:' + #dishDTO.categoryId") public void updateWithFlavor(DishDTO dishDTO) { // 更新业务逻辑 } // 批量清空缓存(key 模糊匹配) @CacheEvict(value = "dish", allEntries = true) public void deleteBatch(List<Long> ids) { // 删除业务逻辑 } }9.3 常用 Redis 操作(StringRedisTemplate)@Service public class ShopStatusService { @Autowired private StringRedisTemplate redisTemplate; private static final String SHOP_STATUS_KEY = "SHOP_STATUS"; public void setStatus(Integer status) { redisTemplate.opsForValue().set(SHOP_STATUS_KEY, String.valueOf(status)); } public Integer getStatus() { String status = redisTemplate.opsForValue().get(SHOP_STATUS_KEY); return status == null ? 0 : Integer.parseInt(status); } }十、Spring Task 定时任务10.1 开启定时任务@SpringBootApplication @EnableScheduling // 开启定时任务 public class SkyApplication { public static void main(String[] args) { SpringApplication.run(SkyApplication.class, args); } }10.2 订单状态定时处理package com.sky.task; import com.sky.service.OrderService; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Component; @Component @Slf4j public class OrderTask { @Autowired private OrderService orderService; // 每天凌晨 1 点处理超时未支付的订单 @Scheduled(cron = "0 0 1 * * ?") public void processTimeoutOrder() { log.info("定时任务:处理超时未支付订单"); orderService.processTimeoutOrders(); } // 每分钟执行一次,处理配送中的订单(送达后自动完成) @Scheduled(cron = "0 * * * * ?") public void processDeliveryOrder() { log.info("定时任务:处理配送中订单"); orderService.processDeliveryOrders(); } }10.3 Cron 表达式速查字段允许值特殊字符秒0-59, - * /分0-59, - * /时0-23, - * /日1-31, - * ? / L W月1-12, - * /周0-7(0 或 7 都代表周日), - * ? / L #年(可选)空或 1970-2099, - * /常用表达式:0 0 1 * * ? → 每天凌晨 1 点0 */5 * * * ? → 每 5 分钟一次0 0 12 * * MON-FRI → 工作日中午 12 点十一、WebSocket 实时通信11.1 WebSocket 配置package com.sky.config; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.web.socket.server.standard.ServerEndpointExporter; @Configuration public class WebSocketConfig { @Bean public ServerEndpointExporter serverEndpointExporter() { return new ServerEndpointExporter(); } }11.2 WebSocket 服务端package com.sky.websocket; import lombok.extern.slf4j.Slf4j; import org.springframework.stereotype.Component; import javax.websocket.*; import javax.websocket.server.PathParam; import javax.websocket.server.ServerEndpoint; import java.util.concurrent.ConcurrentHashMap; @Component @ServerEndpoint("/ws/{sid}") @Slf4j public class WebSocketServer { // 存储所有在线会话(线程安全) private static ConcurrentHashMap<String, Session> sessionMap = new ConcurrentHashMap<>(); @OnOpen public void onOpen(Session session, @PathParam("sid") String sid) { sessionMap.put(sid, session); log.info("WebSocket 连接建立:{}", sid); } @OnClose public void onClose(@PathParam("sid") String sid) { sessionMap.remove(sid); log.info("WebSocket 连接关闭:{}", sid); } @OnMessage public void onMessage(String message, Session session) { log.info("收到消息:{}", message); } @OnError public void onError(Session session, Throwable error) { log.error("WebSocket 错误:{}", error.getMessage()); } // 发送消息给指定用户 public void sendToClient(String sid, String message) { Session session = sessionMap.get(sid); if (session != null && session.isOpen()) { try { session.getBasicRemote().sendText(message); log.info("发送消息给 {}:{}", sid, message); } catch (Exception e) { log.error("发送失败:{}", e.getMessage()); } } } }11.3 来单提醒/催单场景@Service public class OrderServiceImpl implements OrderService { @Autowired private WebSocketServer webSocketServer; // 用户下单 → 来单提醒(推送给管理端) public void submit(OrdersSubmitDTO submitDTO) { // ... 下单业务逻辑 // 来单提醒 Map<String, Object> reminder = new HashMap<>(); reminder.put("type", 1); // 1=来单提醒 reminder.put("orderId", order.getId()); reminder.put("content", "您有一笔新订单"); webSocketServer.sendToClient("admin", JSON.toJSONString(reminder)); } // 用户催单 → 推送给管理端 public void reminder(Long orderId) { Map<String, Object> reminder = new HashMap<>(); reminder.put("type", 2); // 2=用户催单 reminder.put("orderId", orderId); reminder.put("content", "用户催单啦"); webSocketServer.sendToClient("admin", JSON.toJSONString(reminder)); } }十二、文件上传(阿里云 OSS)12.1 OSS 配置属性类package com.sky.properties; import lombok.Data; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.stereotype.Component; @Data @Component @ConfigurationProperties(prefix = "sky.alioss") public class AliOssProperties { private String endpoint; private String accessKeyId; private String accessKeySecret; private String bucketName; }12.2 OSS 工具类package com.sky.utils; import com.aliyun.oss.OSS; import com.aliyun.oss.OSSClientBuilder; import com.sky.properties.AliOssProperties; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import org.springframework.web.multipart.MultipartFile; import java.io.IOException; import java.util.UUID; @Component @Slf4j public class AliOssUtil { @Autowired private AliOssProperties aliOssProperties; public String upload(MultipartFile file) { String endpoint = aliOssProperties.getEndpoint(); String accessKeyId = aliOssProperties.getAccessKeyId(); String accessKeySecret = aliOssProperties.getAccessKeySecret(); String bucketName = aliOssProperties.getBucketName(); // 生成唯一文件名(防止覆盖) String originalFilename = file.getOriginalFilename(); String extension = originalFilename.substring(originalFilename.lastIndexOf(".")); String objectName = UUID.randomUUID().toString() + extension; OSS ossClient = new OSSClientBuilder().build(endpoint, accessKeyId, accessKeySecret); try { ossClient.putObject(bucketName, objectName, file.getInputStream()); // 返回文件访问 URL return "https://" + bucketName + "." + endpoint + "/" + objectName; } catch (IOException e) { log.error("文件上传失败:{}", e.getMessage()); throw new RuntimeException(e); } finally { ossClient.shutdown(); } } }12.3 通用上传接口@RestController @RequestMapping("/admin/common") @Api(tags = "通用接口") @Slf4j public class CommonController { @Autowired private AliOssUtil aliOssUtil; @PostMapping("/upload") @ApiOperation("文件上传") public Result<String> upload(MultipartFile file) { log.info("文件上传:{}", file.getOriginalFilename()); String url = aliOssUtil.upload(file); return Result.success(url); } }十三、Knife4j + Swagger 接口文档13.1 添加依赖<dependency> <groupId>com.github.xiaoymin</groupId> <artifactId>knife4j-spring-boot-starter</artifactId> <version>3.0.3</version> </dependency>13.2 Swagger 配置类package com.sky.config; import com.github.xiaoymin.knife4j.spring.annotations.EnableKnife4j; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import springfox.documentation.builders.ApiInfoBuilder; import springfox.documentation.builders.PathSelectors; import springfox.documentation.builders.RequestHandlerSelectors; import springfox.documentation.service.ApiInfo; import springfox.documentation.spi.DocumentationType; import springfox.documentation.spring.web.plugins.Docket; import springfox.documentation.swagger2.annotations.EnableSwagger2; @Configuration @EnableSwagger2 @EnableKnife4j public class SwaggerConfig { @Bean public Docket adminApi() { return new Docket(DocumentationType.SWAGGER_2) .groupName("管理端接口") .apiInfo(apiInfo()) .select() .apis(RequestHandlerSelectors.basePackage("com.sky.controller.admin")) .paths(PathSelectors.any()) .build(); } @Bean public Docket userApi() { return new Docket(DocumentationType.SWAGGER_2) .groupName("用户端接口") .apiInfo(apiInfo()) .select() .apis(RequestHandlerSelectors.basePackage("com.sky.controller.user")) .paths(PathSelectors.any()) .build(); } private ApiInfo apiInfo() { return new ApiInfoBuilder() .title("苍穹外卖 API 文档") .version("1.0") .description("苍穹外卖后端接口文档") .build(); } }13.3 使用示例@Api(tags = "员工管理接口") @RestController @RequestMapping("/admin/employee") public class EmployeeController { @PostMapping("/login") @ApiOperation("员工登录") @ApiImplicitParams({ @ApiImplicitParam(name = "username", value = "用户名", required = true), @ApiImplicitParam(name = "password", value = "密码", required = true) }) public Result<EmployeeLoginVO> login(@RequestBody EmployeeLoginDTO loginDTO) { // ... } }访问地址:http://localhost:8080/doc.html十四、常用工具类速查14.1 BeanUtils(属性拷贝)// DTO → Entity Employee employee = new Employee(); BeanUtils.copyProperties(employeeDTO, employee);14.2 DigestUtils(MD5 加密)// 密码 MD5 加密 String encryptedPassword = DigestUtils.md5DigestAsHex("123456".getBytes());14.3 Jackson(JSON 序列化)// 对象 → JSON 字符串 String jsonStr = JSON.toJSONString(object); // JSON 字符串 → 对象 User user = JSON.parseObject(jsonStr, User.class);14.4 LocalDateTime(时间处理)// 获取当前时间 LocalDateTime now = LocalDateTime.now(); // 时间格式化 DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); String formatted = now.format(formatter);十五、项目部署与运行15.1 打包# 在项目根目录执行 mvn clean package # 跳过测试打包 mvn clean package -DskipTests15.2 运行# 运行 JAR 包 java -jar sky-server.jar # 指定环境运行 java -jar sky-server.jar --spring.profiles.active=prod # 后台运行 nohup java -jar sky-server.jar > app.log 2>&1 &15.3 Nginx 反向代理配置server { listen 80; server_name api.sky.com; location / { proxy_pass http://127.0.0.1:8080; proxy_set_header Host $host; proxy_set_header X-Real-IP $remote_addr; } }十六、快速记忆口诀启动流程SpringApplication.run → 初始化上下文 → 加载自动配置 → 启动内嵌容器 → 注册 Controller → 应用就绪自动配置@SpringBootApplication → @EnableAutoConfiguration → @Import → spring.factories → @ConditionalOnXxx分层架构Controller 接请求,Service 写逻辑,Mapper 查数据库,Entity 存数据JWT 认证登录签发 Token → 请求携带 Header → 拦截器校验 → ThreadLocal 存用户 ID → 业务使用 → 请求结束清除缓存注解@Cacheable 查缓存,@CachePut 更新缓存,@CacheEvict 清缓存定时任务@EnableScheduling 开启,@Scheduled(cron) 标记,Cron 表达式定时间全局异常@RestControllerAdvice 全局处理,@ExceptionHandler 分类捕获本笔记基于 Spring Boot 2.7 + 苍穹外卖实战项目整理,覆盖了从项目搭建到部署上线的完整流程,适合日常开发查阅和面试前背诵。建议结合源码实践加深理解。
2026年04月03日
12 阅读
0 评论
0 点赞
2026-04-03
多线程编程笔记
Java 多线程编程笔记一、多线程概述1.1 什么是多线程Multi‑threading:一个程序内同时运行多个线程,并发执行不同任务。线程:CPU 调度的最小单位。比进程更轻量(创建/切换开销小)。1.2 多线程的优势提高持续响应性(UI 不卡顿)充分利用多核 CPU改善用户体验提高系统吞吐量1.3 多线程面临的问题问题说明线程安全数据竞争、共享资源并发访问、原子性被破坏死锁线程相互等待资源,程序永久阻塞内存一致性缓存、指令重排、可见性问题调试困难并发 Bug 不可复现、时序依赖、执行路径复杂二、Java 多线程模型JVM 线程模型:线程映射(通常 1:1 映射到 OS 线程)线程调度:由操作系统调度器决定(抢占式 + 时间片轮转)Java 内存模型(JMM):定义变量的可见性、有序性、原子性规则三、线程创建方式(4 种)方式实现特点继承 Thread 类class MyThread extends Thread { run() }简单,但受单继承限制实现 Runnableclass MyRunnable implements Runnable { run() }避免单继承,可共享任务实例实现 Callableclass MyCallable implements Callable<T> { call() }有返回值、可抛异常线程池Executors.newFixedThreadPool(n)生产首选,资源复用示例(Lambda 简化):new Thread(() -> System.out.println("run")).start();3.1 Callable + Future 工作流程(“点菜模型”)你点菜 → 提交 Callable 任务厨师做菜 → 线程池异步执行服务员给你叫号器 → 返回 Future 对象你玩手机 → 主线程不阻塞想吃饭了,去取餐 → 调用 future.get()没做好就等(阻塞),做好了直接拿走Future<Integer> future = executor.submit(() -> 100); Integer result = future.get(); // 阻塞直到结果返回四、线程基础操作 & 常用方法4.1 核心方法清单方法作用状态变化start()启动线程NEW → RUNNABLEThread.sleep(millis)休眠RUNNABLE → TIMED_WAITINGjoin()等待该线程结束RUNNABLE → WAITINGjoin(timeout)等待超时RUNNABLE → TIMED_WAITINGyield()让出 CPU(仍在就绪队列)RUNNABLE(不改变状态)Thread.currentThread()获取当前线程-setName() / getName()设置/获取线程名-setDaemon(true)设为守护线程必须在 start() 前调用4.2 线程属性优先级:setPriority(1~10),仅建议,不保证顺序。守护线程:JVM 退出时自动终止(如 GC 线程)。4.3 异常处理(重点)普通 try-catch 抓不住子线程里的异常。使用 UncaughtExceptionHandler:Thread thread = new Thread(() -> { throw new RuntimeException("崩了"); }); thread.setUncaughtExceptionHandler((t, e) -> { System.out.println("线程 " + t.getName() + " 异常:" + e.getMessage()); }); thread.start();五、线程生命周期(完整 6 种状态)NEW → RUNNABLE → BLOCKED / WAITING / TIMED_WAITING → TERMINATED状态触发条件NEWnew Thread(),尚未 start()RUNNABLEstart() 后,等待 CPU 调度BLOCKED进入 synchronized 但锁被占用WAITINGwait()、join() 无超时、LockSupport.park()TIMED_WAITINGsleep(time)、wait(time)、join(time)TERMINATEDrun() 执行完毕或异常退出六、线程调度机制抢占式调度:高优先级可抢占低优先级(但不保证)。时间片轮转:同优先级线程轮流获得时间片。公平调度:JVM 尽量避免线程饥饿。七、线程监控工具工具用途jstack命令行生成线程堆栈快照(定位死锁)jconsole图形化监控(线程、内存、类加载)VisualVM功能最强大的可视化工具八、线程间协作(wait / notify / notifyAll)必须在 synchronized 块内调用,否则 IllegalMonitorStateException。wait():释放锁,进入 WAITING,等待 notify/notifyAll。notify():随机唤醒一个等待线程。notifyAll():唤醒所有等待线程。synchronized (lock) { while (条件不满足) lock.wait(); // 条件满足,执行操作 lock.notifyAll(); }九、同步机制详解(从简单到复杂)9.1 synchronized(内置锁,无脑首选)自动加锁/解锁,不会忘记释放。锁某个对象 → 最小粒度,性能最好。锁当前对象 this → 整个方法串行。锁 Class 对象 → 全局锁。// 同步代码块(推荐) private final Object lock = new Object(); public void deduct() { synchronized(lock) { /* 临界区 */ } } // 同步方法 public synchronized void add() { count++; } // 静态同步方法(类锁) public static synchronized void globalMethod() { }9.2 ReentrantLock(显式锁,更灵活)需要手动 lock() / unlock(),必须放在 finally 中。高级功能:tryLock(timeout):超时放弃,避免死锁。lockInterruptibly():可中断抢锁。new ReentrantLock(true):公平锁(排队不插队)。多 Condition:精准唤醒指定线程。private final Lock lock = new ReentrantLock(); public void update() { lock.lock(); try { // 业务逻辑 } finally { lock.unlock(); } }9.3 volatile(轻量级同步)只保证:可见性 + 禁止指令重排。不保证:原子性(不能用于 count++)。适用场景:布尔状态标记、单例双重检查中的 instance。private volatile boolean stop = false; public void run() { while (!stop) { /* 执行任务 */ } } public void stopTask() { stop = true; }9.4 原子类(无锁,CAS)AtomicInteger、AtomicLong、AtomicReference 等。高并发计数用 LongAdder(分段累加,性能更高)。private AtomicInteger count = new AtomicInteger(0); count.incrementAndGet(); // 原子+19.5 线程安全集合(必记两个)ConcurrentHashMap:分段锁 + CAS,高并发 Map。CopyOnWriteArrayList:读多写少场景,写时复制数组,不阻塞读。Map<String, Integer> map = new ConcurrentHashMap<>(); List<String> list = new CopyOnWriteArrayList<>();9.6 ThreadLocal(线程隔离)每个线程拥有自己的变量副本,不竞争。必须 remove(),否则内存泄漏(尤其在线程池中)。private ThreadLocal<String> userLocal = new ThreadLocal<>(); userLocal.set("张三"); String user = userLocal.get(); userLocal.remove(); // 用完必须清9.7 同步机制选择总结(背诵)场景推荐方案简单逻辑同步(库存、判断)synchronized需要超时、中断、公平锁ReentrantLock线程开关、状态标记volatile计数、累加、ID 生成AtomicInteger / LongAdder多线程操作 Map / ListConcurrentHashMap / CopyOnWriteArrayList单例中存线程私有数据ThreadLocal禁忌:❌ 不要用 volatile 做 i++❌ 不要在单例 Bean 中写普通成员变量(若多线程修改)❌ 不要用 HashMap / ArrayList 做并发写入❌ 不要只锁一半逻辑(如只锁修改,不锁判断)十、死锁10.1 死锁的四个必要条件(缺一不可)互斥:资源只能被一个线程占用。请求与保持:拿着自己的资源,还请求别人的资源。不可剥夺:资源不能被强行抢走。循环等待:线程间形成等待闭环。10.2 避免死锁的方法破坏任意一个条件(如规定锁的顺序、使用 tryLock 超时放弃)。十一、优雅停止线程废弃:stop()、suspend()、resume()(不安全)。推荐方式:中断机制:thread.interrupt() 设置中断标志,线程内检查 Thread.currentThread().isInterrupted() 自行停止。标志位:volatile boolean running,线程定期检查。// 标志位方式 public class MyTask implements Runnable { private volatile boolean running = true; public void stop() { running = false; } public void run() { while (running) { /* 任务 */ } // 清理资源 } }十二、线程间通信(5 种实战方式)方式说明常用场景共享变量(volatile / 原子类)一个线程改,另一个读状态标记、计数器wait()/notify()配合 synchronized生产者‑消费者Condition(配合 ReentrantLock)多路等待,精准唤醒复杂协调Semaphore控制同时访问的线程数限流、连接池CountDownLatch / CyclicBarrier线程协调工具主等子、互相等待12.1 Semaphore 示例Semaphore semaphore = new Semaphore(3); // 最多3个线程同时访问 semaphore.acquire(); // 获取许可(阻塞) semaphore.release(); // 释放许可12.2 CountDownLatch(倒计时门栓)CountDownLatch latch = new CountDownLatch(3); for (int i = 0; i < 3; i++) { new Thread(() -> { // 执行任务 latch.countDown(); }).start(); } latch.await(); // 主线程等待3个任务完成12.3 CyclicBarrier(循环栅栏)CyclicBarrier barrier = new CyclicBarrier(3, () -> System.out.println("三人齐了,出发")); for (int i = 0; i < 3; i++) { new Thread(() -> { barrier.await(); // 互相等待 }).start(); }十三、线程池(企业开发核心)13.1 为什么需要线程池资源复用(避免频繁创建/销毁线程)响应速度快(任务来了直接执行)可管理(控制并发数、监控)13.2 ThreadPoolExecutor 7 大核心参数(必背)new ThreadPoolExecutor( corePoolSize, // 核心线程数(常驻) maximumPoolSize, // 最大线程数(核心+临时) keepAliveTime, // 临时线程空闲存活时间 TimeUnit unit, // 时间单位 BlockingQueue<Runnable> workQueue, // 任务队列 ThreadFactory threadFactory, // 线程工厂(一般用默认) RejectedExecutionHandler handler // 拒绝策略 );13.3 执行流程(5 步,必背)提交任务 → 核心线程未满 → 新建核心线程执行核心线程满 → 任务加入阻塞队列队列满 → 新建临时线程执行总线程数 = 最大线程数 → 执行拒绝策略临时线程空闲超时 → 销毁(回到核心线程数)13.4 4 种拒绝策略(背)策略行为比喻AbortPolicy(默认)抛 RejectedExecutionException直接赶走客人,还吵架CallerRunsPolicy调用者线程自己执行任务让老板自己服务客人DiscardPolicy悄悄丢弃任务,不报错悄悄赶走客人,不吭声DiscardOldestPolicy丢弃队列中最老的任务,然后重提赶走排队最久的,让新人进13.5 Executors 提供的 4 种常用线程池(开发规范禁止使用,需手动创建)工厂方法特点适用场景newFixedThreadPool(n)固定线程数,无界队列重负载服务器newCachedThreadPool()动态线程数,空闲60s回收大量短期异步任务newSingleThreadExecutor()单线程,顺序执行任务需顺序处理newScheduledThreadPool(n)支持定时/延迟执行定时任务规范:不要用 Executors 创建,因为 FixedThreadPool 使用无界队列可能 OOM,CachedThreadPool 可能创建过多线程。必须使用 new ThreadPoolExecutor 自定义。13.6 线程数设置公式(背)CPU 密集型(计算为主):线程数 = CPU 核心数 + 1IO 密集型(网络、磁盘):线程数 = CPU 核心数 × 2 (或更多)十四、定时任务线程池(ScheduledThreadPoolExecutor)14.1 三种定时任务类型方法作用特点schedule(Runnable, delay, unit)延迟执行一次-scheduleAtFixedRate(...)固定频率执行不关心上次任务是否完成,可能叠加scheduleWithFixedDelay(...)固定延迟执行必须等上次完成,再等延迟时间ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(2); scheduler.scheduleAtFixedRate(() -> System.out.println("ping"), 0, 2, TimeUnit.SECONDS);14.2 对比 Timer 的优势多线程,任务互不干扰(Timer 单线程,一个任务异常会影响全部)支持线程池复用十五、线程池优雅关闭(餐厅下班逻辑)挂出「停止接客」牌子 → shutdown()(不再接收新任务)让客人慢慢吃完 → awaitTermination(timeout, unit) 等待现有任务完成等了很久还没吃完 → 直接清场 → shutdownNow()(强制中断所有任务)executor.shutdown(); try { if (!executor.awaitTermination(60, TimeUnit.SECONDS)) { executor.shutdownNow(); } } catch (InterruptedException e) { executor.shutdownNow(); }十六、性能调优与监控16.1 关键指标活跃线程数:< 最大线程数的 80%队列长度:< 队列容量的 70%任务完成数:持续增长,增长停滞说明处理异常拒绝任务数:应为 0,频繁拒绝需优化配置16.2 监控工具JMX(ThreadPoolExecutor 暴露的 getActiveCount() 等)Micrometer(配合 Prometheus + Grafana)jstack 检查死锁十七、快速记忆口诀创建线程四种法:继承、实现、Callable、池。同步锁的选择:简单用 syn,灵活用 Re,开关用 volatile,计数用原子类。线程池七参数:核心最大时单队工拒。关闭优雅三步走:shutdown → 等待 → shutdownNow。死锁四条件:互斥请求不剥夺,循环等待一把锁。
2026年04月03日
8 阅读
0 评论
0 点赞
2026-04-03
Java 集合框架笔记(基于 Java 8)
Java 集合框架笔记(基于 Java 8)一、集合框架概述集合框架是一个统一的架构,用于存储和操作一组对象。根接口是 Iterable,核心接口包括 Collection(单列集合)和 Map(双列集合)。1.1 接口继承关系图Iterable<E> ↓ Collection<E> ├── List<E> │ ├── ArrayList<E> │ ├── LinkedList<E> │ ├── Vector<E> │ └── Stack<E> ├── Set<E> │ ├── HashSet<E> │ ├── LinkedHashSet<E> │ └── SortedSet<E> (接口) │ └── TreeSet<E> └── Queue<E> ├── PriorityQueue<E> └── Deque<E> └── ArrayDeque<E> Map<K,V> (独立体系) ├── HashMap<K,V> ├── LinkedHashMap<K,V> ├── Hashtable<K,V> └── SortedMap<K,V> (接口) └── TreeMap<K,V>1.2 核心接口简介接口描述特点Collection所有单列集合的根接口定义基本操作如添加、删除、大小等List有序集合,允许重复元素支持索引访问,元素顺序与插入顺序一致Set不允许重复元素无索引,通常基于哈希或排序实现Queue队列(先进先出)支持队首队尾操作,也有双端队列 DequeMap键值对映射键唯一,每个键映射到一个值二、Collection 接口所有集合类的根接口,定义了集合的基本操作。2.1 基本方法方法描述boolean add(E e)添加元素boolean remove(Object o)移除元素int size()返回元素个数boolean isEmpty()是否为空void clear()清空所有元素boolean contains(Object o)是否包含某元素Iterator<E> iterator()返回迭代器Object[] toArray()转换为数组三、List 接口有序集合,允许重复元素,支持通过索引操作。3.1 List 特有方法方法描述E get(int index)获取指定位置元素E set(int index, E element)替换指定位置元素void add(int index, E element)在指定位置插入元素E remove(int index)删除指定位置元素int indexOf(Object o)返回首次出现的索引int lastIndexOf(Object o)返回最后一次出现的索引List<E> subList(int from, int to)获取子列表3.2 ArrayList数据结构:动态数组 初始容量:10(默认构造函数) 扩容机制:新容量 = 旧容量 + 旧容量 >> 1(即 1.5 倍) 时间复杂度:get(index) / set(index, element):O(1)add(E e) 平均 O(1),最坏 O(n)(扩容)add(index, E e) / remove(index):O(n) 线程安全:否创建方式// 默认初始容量10 ArrayList<String> list1 = new ArrayList<>(); // 指定初始容量 ArrayList<String> list2 = new ArrayList<>(20); // 从其他集合创建(Java 9+ List.of 返回不可变List) List<String> source = List.of("Java", "Python", "C++"); ArrayList<String> list3 = new ArrayList<>(source); // 使用接口引用(推荐) List<String> list4 = new ArrayList<>();常用操作// 添加 list.add("Java"); // 末尾添加 list.add(1, "C++"); // 指定位置插入 list.addAll(List.of("Go", "Rust")); // 添加多个 // 访问与修改 String first = list.get(0); String last = list.get(list.size() - 1); list.set(1, "Python"); // 查找 int idx = list.indexOf("Java"); boolean has = list.contains("C++"); // 删除 list.remove(0); // 按索引 list.remove("Java"); // 按对象(删除第一个匹配) list.removeAll(List.of("Go", "Rust")); list.removeIf(s -> s.length() > 4); // Java 8+ list.clear(); // 缩容释放内存 list.trimToSize();线程安全解决方案Collections.synchronizedList(new ArrayList<>())CopyOnWriteArrayList(读多写少场景)手动 synchronized 同步注意事项遍历时(增强 for 或迭代器)不能直接调用 list.remove() 或 list.add(),应使用迭代器的 remove() 或 removeIf()。List.of() 返回的集合不可修改。3.3 LinkedList数据结构:双向链表 时间复杂度:get(index) / set(index, element):O(n)addFirst / addLast / removeFirst / removeLast:O(1)中间插入/删除:O(n)(需先定位)特点:实现 List 和 Deque 接口,可作列表、队列、双端队列、栈。创建与添加LinkedList<String> list = new LinkedList<>(); List<String> list2 = new LinkedList<>(); // 接口引用 list.add("Apple"); // 末尾 list.addFirst("Grape"); // 头部 list.addLast("Banana"); // 尾部 list.add(1, "Orange"); // 指定位置 list.offer("Cherry"); // 队列入队(末尾)访问操作String first = list.getFirst(); String last = list.getLast(); String element = list.get(2); // O(n) // 队列/双端队列查看 String head = list.peek(); String headFirst = list.peekFirst(); String headLast = list.peekLast(); int index = list.indexOf("Apple");删除操作list.removeFirst(); list.removeLast(); list.remove(1); list.remove("Apple"); // 队列出队 String polled = list.poll(); list.pollFirst(); list.pollLast();作为队列(FIFO)Queue<String> queue = new LinkedList<>(); queue.offer("客户1"); queue.offer("客户2"); String head = queue.peek(); // 查看队首 String customer = queue.poll(); // 出队作为栈(LIFO)Deque<String> stack = new LinkedList<>(); stack.push("方法A"); stack.push("方法B"); String top = stack.peek(); // 查看栈顶 String method = stack.pop(); // 出栈作为双端队列Deque<Integer> deque = new LinkedList<>(); deque.addFirst(2); deque.addLast(3); deque.addFirst(1); deque.addLast(4); // 结果 [1,2,3,4] deque.peekFirst(); deque.peekLast(); deque.removeFirst(); deque.removeLast();注意事项get(index) 在大列表上频繁使用会导致性能问题(O(n))。没有 trimToSize 方法(链表动态分配)。3.4 Vector 和 StackVector:早期动态数组,方法使用 synchronized 实现线程安全,性能较差,现已不推荐使用。Stack:继承 Vector,提供栈操作 push、pop、peek。同样不推荐,建议使用 Deque 实现(如 ArrayDeque)。四、Set 接口不允许重复元素(最多一个 null),无索引。底层依赖 equals() 和 hashCode() 判断重复。4.1 实现类对比实现类排序性能允许 null线程安全适用场景HashSet无序最高是否一般用途,高性能LinkedHashSet插入顺序中等是否需要保持插入顺序TreeSet自然排序/定制排序较低否否需要排序和范围查询4.2 使用示例Set<String> fruits = new HashSet<>(); fruits.add("苹果"); fruits.add("香蕉"); fruits.add("苹果"); // 重复,不会添加 System.out.println(fruits.size()); // 24.3 最佳实践默认使用 HashSet,需要有序用 LinkedHashSet,需要排序用 TreeSet。线程安全包装:Collections.synchronizedSet(new HashSet<>()) 或 ConcurrentHashMap.newKeySet()。自定义对象存入 Set 时必须重写 equals() 和 hashCode();存入 TreeSet 还需实现 Comparable 或提供 Comparator。合理设置 HashSet 初始容量(避免频繁扩容),负载因子默认 0.75。4.4 注意事项HashSet 迭代顺序不确定,不要依赖。TreeSet 不允许 null,会抛出 NullPointerException。修改已添加到 Set 中的对象可能导致数据丢失或重复。五、Queue 接口FIFO 队列,提供队首队尾操作。Deque 子接口支持双端操作。5.1 主要方法方法描述失败时offer(E e)入队返回 falsepoll()出队并移除返回 nullpeek()查看队首不移除返回 nulladd(E e)入队抛出异常remove()出队并移除抛出异常element()查看队首抛出异常推荐使用:offer、poll、peek(避免异常)。5.2 实现类对比实现类数据结构是否支持 null特点LinkedList双向链表是同时实现 List 和 Deque,插入删除效率高ArrayDeque动态数组否性能优于 LinkedList,一般队列首选PriorityQueue堆(数组)否按优先级排序(自然排序或 Comparator)5.3 使用示例Queue<String> queue = new ArrayDeque<>(); queue.offer("A"); queue.offer("B"); String head = queue.peek(); // A String out = queue.poll(); // A // 优先级队列 Queue<Integer> pq = new PriorityQueue<>(); pq.offer(5); pq.offer(1); pq.offer(3); System.out.println(pq.poll()); // 15.4 线程安全队列ArrayBlockingQueue(有界阻塞队列)LinkedBlockingQueue(可选有界)PriorityBlockingQueue(无界优先级队列)六、Map 接口键值对映射,键唯一,值可重复。不继承 Collection。6.1 常用方法方法描述V put(K key, V value)添加键值对,返回旧值V get(Object key)根据键获取值V remove(Object key)删除键值对boolean containsKey(Object key)是否包含键boolean containsValue(Object value)是否包含值int size()元素个数Set<K> keySet()返回所有键的 Set 视图Collection<V> values()返回所有值的 Collection 视图Set<Map.Entry<K,V>> entrySet()返回所有键值对的 Set 视图6.2 实现类对比实现类排序允许 null线程安全底层结构适用场景HashMap无序键/值均可否哈希表通用,最高性能LinkedHashMap插入顺序/访问顺序键/值均可否哈希表+双向链表需要保持顺序TreeMap自然排序/定制排序键不可为 null否红黑树需要排序和范围查询Hashtable无序键/值均不可是(全表锁)哈希表已淘汰,不推荐6.3 使用示例Map<String, Integer> map = new HashMap<>(); map.put("apple", 10); map.put("banana", 20); int value = map.get("apple"); // 10 map.remove("banana"); // 遍历 for (Map.Entry<String, Integer> entry : map.entrySet()) { System.out.println(entry.getKey() + "=" + entry.getValue()); }6.4 最佳实践声明时使用接口类型:Map<K,V> map = new HashMap<>()。作为键的类必须重写 equals() 和 hashCode(),最好使用不可变对象。合理设置 HashMap 初始容量,避免频繁 rehash。多线程环境使用 ConcurrentHashMap(不要使用 Hashtable)。七、迭代器(Iterator)7.1 Iterator 接口统一遍历集合的方式,支持安全删除。Iterator<String> it = list.iterator(); while (it.hasNext()) { String s = it.next(); if (condition) it.remove(); // 安全删除 }方法:boolean hasNext()E next()void remove()(可选)7.2 ListIterator 接口Iterator 的子接口,专门用于 List,支持双向遍历和修改。额外方法:boolean hasPrevious()E previous()int nextIndex() / int previousIndex()void set(E e) 修改当前元素void add(E e) 在当前位置插入7.3 增强 for 循环(for-each)语法糖,编译器自动转换为 Iterator 遍历。int[] numbers = {1,2,3,4,5}; for (int num : numbers) { System.out.println(num); } List<String> list = ...; for (String s : list) { // 不能修改 list 结构(添加/删除) }限制:无法获取当前索引遍历时不能修改集合结构(否则抛出 ConcurrentModificationException)不能修改基本类型数组的元素值(但可以修改引用类型对象的属性)八、并发集合(java.util.concurrent)8.1 ConcurrentHashMapJava 7 及之前:分段锁(Segment),默认 16 个段。Java 8+:Node 数组 + CAS + synchronized 对链表/红黑树头节点加锁,锁粒度更细。支持并发读(无锁)和并发写(部分加锁)。不允许 null 键/值。8.2 CopyOnWriteArrayList写时复制策略:修改时复制整个数组,修改后替换原数组。读操作无锁,性能极高。适用于读多写极少的场景(如配置列表、监听器列表)。缺点:写操作内存开销大,数据一致性弱(可能读到旧数据)。8.3 BlockingQueue 阻塞队列实现类特点ArrayBlockingQueue有界,基于数组,FIFO,支持公平/非公平锁LinkedBlockingQueue可选有界(默认 Integer.MAX_VALUE),基于链表PriorityBlockingQueue无界,按优先级排序核心阻塞方法:put(E e):队列满时阻塞take():队列空时阻塞典型应用:生产者-消费者模式。九、选择指南9.1 根据数据结构选择需求推荐实现需要索引访问List(ArrayList / LinkedList)不允许重复元素Set(HashSet / LinkedHashSet / TreeSet)键值对存储Map(HashMap / LinkedHashMap / TreeMap)FIFO 队列Queue(ArrayDeque 首选)频繁随机访问ArrayList频繁插入/删除(中间)LinkedList需要排序TreeSet / TreeMap保持插入顺序LinkedHashSet / LinkedHashMap线程安全(单元素操作)Vector / Hashtable(不推荐),推荐 ConcurrentHashMap、CopyOnWriteArrayList 等9.2 时间复杂度速查操作ArrayListLinkedListHashSetTreeSet添加(末尾)O(1)*O(1)O(1)**O(log n)插入(中间)O(n)O(n)--删除(中间)O(n)O(n)O(1)**O(log n)按值查找O(n)O(n)O(1)**O(log n)按索引查找O(1)O(n)--扩容时 O(n) ** 假设哈希函数良好十、常见问题与注意事项ConcurrentModificationException:在遍历集合时直接修改结构(增删),应使用迭代器的 remove() 或 removeIf()。HashSet / HashMap 中自定义对象:必须正确重写 equals() 和 hashCode(),否则无法去重或查找。TreeSet / TreeMap 中元素:必须实现 Comparable 或传入 Comparator,否则抛出 ClassCastException。ArrayList 删除元素后不会自动缩容:可调用 trimToSize() 释放多余内存。LinkedList 的 get(index):效率低(O(n)),避免在循环中频繁调用。Arrays.asList() 返回的 List:是固定大小的视图,不支持 add / remove,会抛出 UnsupportedOperationException。优先使用接口类型声明:List<String> list = new ArrayList<>();,便于更换实现。
2026年04月03日
10 阅读
0 评论
0 点赞
2026-04-03
Java基础笔记
Java基础笔记一、数据类型与变量1. 8种基本数据类型(4大类)类型关键字字节默认值包装类整型byte10Byte short20Short int40Integer long80LLong浮点型float40.0fFloat double80.0dDouble字符型char2'\u0000' (空)Character布尔型boolean1位falseBooleanlong 后缀 L,float 后缀 f,char 字面量用单引号。所有数值类型默认值都是 0(或 0.0)。2. 变量类型局部变量:声明在方法/构造/代码块中,必须显式初始化,作用域仅限于当前块,存储在栈内存。实例变量:声明在类中但方法外,属于对象,有默认值,存储在堆内存,随对象销毁。类变量(静态变量):使用 static 声明,属于类,所有实例共享,类加载时初始化,可通过类名直接访问,存储在方法区(元空间)。3. 类型转换自动类型转换:小范围→大范围,编译器自动完成,不会丢失精度(long→float 可能丢失精度)。强制类型转换:大范围→小范围,需显式使用 (类型),可能丢失精度或溢出。包装类转换:自动装箱/拆箱。例如 Integer b = 10;(装箱),int d = b;(拆箱)。 注意:-128~127 范围内的 Integer 会被缓存,复用同一对象。字符串转换:基本类型→字符串用 String.valueOf(),字符串→基本类型用 parseXxx()。二、运算符与表达式优先级:* / % > + - > == != > && > ||。建议使用括号明确顺序。复合赋值:+=、-= 等可简化代码。浮点数比较:使用误差范围(如 Math.abs(a - b) < 1e-6),避免直接 ==。字符串比较:== 比较引用地址。equals() 比较内容。compareTo() 返回大小关系(正/零/负)。自增/自减:避免在复杂表达式中使用,如 i++ + ++i。三、输入输出(Scanner)Scanner sc = new Scanner(System.in); int n = sc.nextInt(); // 读取整数 double d = sc.nextDouble(); // 读取浮点数 String s1 = sc.next(); // 读到空格或换行 String s2 = sc.nextLine(); // 读取整行 boolean b = sc.nextBoolean(); // 检查方法 if (sc.hasNextInt()) { ... }缓存问题:nextInt() 后调用 nextLine() 需先执行一次 nextLine() 吸收换行符。资源关闭:sc.close(),建议使用 try-catch 处理异常。格式化输出:System.out.printf("%.2f", d);四、数组1. 一维数组声明:int[] arr; (不能指定长度)创建:arr = new int[5]; → 元素自动初始化为默认值 [0,0,0,0,0]静态初始化:int[] arr = {1,2,3}; 或 new int[]{1,2,3}动态初始化:先声明后创建,再逐个赋值。访问:索引从0开始,arr[0],注意 ArrayIndexOutOfBoundsException。长度:arr.length(不是方法)。2. 多维数组(以二维为例)int[][] matrix = new int[3][4]; // 规则3行4列 int[][] ragged = new int[3][]; // 不规则 ragged[0] = new int[2]; ragged[1] = new int[5];获取行数:matrix.length,获取当前行列数:matrix[i].length。遍历时注意每一行长度可能不同。3. 数组操作复制:System.arraycopy()(原生高效)、Arrays.copyOf()、clone()。排序:Arrays.sort(arr)(Dual-Pivot Quicksort)、手写冒泡/选择/快排。搜索:线性搜索(未排序)。二分搜索(已排序):Arrays.binarySearch(arr, key),返回下标或负数插入点。五、类与对象1. 类的组成实例变量/方法:属于对象,通过对象访问。静态变量/方法:属于类,通过类名访问,不能直接访问实例成员。构造函数:与类同名,无返回类型,可重载。一旦定义了有参构造,默认无参构造不再自动提供。2. 封装使用 private 隐藏字段,提供 public 的 getter/setter。保护内部实现,提供清晰的公共接口。3. 方法定义:[修饰符] [static] 返回值 方法名(参数) { ... }调用:静态方法:类名.方法名()实例方法:对象.方法名()参数传递:基本类型:传值副本,不影响原变量。引用类型:传引用副本,可修改对象内容,但不能改变原引用指向。4. 构造方法链式调用使用 this(参数) 调用同一类的另一个构造方法,必须是构造方法的第一条语句。public class Person { private String name; public Person() { this("Unknown"); } public Person(String name) { this.name = name; } }5. this 关键字引用当前对象,用于区分成员变量和参数。不能在静态上下文中使用。this() 必须是构造方法第一条语句。6. 访问修饰符修饰符同类同包子类(不同包)任意private✔✘✘✘default✔✔✘✘protected✔✔✔✘public✔✔✔✔7. 包(package)文件第一行非注释代码:package com.example.util;import 语句位于 package 之后,类定义之前。静态导入:import static java.lang.Math.PI; 可直接使用 PI。六、继承使用 extends 关键字,Java 只支持单继承。所有类隐式继承 Object。方法重写(Override):子类重新定义父类方法,方法签名必须相同。使用 @Override 注解。访问修饰符不能比父类更严格(父类 protected → 子类 public 可以,反之不行)。返回类型可以是子类类型(协变返回)。不能抛出比父类更宽泛的检查异常。final 方法不能被重写;static 方法不能被重写(但可以隐藏);private 方法不可见,不能重写。重载(Overload):同一类中方法名相同,参数列表不同。super 关键字:调用父类构造方法:super() 必须是子类构造的第一条语句。访问父类成员:super.method()、super.field。不能与 this() 同时使用。final 关键字final 类:不能被继承(如 String、Integer)。final 方法:不能被重写。final 变量:静态 final:声明时或静态代码块初始化。实例 final:声明时或构造方法中初始化。局部 final:使用前赋值即可。参数 final:方法内不能修改参数值。七、抽象类与接口特性抽象类(abstract class)接口(interface)关键字abstract classinterface继承/实现单继承多实现构造方法有无成员变量可以是实例变量或常量只能是 public static final 常量方法抽象方法 + 具体方法抽象方法 + default/static 方法(JDK8+)访问修饰符任意方法默认 public abstract使用场景代码复用、模板方法模式定义规范、多继承能力抽象类不能实例化,子类必须实现所有抽象方法(除非子类也是抽象类)。接口中的 default 方法提供默认实现,static 方法属于接口本身。八、多态实现条件:继承 + 重写 + 父类引用指向子类对象。编译时多态:方法重载。运行时多态:通过父类引用调用被子类重写的方法,实际执行子类版本。向上转型:Parent p = new Child(); 安全,自动进行。向下转型:Child c = (Child) p; 可能抛出 ClassCastException,推荐先用 instanceof 检查。instanceof:object instanceof ClassName 返回 true/false。用于类型检查,避免转换异常。null instanceof X 总是 false。不能用于基本类型。注意:静态方法、私有方法、构造方法不支持多态。九、内部类与嵌套类类型关键字能否访问外部类实例成员持有外部类引用能否定义静态成员创建方式成员内部类无✔✔不能(常量除外)外部类实例.new 内部类()静态嵌套类static✘(只能访问外部静态)✘✔new 外部类.静态嵌套类()局部内部类方法内✔✔不能在作用域内直接 new匿名类无名字✔✔不能new 接口/父类() { ... }内存泄漏风险:成员内部类隐式持有外部类引用,可能导致外部类无法被GC。局部内部类访问局部变量:变量必须是 final 或事实不可变(effectively final)。十、枚举(Enum)enum Color { RED, GREEN, BLUE }继承 java.lang.Enum,不能手动继承其他类。常用方法:name():返回常量名(如 "RED")ordinal():返回索引(从0开始)toString():默认返回 name()valueOf(String):根据名称获取枚举常量values():返回所有枚举常量的数组compareTo():比较顺序(基于 ordinal)避免在业务逻辑中使用 ordinal(),因为顺序改变会出错。可添加字段和构造方法来实现语义属性。十一、注解(Annotation)内置注解:@Override、@Deprecated、@SuppressWarnings、@SafeVarargs、@FunctionalInterface。元注解(用于自定义注解):@Retention:指定生命周期(SOURCE、CLASS、RUNTIME)@Target:指定可放置位置(TYPE、FIELD、METHOD、PARAMETER、CONSTRUCTOR 等)@Documented:生成Javadoc时包含该注解@Inherited:允许子类继承父类的注解@Retention(RetentionPolicy.RUNTIME) @Target(ElementType.METHOD) public @interface MyAnnotation { String value() default ""; }十二、泛型参数化类型:List<String> 保证类型安全,避免强制转换。泛型类/接口:class Box<T> { private T t; ... }泛型方法:public <T> T getValue(T t) { return t; }通配符:?? extends T:上界通配符,用于读取(生产者)。? super T:下界通配符,用于写入(消费者)。PECS原则:Producer Extends, Consumer Super。有界类型参数:<T extends Number> 限定 T 必须是 Number 或其子类。注意:泛型在编译时擦除,运行时无法获知具体类型参数。十三、反射(Reflection)获取 Class 对象的三种方式:类名.class(编译时确定)对象.getClass()(运行时)Class.forName("全限定类名")(动态加载)常用操作:创建实例:clazz.newInstance() 或构造器 newInstance()获取方法:getMethod()、getDeclaredMethod()调用方法:method.invoke(obj, args)访问字段:getField()、setAccessible(true)反射会破坏封装,带来性能开销,慎用。十四、递归定义:方法调用自身。线性递归:每次调用一个子问题,深度 = 规模,时间复杂度 O(n)。二分递归:每次调用两个子问题,如斐波那契 fib(n-1) + fib(n-2),存在重复计算,可优化(记忆化)。注意防止栈溢出,确保有终止条件。十五、String 字符串不可变:任何修改都会产生新字符串对象。字符串池:字面量 "abc" 会放入常量池,复用相同内容;new String("abc") 创建堆中新对象,不自动入池,可调用 intern() 手动入池。常用方法:length()、isEmpty()、isBlank()(Java 11+)indexOf()、lastIndexOf()equals()、equalsIgnoreCase()、compareTo()contains()、startsWith()、endsWith()substring()replace()、replaceAll()trim()、strip()(Java 11+)toUpperCase()、toLowerCase()split()、join()高效拼接:使用 StringBuilder(线程不安全)或 StringBuffer(线程安全)。十六、代码最佳实践提示使用括号明确运算优先级。浮点数比较使用误差范围。字符串内容比较用 equals() 而非 ==。关闭 Scanner 资源,处理异常。数组遍历注意边界,多用 length 属性。重写方法时加 @Override。优先使用多态和接口而非 instanceof 链。枚举代替常量整数。泛型提供编译时类型安全。合理使用内部类,注意内存泄漏。反射仅用于框架或特殊场景。
2026年04月03日
8 阅读
0 评论
1 点赞
2026-03-30
MySQL SQL性能调优实战指南
# MySQL SQL性能调优实战指南一、核心思想:系统化调优四步法SQL调优不是盲目加索引,而是一个系统化工程,应遵循 测量 -> 分析 -> 优化 -> 验证 的闭环流程。测量:通过慢查询日志、监控系统定位需要优化的慢SQL。分析:使用EXPLAIN等工具分析SQL的执行计划,找到性能瓶颈。优化:根据分析结果,采取索引优化、SQL重写、架构调整等手段。验证:在测试环境验证优化效果,并上线监控。二、性能诊断与分析工具1. 慢查询日志作用:记录执行时间超过阈值的SQL语句。配置:-- 开启慢查询日志 SET GLOBAL slow_query_log = ON; -- 设置阈值(如0.1秒) SET GLOBAL long_query_time = 0.1; -- 查看日志文件路径 SHOW VARIABLES LIKE 'slow_query_log_file';2. EXPLAIN 执行计划分析作用:查看MySQL如何执行你的SQL,是调优的核心工具。关键字段解读:type:访问类型,性能从好到差:system > const > eq_ref > ref > range > index > ALL。ALL:全表扫描,性能最差,必须优化。range:范围扫描,使用了索引,常见于BETWEEN、>、LIKE 'abc%'。ref:非唯一性索引查找。possible_keys:可能用到的索引。key:实际使用的索引。rows:MySQL估计需要扫描的行数,这个值越小越好。Extra:额外信息,关键提示:Using index:覆盖索引,查询的列都在索引中,无需回表,性能高。Using where:需要回表过滤数据。Using filesort:文件排序,需要优化(通常需要为ORDER BY字段建立索引)。Using temporary:使用临时表,常见于GROUP BY或DISTINCT,性能差。使用方式:EXPLAIN SELECT * FROM users WHERE age > 25; EXPLAIN FORMAT=JSON SELECT * FROM users WHERE age > 25; -- 更详细的结构化信息 -- MySQL 8.0+ 可以查看实际执行计划 EXPLAIN ANALYZE SELECT * FROM users WHERE age > 25;三、索引优化策略(核心)索引是提升查询性能的关键,但也是一把双刃剑(占用空间、影响写入性能)。1. 索引设计原则高选择性字段优先:字段的唯一性越高,索引效果越好。-- 计算选择性:值越接近1,选择性越高 SELECT COUNT(DISTINCT user_id) / COUNT(*) FROM orders;最左前缀原则:复合索引 (a, b, c) 会创建 (a), (a,b), (a,b,c) 三个索引。查询条件必须包含索引的最左列,否则索引失效。索引列顺序:等值查询放前面,范围查询放后面。避免冗余索引:尽量复用现有索引,删除无用索引。-- 查看表上的索引 SHOW INDEX FROM table_name;2. 索引优化实战覆盖索引:查询的列都在索引中,避免回表。-- 假设有复合索引 idx_status_covering(status, user_id, order_no) SELECT user_id, order_no FROM orders WHERE status = 'pending'; -- 走覆盖索引范围查询优化:-- 原始(慢):索引可能只用到create_time CREATE INDEX idx_time ON orders(create_time); SELECT * FROM orders WHERE create_time BETWEEN '2024-01-01' AND '2024-01-31' AND status = 'completed'; -- 优化后:将等值条件放在前面 CREATE INDEX idx_status_time ON orders(status, create_time);3. 索引失效的常见场景(避坑指南)函数操作:对索引列使用函数或计算。-- ❌ 失效 SELECT * FROM users WHERE DATE(create_time) = '2024-01-01'; -- ✅ 优化 SELECT * FROM users WHERE create_time >= '2024-01-01' AND create_time < '2024-01-02';隐式类型转换:数据类型不匹配。-- ❌ 失效(phone是varchar类型) SELECT * FROM users WHERE phone = 13800138000; -- ✅ 优化 SELECT * FROM users WHERE phone = '13800138000';前导通配符:LIKE以%开头。-- ❌ 失效 SELECT * FROM users WHERE name LIKE '%john%'; -- ✅ 可能有效(后缀通配符) SELECT * FROM users WHERE name LIKE 'john%';OR条件:如果OR连接的条件中有一个没有索引,整个查询都可能不走索引。NOT IN 或 !=:可能导致索引失效(或效果变差)。四、SQL语句优化技巧1. 查询重写用JOIN代替子查询:子查询会创建临时表,JOIN通常性能更好。-- ❌ 低效的子查询 SELECT * FROM orders WHERE user_id IN (SELECT id FROM users WHERE vip_level > 3); -- ✅ 高效JOIN SELECT o.* FROM orders o JOIN users u ON o.user_id = u.id WHERE u.vip_level > 3;用EXISTS代替IN:当外表数据量大时,EXISTS可能更优。避免SELECT *:只查询需要的字段,特别是对于TEXT/BLOB等大字段。HAVING改为WHERE:HAVING用于分组后过滤,尽量将过滤条件放在WHERE中提前过滤。2. 分页查询优化问题:LIMIT 1000000, 20 会扫描100万条记录,性能极差。方案一:游标分页:基于有序且唯一的列(如自增ID)进行翻页。-- 每次记录上一页最后一条的ID SELECT * FROM orders WHERE id > 1000000 ORDER BY id LIMIT 20;方案二:延迟关联:先通过覆盖索引查询出主键,再回表获取其他字段。SELECT * FROM orders o JOIN (SELECT id FROM orders ORDER BY id LIMIT 1000000, 20) t ON o.id = t.id;3. 大数据量处理分批处理:避免一次性操作大量数据,应分批次循环处理。-- 每次处理1000条 UPDATE orders SET status = 'processed' WHERE status = 'pending' LIMIT 1000;分区表:对于历史数据,可以使用分区表进行物理隔离,按时间或键值分区。CREATE TABLE logs ( id BIGINT, log_date DATE ) PARTITION BY RANGE (YEAR(log_date)) ( PARTITION p2023 VALUES LESS THAN (2024), PARTITION p2024 VALUES LESS THAN (2025) );五、数据库架构与配置优化1. 架构优化读写分离:主库负责写入,从库负责读取,分散压力。分库分表:当单表数据量过大(如超过千万级)时,垂直分库(按业务拆分)或水平分表(按ID取模)。引入缓存:热点数据(如用户信息、商品详情)使用Redis等缓存,减少数据库压力。2. 配置参数调优(关键)在my.cnf或my.ini中调整:InnoDB缓冲池(最重要):设置为物理内存的50%-80%。innodb_buffer_pool_size = 16G日志缓冲:innodb_log_buffer_size = 64M连接数:max_connections = 1000排序/临时表:sort_buffer_size = 2M tmp_table_size = 64M六、实战调优案例:电商订单查询场景查询用户近3个月内状态为pending或shipped的订单,按时间倒序取前20条。SELECT * FROM orders WHERE user_id = 100 AND status IN ('pending', 'shipped') AND create_time BETWEEN '2024-01-01' AND '2024-03-01' ORDER BY create_time DESC LIMIT 20;问题分析原表可能没有索引,导致全表扫描。ORDER BY导致Using filesort。优化步骤分析:EXPLAIN 显示 type=ALL,Extra=Using where; Using filesort。建立复合索引:遵循最左前缀原则,将等值查询的字段user_id和status放在前面,范围查询字段create_time放在后面,且确保排序的字段也在索引中。CREATE INDEX idx_user_status_time ON orders(user_id, status, create_time);改写查询:利用覆盖索引,只查需要的字段,减少回表。SELECT id, user_id, order_no, amount, create_time FROM orders WHERE user_id = 100 AND status IN ('pending', 'shipped') AND create_time BETWEEN '2024-01-01' AND '2024-03-01' ORDER BY create_time DESC LIMIT 20;验证:再次EXPLAIN,type变为range或ref,rows大幅减少,Extra中不再有Using filesort,且出现Using index(覆盖索引)。七、调优检查清单[ ] 慢查询日志:是否已开启并定期分析?[ ] 索引设计:高频查询条件是否有索引?复合索引顺序是否合理?[ ] 索引失效:是否对索引列使用了函数、计算、隐式转换?[ ] SQL写法:是否避免了SELECT *?分页是否使用了大偏移量?[ ] 表结构:字段类型是否选择合适(如能用int就不用varchar)?[ ] 架构层面:读写分离、缓存、分库分表是否按需引入?[ ] 配置参数:缓冲池大小是否合理?连接数是否充足?总结MySQL SQL调优的核心是通过EXPLAIN理解SQL的执行路径,并利用索引减少需要扫描的数据行数。它是一个持续优化的过程,需要结合业务场景、数据规模和系统架构来综合考量。记住口诀:测量先行,索引为王,SQL避坑,架构护航。
2026年03月30日
6 阅读
0 评论
0 点赞
1
...
4
5
6
...
8