当前位置: 首页 > article >正文

Spring Boot统一功能处理深度解析

第一章:为什么需要统一功能处理?

想象你正在开发一个电商系统,包含用户管理、商品管理、订单管理等模块。每个模块都需要:

  1. 用户身份验证
  2. 操作日志记录
  3. 异常统一处理
  4. 数据格式标准化

如果每个模块都单独实现这些功能:

// 用户模块
@PostMapping("/users")
public User createUser(@RequestBody User user) {try {// 1. 验证token// 2. 记录日志// 3. 业务处理// 4. 统一返回格式} catch (Exception e) {// 5. 异常处理}
}// 商品模块
@PostMapping("/products")
public Product createProduct(@RequestBody Product product) {try {// 重复步骤1-5...} catch (Exception e) {// ...}
}

问题暴露

  • 代码重复率高(DRY原则被破坏)
  • 维护困难(修改需多处调整)
  • 系统一致性差(不同开发者实现方式不同)

第二章:Spring Boot统一处理四大核心组件

2.1 拦截器(Interceptor) - 请求的"安检门"

作用:在请求到达Controller前/后执行通用逻辑

public class AuthInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {// 1. 从请求头获取Token(就像门卫检查你的门禁卡)String token = request.getHeader("Authorization");// 2. 验证Token是否有效(门卫刷卡机验证)if (!validateToken(token)) {// 3. 无效则拒绝进入(设置401状态码)response.setStatus(401);return false; // 中断请求}return true; // 允许进入}// 验证Token的方法(门卫的验证设备)private boolean validateToken(String token) {// 实际验证逻辑(这里简化为非空检查)return token != null && !token.isEmpty();}
}

通俗解释

  • 这个类就像小区的门卫,检查每个进入的人(请求)
  • preHandle方法:在业主回家前检查门禁卡(Token)
  • 如果门禁卡无效(validateToken返回false),门卫会拒绝进入(返回401状态码)
  • 如果有效,就放行(return true)

配置方式

@Configuration
public class WebConfig implements WebMvcConfigurer {@Overridepublic void addInterceptors(InterceptorRegistry registry) {// 给门卫分配工作区域registry.addInterceptor(new AuthInterceptor()).addPathPatterns("/api/**")   // 保护所有/api路径.excludePathPatterns("/api/login"); // 但登录入口不需要检查}
}

通俗解释

  • 这相当于给门卫划定工作范围:
    • /api/**:需要检查的所有区域
    • /api/login:特殊通道(登录入口),不需要检查
2.2 过滤器(Filter) - 请求的"净化器"

作用:处理HTTP请求/响应的原始数据

public class LoggingFilter implements Filter {@Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) {// 1. 记录请求开始时间(水电表开始计数)long startTime = System.currentTimeMillis();// 2. 继续处理请求(让水流/电流继续)chain.doFilter(request, response);// 3. 计算耗时(计算水电用量)long duration = System.currentTimeMillis() - startTime;// 4. 记录日志(生成水电账单)System.out.println("请求耗时: " + duration + "ms");}
}
  • 这个过滤器就像水电表:
    1. 请求开始时记录时间(开水龙头)
    2. chain.doFilter让请求继续处理(水流)
    3. 请求结束后计算耗时(关水龙头,计算用水量)
    4. 记录日志(生成水电账单)

与拦截器区别

特性过滤器(Filter)拦截器(Interceptor)
作用范围Servlet规范Spring MVC特有
依赖不依赖Spring容器依赖Spring容器
获取信息原始HTTP请求/响应HandlerMethod信息
执行顺序最先执行在DispatcherServlet后执行
2.3 切面(AOP) - 功能的"手术刀"

作用:在方法执行前后插入通用逻辑

@Aspect
@Component
public class ServiceMonitorAspect {// 监控Service层方法@Around("execution(* com.example..service.*.*(..))")public Object monitorService(ProceedingJoinPoint pjp) throws Throwable {long start = System.currentTimeMillis();try {return pjp.proceed(); // 执行目标方法} finally {long duration = System.currentTimeMillis() - start;// 记录慢查询if (duration > 500) {log.warn("慢方法: {} 耗时: {}ms", pjp.getSignature(), duration);}}}
}
2.4 控制器增强(@ControllerAdvice) - 全局的"管理员"

作用:统一处理控制器异常和返回格式

@ControllerAdvice
public class GlobalExceptionHandler {// 处理业务异常@ExceptionHandler(BusinessException.class)@ResponseBodypublic ResponseResult<Void> handleBusinessEx(BusinessException e) {return ResponseResult.error(e.getCode(), e.getMessage());}// 处理所有未捕获异常@ExceptionHandler(Exception.class)@ResponseBodypublic ResponseResult<Void> handleException(Exception e) {log.error("系统异常", e);return ResponseResult.error(500, "系统繁忙");}
}

第三章:统一功能处理实战案例

3.1 统一响应格式
// 标准响应体
public class ResponseResult<T> {private int code;private String msg;private T data;// 成功响应public static <T> ResponseResult<T> success(T data) {return new ResponseResult<>(200, "成功", data);}// 错误响应public static <T> ResponseResult<T> error(int code, String msg) {return new ResponseResult<>(code, msg, null);}
}// 控制器统一返回
@RestController
public class UserController {@GetMapping("/users/{id}")public ResponseResult<User> getUser(@PathVariable Long id) {User user = userService.findById(id);return ResponseResult.success(user);}
}
3.2 统一异常处理
@ControllerAdvice // 1. 声明这是全局异常处理器
public class GlobalExceptionHandler {// 2. 处理业务异常(如订单处理失败)@ExceptionHandler(BusinessException.class)@ResponseBodypublic ResponseResult<Void> handleBusinessEx(BusinessException e) {// 返回标准错误格式return ResponseResult.error(e.getCode(), e.getMessage());}// 3. 处理所有未预料异常(系统崩溃)@ExceptionHandler(Exception.class)@ResponseBodypublic ResponseResult<Void> handleException(Exception e) {// 记录错误日志(就像应急系统记录事故)e.printStackTrace();// 返回友好提示return ResponseResult.error(500, "系统繁忙,请稍后再试");}
}
  • @ControllerAdvice:这是整个小区的应急中心
  • @ExceptionHandler:不同类型的应急处理小组
    • BusinessException:处理业务问题(如订单错误)
    • Exception:处理所有未预料的问题(系统崩溃)
  • 当发生问题时,系统会自动调用对应的方法处理
3.3 统一日志记录
@Aspect
@Component
public class ControllerLogAspect {@Around("@within(org.springframework.web.bind.annotation.RestController)")public Object logController(ProceedingJoinPoint pjp) throws Throwable {// 获取请求信息HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.currentRequestAttributes()).getRequest();// 记录请求日志log.info("请求开始 => URI: {}, 参数: {}", request.getRequestURI(), Arrays.toString(pjp.getArgs()));long start = System.currentTimeMillis();try {Object result = pjp.proceed();// 记录响应日志log.info("请求完成 <= URI: {}, 耗时: {}ms", request.getRequestURI(), System.currentTimeMillis() - start);return result;} catch (Exception e) {// 记录异常日志log.error("请求异常 <= URI: {}, 错误: {}", request.getRequestURI(), e.getMessage());throw e;}}
}
3.4 统一身份认证
public class JwtAuthInterceptor implements HandlerInterceptor {@Autowiredprivate JwtTokenService tokenService;@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {// 1. 获取TokenString token = request.getHeader("Authorization");if (StringUtils.isEmpty(token)) {sendError(response, 401, "未提供认证信息");return false;}// 2. 验证TokenClaims claims = tokenService.parseToken(token);if (claims == null) {sendError(response, 401, "无效的Token");return false;}// 3. 存储用户信息UserContext.setCurrentUser(claims.getSubject());return true;}private void sendError(HttpServletResponse response, int status, String msg) {response.setStatus(status);response.setContentType("application/json");try {response.getWriter().write(new ResponseResult<>(status, msg).toString());} catch (IOException e) {log.error("响应写入失败", e);}}
}

第四章:高级应用场景

4.1 接口限流(Rate Limiting)
@Aspect
@Component
public class RateLimitAspect {private final Map<String, RateLimiter> limiters = new ConcurrentHashMap<>();@Around("@annotation(rateLimit)")public Object rateLimit(ProceedingJoinPoint pjp, RateLimit rateLimit) throws Throwable {String key = getRateLimitKey(pjp);RateLimiter limiter = limiters.computeIfAbsent(key, k -> RateLimiter.create(rateLimit.value()));if (limiter.tryAcquire()) {return pjp.proceed();} else {throw new BusinessException(429, "请求过于频繁");}}private String getRateLimitKey(ProceedingJoinPoint pjp) {// 根据方法+IP生成唯一keyMethodSignature signature = (MethodSignature) pjp.getSignature();HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();return signature.getMethod().getName() + ":" + request.getRemoteAddr();}
}// 使用注解
@GetMapping("/api/data")
@RateLimit(value = 10) // 每秒10次
public ResponseResult<Data> getData() {// ...
}
4.2 多租户数据隔离
public class TenantInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {// 从请求头获取租户IDString tenantId = request.getHeader("X-Tenant-ID");if (StringUtils.isNotBlank(tenantId)) {TenantContext.setCurrentTenant(tenantId);}return true;}
}// 在MyBatis拦截器中应用
@Intercepts({@Signature(type = Executor.class, method = "query", ...)})
public class TenantInterceptor implements Interceptor {@Overridepublic Object intercept(Invocation invocation) throws Throwable {// 自动添加租户过滤条件String tenantId = TenantContext.getCurrentTenant();if (tenantId != null) {MappedStatement ms = (MappedStatement) invocation.getArgs()[0];Object parameter = invocation.getArgs()[1];// 修改SQL添加 tenant_id = #{tenantId}}return invocation.proceed();}
}
4.3 API版本控制
public class ApiVersionInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {// 从URL路径获取版本号String path = request.getRequestURI();Matcher matcher = Pattern.compile("/v(\\d+)/").matcher(path);if (matcher.find()) {int version = Integer.parseInt(matcher.group(1));ApiVersionContext.setVersion(version);}return true;}
}// 控制器根据版本路由
@RestController
@RequestMapping("/api/users")
public class UserController {@GetMapping@ApiVersion(1) // v1版本public ResponseResult<List<User>> getUsersV1() {// 旧版逻辑}@GetMapping@ApiVersion(2) // v2版本public ResponseResult<List<UserDto>> getUsersV2() {// 新版逻辑}
}

第五章:性能优化与最佳实践

5.1 组件执行顺序优化
请求进入

Filter-Interceptor-preHandle-Controller-Interceptor-postHandle-Filter-after-Response

优化策略

  1. 过滤器优先处理原始数据(如请求体缓存)
  2. 拦截器处理业务相关逻辑(如认证、日志)
  3. 切面处理具体方法逻辑(如性能监控)
  4. @ControllerAdvice最后处理异常和响应
5.2 避免的陷阱
  1. 循环依赖问题

  2. @Component
    public class AuthInterceptor implements HandlerInterceptor {@Autowiredprivate AuthService authService; // 可能导致循环依赖
    }
    

    解决方案:使用@Lazy延迟注入

  3. @Lazy
    @Autowired
    private AuthService authService;
    

    线程安全问题

  4. public class UserContext {private static final ThreadLocal<User> currentUser = new ThreadLocal<>();
    }
    

    性能热点

  5. // 避免在拦截器/过滤器中执行耗时操作
    public boolean preHandle(...) {// 错误:执行数据库查询User user = userRepository.findById(userId);
    }
    
    5.3 最佳实践总结
  6. 职责分离原则

    • 过滤器:处理原始HTTP请求/响应
    • 拦截器:处理业务相关前置/后置逻辑
    • 切面:处理具体方法逻辑
    • 控制器增强:统一异常和响应处理
  7. 配置顺序原则

  8. @Override
    public void addInterceptors(InterceptorRegistry registry) {// 1. 认证拦截器(最先执行)registry.addInterceptor(authInterceptor);// 2. 日志拦截器registry.addInterceptor(logInterceptor);// 3. 其他业务拦截器
    }
    

    性能监控指标

  9. // 使用Micrometer监控
    @Autowired
    private MeterRegistry meterRegistry;public void afterCompletion(...) {meterRegistry.counter("api.requests", "uri", request.getRequestURI(),"status", String.valueOf(response.getStatus())).increment();
    }
    

    第六章:综合实战 - 电商系统统一处理

  6.1 系统架构图

 6.2 核心配置代码

// 网关统一配置
@Configuration
public class GatewayConfig {@Beanpublic GlobalFilter customFilter() {return (exchange, chain) -> {// 1. 统一添加请求IDServerHttpRequest request = exchange.getRequest().mutate().header("X-Request-ID", UUID.randomUUID().toString()).build();return chain.filter(exchange.mutate().request(request).build());};}
}// 微服务统一配置
@Configuration
public class ServiceConfig implements WebMvcConfigurer {@Overridepublic void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new TenantInterceptor());registry.addInterceptor(new AuthInterceptor());}@Beanpublic FilterRegistrationBean<LoggingFilter> loggingFilter() {FilterRegistrationBean<LoggingFilter> bean = new FilterRegistrationBean<>();bean.setFilter(new LoggingFilter());bean.setOrder(Ordered.HIGHEST_PRECEDENCE);return bean;}
}
6.3 业务服务示例
@RestController
@RequestMapping("/orders")
public class OrderController {@PostMappingpublic ResponseResult<Order> createOrder(@Valid @RequestBody OrderCreateDTO dto) {// 无需处理认证、日志、异常等Order order = orderService.createOrder(dto);return

相关文章:

Spring Boot统一功能处理深度解析

第一章&#xff1a;为什么需要统一功能处理&#xff1f; 想象你正在开发一个电商系统&#xff0c;包含用户管理、商品管理、订单管理等模块。每个模块都需要&#xff1a; 用户身份验证操作日志记录异常统一处理数据格式标准化 如果每个模块都单独实现这些功能&#xff1a; …...

世事无常,比较复杂,人可以简单一点

2025年6月5日日&#xff0c;17~28℃&#xff0c;一般 待办&#xff1a; 宣讲会 职称材料的最后检查 职称材料有错误&#xff0c;需要修改 期末考试试题启用 教学技能大赛PPT 遇见&#xff1a;部门宣传泰国博士项目、硕士项目、本科项目。 感受或反思&#xff1a;东南亚博士…...

使用 Docker Compose 安装 PostgreSQL 16

前面是指南&#xff0c;后面是实际工作日志。 1. 创建 docker-compose.yml 文件 yaml 复制 下载 version: 3.9 services:postgres:image: postgres:16container_name: postgres-16environment:POSTGRES_USER: your_username # 替换为你的用户名POSTGRES_PASSWORD: your…...

每日算法刷题Day23 6.5:leetcode二分答案3道题,用时1h40min(有点慢)

8. 3007.价值和小于等于K的最大数字(中等&#xff0c;学习,太难&#xff0c;先过) 3007. 价值和小于等于 K 的最大数字 - 力扣&#xff08;LeetCode&#xff09; 思想 1.给你一个整数 k 和一个整数 x 。整数 num 的价值是它的二进制表示中在 x&#xff0c;2x&#xff0c;3x …...

【Android基础回顾】七:内存管理机制

Android 的内存管理机制是一个多层次的复杂系统&#xff0c;旨在高效利用有限的物理内存&#xff08;RAM&#xff09;&#xff0c;在保证前台应用流畅运行的同时&#xff0c;尽可能在后台保留更多应用以提高启动速度&#xff08;多任务&#xff09;。 它的核心机制结合了 Linu…...

数据结构哈希表总结

349. 两个数组的交集 力扣题目链接(opens new window) 题意&#xff1a;给定两个数组&#xff0c;编写一个函数来计算它们的交集。 说明&#xff1a; 输出结果中的每个元素一定是唯一的。 我们可以不考虑输出结果的顺序。 public int[] intersection(int[] nums1, int[] num…...

Spring事务失效-----十大常见场景及解决方案全解析

Spring事务失效的常见场景及原因分析 Spring事务管理是开发中的核心功能,但在实际应用中可能因各种原因导致事务失效。以下是常见的事务失效场景及详细解析: 1. 方法未被Spring管理 场景:使用new关键字直接创建对象,而非通过Spring容器注入原因:Spring事务基于AOP代理,…...

KMP 算法中 next 数组的构建函数 get_next

KMP 算法中 next 数组的构建函数 get_next &#xff0c;负责计算模式串的 next 数组&#xff0c;核心是通过递推找到每个位置的 “最长相等前缀后缀长度”。&#xff08;下标从 1 开始&#xff09;&#xff1a; 一、函数作用 get_next(SString T, int next[]) 的任务&#xf…...

IDEA 开发PHP配置调试插件XDebug

1、安装PHP环境 为了方便&#xff0c;使用的PhpStudy。 安装路径&#xff1a;D:\resources\phpstudy_pro\Extensions\php\php7.3.4nts 2、下载Xdebug Xdebug: Downloads 选择对应的版本下载&#xff0c;本次使用的是7.3。 3、配置Xdebug 在php.ini中添加Xdebug配置。 D…...

奇异值分解(SVD):线性代数在AI大模型中的核心工具

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家、CSDN平台优质创作者&#xff0c;高级开发工程师&#xff0c;数学专业&#xff0c;10年以上C/C, C#, Java等多种编程语言开发经验&#xff0c;拥有高级工程师证书&#xff1b;擅长C/C、C#等开发语言&#xff0c;熟悉Java常用开…...

矩阵分解相关知识点总结(二)

文章目录 三、矩阵的QR分解3.1、Givens矩阵与Givens变换3.2、Householder矩阵与Householder变换3.3、QR分解 书接上文矩阵分解相关知识点总结&#xff08;一&#xff09; 三、矩阵的QR分解 3.1、Givens矩阵与Givens变换 设非零列向量 x ∈ R n \bm{x}\in {\bf{R}}^n x∈Rn及单…...

MySQL——视图 用户管理 语言访问

目录 视图 用户管理 数据库权限 访问 准备工作 使用函数 mysql界面级工具 连接池 视图 这里的视图与事务中的读视图是两个不同的概念&#xff1a;视图是一个虚拟表&#xff0c;其内容由查询定义。同真实的表一样&#xff0c;视图包含一系列带有名称的列和行数据。视图的…...

二、Sqoop 详细安装部署教程

作者&#xff1a;IvanCodes 日期&#xff1a;2025年6月2日 专栏&#xff1a;Sqoop教程 Apache Sqoop 是一个强大的工具&#xff0c;用于在 Hadoop (HDFS, Hive, HBase) 与关系型数据库 (如 MySQL, PostgreSQL, Oracle) 之间高效传输数据。本教程将详细指导您如何根据官方网站截…...

用Python开启游戏开发之旅

在当今丰富多彩的数字娱乐世界中&#xff0c;游戏以其独特的魅力吸引着无数人的目光。而Python这门功能强大又简洁易懂的编程语言&#xff0c;也为游戏开发打开了一扇充满创意的大门。 一、选择Python的理由 Python之所以备受游戏开发者青睐&#xff0c;有诸多原因。其一&#…...

React 第五十四节 Router中useRevalidator的使用详解及案例分析

前言 useRevalidator 是 React Router v6.4 引入的一个强大钩子&#xff0c;用于在数据路由&#xff08;Data Router&#xff09;中手动触发路由数据的重新验证&#xff08;revalidation&#xff09;。 它在需要主动刷新数据而不改变路由位置的场景中非常有用。 一、useReval…...

【C语言预处理详解(下)】--#和##运算符,命名约定,命令行定义 ,#undef,条件编译,头文件的包含,嵌套文件包含,其他预处理指令

目录 五.#和##运算符 5.1--#运算符 5.2--##运算符 六.命名约定&#xff0c;#undef&#xff0c;命令行定义 6.1--命名约定 6.2--#undef 6.3--命名行定义 七.条件编译 常见的条件编译指令&#xff1a; 1.普通的条件编译&#xff1a; 2.多个分支的条件编译(可以利用条…...

03.搭建K8S集群

K8S集群搭建的方式 目前主流的搭建k8s集群的方式有kubeadm、minikube、二进制包三种方式&#xff1a; kubeadm&#xff08;本案例搭建方式&#xff09; 是一个工具&#xff0c;用于快速搭建kubernetes集群&#xff0c;目前应该是比较方便和推荐的&#xff0c;简单易用 kubea…...

RDMA简介3之四种子协议对比

RDMA协议共有四种子协议&#xff0c;分别为InfiniBand、iWARP、RoCE v1和RoCE v2协议。这四种协议使用统一的RDMA API&#xff0c;但在具体的网络层级实现上有所不同&#xff0c;如图1所示&#xff0c;接下来将分别介绍这四种子协议。 图1 RDMA四种子协议网络层级关系图 Infin…...

【最新版】西陆洗车系统源码全开源+uniapp前端+搭建教程

一.系统介绍 一款基于ThinkPHPUniapp开发的多门店洗车系统&#xff0c;包含用户端&#xff08;小程序&#xff09;、门店员工端&#xff08;小程序&#xff09;、门店端&#xff08;PC&#xff09;、平台管理端&#xff08;PC&#xff09;。 门店分连锁门店和独立门店&#xf…...

力扣LeetBook数组和字符串--二维数组

1.旋转矩阵 题目链接 想了那么久的各种旋转&#xff0c;对角线&#xff0c;其实把问题搞复杂了。 旋转90度的本质无非就是转置镜像对称 转置是什么&#xff1f;&#xff1a;将矩阵的行和列互换。 镜像对称&#xff1a;把矩阵从中间对折&#xff0c;互换位置 矩阵 A A [ 1 3 0…...

Linux开发工具(apt,vim,gcc)

目录 yum/apt包管理器 Linux编辑器 vim 1.见一见vim 2.vim的多模式 3.命令模式底行模式等 4.vim的配置 Linux编译器 gcc/g 1.预处理&#xff08;宏替换&#xff09; 2.编译&#xff08;生成汇编&#xff09; 3.汇编&#xff08;生成机器可识别代码&#xff09; 4.连…...

C# ExcelWorksheet 贴图

C# ExcelWorksheet 贴图 在C#中,如果你想在Excel工作表中插入图片(例如,在ExcelWorksheet中贴图),你可以使用ClosedXML或EPPlus这样的库来操作Excel文件。下面我将分别介绍如何使用这两个库来实现这一功能。 使用ClosedXML 首先,确保你已经安装了ClosedXML包。你可以通…...

鸿蒙Next开发真机调试签名申请流程

背景&#xff1a; 在学习鸿蒙next开发应用的初期总是会遇到一堆的问题&#xff0c;毕竟鸿蒙next开发不管是他的ArKTS语言还是他的开发工具DevEco Studio都还在起步阶段&#xff0c;就像当初的Android起步一样&#xff0c;总会有资料不足的一些问题。就比如我们学习下载完DevEco…...

[yolov11改进系列]基于yolov11引入上下文锚点注意力CAA的python源码+训练源码

【CAA介绍】 本文记录的是基于CAA注意力模块的RT-DETR目标检测改进方法研究。在远程遥感图像或其他大尺度变化的图像中目标检测任务中&#xff0c;为准确提取其长距离上下文信息&#xff0c;需要解决大目标尺度变化和多样上下文信息时的不足的问题。CAA能够有效捕捉长距离依赖…...

【Elasticsearch】 查询优化方式

在优化Elasticsearch的查询性能时&#xff0c;可以从多个维度着手&#xff0c;包括索引设计、查询优化、集群配置、数据管理以及监控分析等。常见的优化方式和策略有以下几种&#xff1a; 一、索引优化 合理设计字段类型&#xff1a; 字段类型选择&#xff1a; 对于不需要分词的…...

Xcode 16.4 + iOS 18 系统运行时崩溃:___cxa_current_primary_exception 符号丢失的原因与解决方案

在使用 Xcode 16.4 构建项目&#xff0c;运行到 iOS 18.3 或更早版本系统&#xff08;包括模拟器&#xff09;时&#xff0c;出现了如下的运行时崩溃&#xff1a; dyld[22183]: Symbol not found: ___cxa_current_primary_exceptionReferenced from: /.../WidgetOn.app/Widget…...

【linux】全志Tina预编译一个so库文件到根文件系统/usr/lib/下

一、sdk中新建文件夹 路径&#xff1a; V:\t113\work3\t113\openwrt\package\feeds\libs\md5util md5util为需要注入的库文件夹。 文件结构 libs md5util files libmd5util.so makefile etc.. 二、编写makefile include $(TOPDIR)/rules.mkPKG_NAME : md5util PKG_VERSIO…...

C# 类和继承(成员访回修饰符)

成员访回修饰符 本章之前的两节阐述了类的可访问性。对类的可访问性&#xff0c;只有两种修饰符&#xff1a;internal和public。 本节阐述成员的可访问性。类的可访问性描述了类的可见性&#xff1b;成员的可访问性描述了类成员的可 见性。 声明在类中的每个成员对系统的不同…...

c++ stl容器之map用法

目录 &#xff08;1&#xff09;map介绍 &#xff08;2&#xff09;map、multimap、unordered_map区别 &#xff08;3&#xff09;map用法 1.map接口表 2.使用举例 插入数据与遍历数据 查找关键字和值 删除元素 按照值排序 &#xff08;4&#xff09;multimap用法 &…...

Linux-文件管理及归档压缩

1.根下的目录作用说明&#xff1a; /&#xff1a;Linux系统中所有的文件都在根下/bin&#xff1a;(二进制命令目录)存放常用的用户命令/boot&#xff1a;系统启动时的引导文件&#xff08;内核的引导配置文件&#xff0c;grub配置文件&#xff0c;内核配置文件&#xff09; 例…...