花10分钟写个漂亮的后端API接口模板!
你好,我是田哥
在这微服务架构盛行的黄金时段,加上越来越多的前后端分离,导致后端API接口规范变得越来越重要了。
比如:统一返回参数形式、统一返回码、统一异常处理、集成swagger等。
目的主要是规范后端项目代码,以及便于前端沟通联通以及问题的排查。
本文内容:

统一返回参数形式
目前主流的返回参数形式:
{"code": 200000,"message": "成功","data": {"id": 1,"userName": "tiange","password": "123456","phone": "18257160375","gender": 0,"status": 0,"createTime": "2024-05-17 20:24:40"}
} code是接口返回编码,message是消息提示,data是具体返回数据内容。
返回码
返回码定义很重要,我们应该可以参考HTTP请求返回的状态码(下面是常见的HTTP状态码):
200 - 请求成功
301 - 资源(网页等)被永久转移到其它URL
404 - 请求的资源(网页等)不存在
500 - 内部服务器错误 这样前端开发人员在得到返回值后,根据状态码就可以知道,大概什么错误,再根据message相关的信息描述,可以快速定位。
由于我们业务系统中可能会又大量的code,所以,我们对此做一个改良。
/*** {@code @description:} 返回码** @author tianwc 公众号:Java后端技术全栈* 在线刷题 1200+java面试题和1000+篇技术文章:<a href="https://woaijava.cc/">博客地址</a>* {@code @date:} 2024-07-28 15:10* {@code @version:} 1.0*/
@Getter
public enum ResultCode implements Serializable {SUCCESS(200000, "成功"),FAIL(500000, "系统错误,请稍后重试!"),USER_NOT_EXIST(401000, "用户不存在"),USER_CANCELLED(401001, "用户已注销"),USER_ROLE_ERROR(401002, "用户角色不对"),NOT_FOUND(404000, "接口不存在"),PARAMETER_ERROR(404001, "参数有误"),PARAMETER_IS_NULL(404002, "参数为空");private final int code;private final String message;ResultCode(int code, String message) {this.code = code;this.message = message;}
} 对此,我们还可以进一步细分,比如402开头的是用户相关的 、403开头又是xxx的,.....
这样后期如果又什么问题,这样就能快速定位到具体模块中。
统一返回
我们可以专门写一个类来对返回数据进行包装。
/*** {@code @description:} 返回结果马甲** @author tianwc 公众号:Java后端技术全栈* 在线刷题 1200+java面试题和1000+篇技术文章:<a href="https://woaijava.cc/">博客地址</a>* {@code @date:} 2024-07-28 15:12* {@code @version:} 1.0*/
@Data
public class Result implements Serializable {private Integer code;private String message;private Object data;public Result(Integer code, String message, Object data) {this.code = code;this.message = message;this.data = data;}public static Result success() {return new Result(ResultCode.SUCCESS.getCode(), ResultCode.SUCCESS.getMessage(), null);}public static Result success(Object data) {return new Result(ResultCode.SUCCESS.getCode(), ResultCode.SUCCESS.getMessage(), data);}public static Result fail(ResultCode resultCode) {return new Result(resultCode.getCode(), resultCode.getMessage(), null);}public static Result fail(int code, String message) {return new Result(code, message, null);}
} 我们定义了常用的四种格式。
具体使用如下:
我们在Service层和实现层:
public interface UserInfoService extends IService<UserInfo> {Result findByCondition(UserInfoReqDto userInfoReqDto);
} @Service
public class UserInfoServiceImpl extends ServiceImpl<UserInfoMapper, UserInfo> implements UserInfoService {@Overridepublic Result findByCondition(UserInfoReqDto userInfoReqDto) {Wrapper<UserInfo> wrapper = Wrappers.<UserInfo>lambdaQuery().eq(UserInfo::getUserName, userInfoReqDto.getUserName()).eq(UserInfo::getPassword, userInfoReqDto.getPassword());return Result.success(this.baseMapper.selectList(wrapper));}
} 在controller层:我们会在controller层处理业务请求,并返回给前端 。
@RestController
@RequestMapping("/user/info")
public class UserInfoController {@Resourceprivate UserInfoService userInfoService; @GetMapping("/condition")public Result findByCondition(UserInfoReqDto userInfoReqDto) {return userInfoService.findByCondition(userInfoReqDto);}
} 执行:
GET http://localhost:8089/user/info/condition?userName=tiange&password=123456
返回:
{"code": 200000,"message": "成功","data": [{"id": 1,"userName": "tiange","password": "123456","phone": "18257160375","gender": 0,"status": 0,"createTime": "2024-05-17T20:24:40.000+00:00"}]
} 前端根据我们但会的code判断是否需要取data字段。
统一异常处理
统一异常处理我们分业务异常、系统异常以及参数异常:
业务异常
我们自定义一个业务异常:BusinessException
/*** @author tianwc 公众号:java后端技术全栈、面试专栏* @version 1.0.0* @date 2024-07-28 15:12* 在线刷题 1200+java面试题和1000+篇技术文章:<a href="https://woaijava.cc/">博客地址</a>* <p>* 自定义业务异常*/
@Getter
public class BusinessException extends RuntimeException {/*** http状态码*/private Integer code;private Object object;public BusinessException(String message, Integer code, Object object) {super(message);this.code = code;this.object = object;}public BusinessException(String message, Integer code) {super(message);this.code = code;}public BusinessException(ResultCode resultCode) {super(resultCode.getMessage());this.code = resultCode.getCode();this.object = resultCode.getMessage();}public BusinessException(ResultCode resultCode, String message) {this.code = resultCode.getCode();this.object = message;}public BusinessException(String message) {super(message);}} 异常处理:GlobalAdvice
@RestControllerAdvice
@Slf4j
public class GlobalAdvice {@ExceptionHandler(Exception.class)public Result doException(Exception e) {log.error("统一异常处理机制,触发异常 msg ", e);String message = null;int errorCode = ResultCode.FAIL.getCode();//自定义异常if (e instanceof BusinessException) {BusinessException exception = (BusinessException) e;message = exception.getMessage();errorCode = exception.getCode();} else if (e instanceof HttpRequestMethodNotSupportedException) {message = "不支持GET/POST方法";} else if (e instanceof NoHandlerFoundException) {message = "请求接口不存在";} else if (e instanceof MissingServletRequestParameterException) {errorCode = ResultCode.PARAMETER_IS_NULL.getCode();message = String.format("缺少必要参数[%s]", ((MissingServletRequestParameterException) e).getParameterName());} else if (e instanceof MethodArgumentNotValidException) {BindingResult result = ((MethodArgumentNotValidException) e).getBindingResult();FieldError error = result.getFieldError();errorCode = ResultCode.PARAMETER_IS_NULL.getCode();message = error == null ? ResultCode.PARAMETER_ERROR.getMessage() : error.getDefaultMessage();} else if (e instanceof BindException) {errorCode = ResultCode.PARAMETER_IS_NULL.getCode();message = ((BindException) e).getFieldError().getDefaultMessage();} else if (e instanceof IllegalArgumentException) {errorCode = ResultCode.PARAMETER_IS_NULL.getCode();message = e.getMessage();}return Result.fail(errorCode, message);}
} 使用:
@Service
public class UserInfoServiceImpl extends ServiceImpl<UserInfoMapper, UserInfo> implements UserInfoService {@Overridepublic Result findByCondition(UserInfoReqDto userInfoReqDto) {if("admin".equals(userInfoReqDto.getUserName())){//对于某些业务问题抛出自定义异常throw new BusinessException(ResultCode.USER_ROLE_ERROR);}Wrapper<UserInfo> wrapper = Wrappers.<UserInfo>lambdaQuery().eq(UserInfo::getUserName, userInfoReqDto.getUserName()).eq(UserInfo::getPassword, userInfoReqDto.getPassword());return Result.success(this.baseMapper.selectList(wrapper));}
} 系统异常
假设系统异常:
@Service
public class UserInfoServiceImpl extends ServiceImpl<UserInfoMapper, UserInfo> implements UserInfoService {@Overridepublic Result findByCondition(UserInfoReqDto userInfoReqDto) {if("123456".equals(userInfoReqDto.getPassword())){throw new RuntimeException("你的系统异常了");}Wrapper<UserInfo> wrapper = Wrappers.<UserInfo>lambdaQuery().eq(UserInfo::getUserName, userInfoReqDto.getUserName()).eq(UserInfo::getPassword, userInfoReqDto.getPassword());return Result.success(this.baseMapper.selectList(wrapper));}
} 执行:
GET http://localhost:8089/user/info/condition?userName=tian&password=123456 返回结果:
{"code": 500000,"message": "系统异常","data": null
} 参数校验
添加pom依赖
<!--参数验证-->
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-validation</artifactId>
</dependency> 请求参数:
@Data
@AllArgsConstructor
@NoArgsConstructor
public class UserInfoReqDto {@NotBlank(message = "姓名不能为空")private String userName;@NotBlank(message = "密码不能为空")private String password;
} 其他相关注解:
| 注解 | 作用 |
|---|---|
| @NotNull | 判断包装类是否为null |
| @NotBlank | 判断字符串是否为null或者是空串(去掉首尾空格) |
| @NotEmpty | 判断集合是否为空 |
| @Length | 判断字符的长度(最大或者最小) |
| @Min | 判断数值最小值 |
| @Max | 判断数值最大值 |
| 判断邮箱是否合法 |
controller层添加注解@Validated
@RestController
@RequestMapping("/user/info")
public class UserInfoController {@Resourceprivate UserInfoService userInfoService; @GetMapping("/condition")public Result findByCondition(@Validated UserInfoReqDto userInfoReqDto) {return userInfoService.findByCondition(userInfoReqDto);}
} 最后在统一异常处理里处理。
执行:
GET http://localhost:8089/user/info/condition?userName=tian 返回:
{"code": 404002,"message": "密码不能为空","data": null
} 执行:
GET http://localhost:8089/user/info/condition?password=123456 返回:
{"code": 404002,"message": "姓名不能为空","data": null
} 集成mybatis-plus
添加依赖
<!--mybatis-plus 依赖-->
<dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-boot-starter</artifactId><version>${mybatis-plus.version}</version>
</dependency>
<!--mysql依赖-->
<dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><scope>runtime</scope>
</dependency> 数据库信息配置:
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
spring.datasource.jdbc-url=jdbc:mysql://localhost:3306/user-center?useSSL=false&serverTimezone=UTC&useUnicode=true&characterEncoding=utf-8&allowPublicKeyRetrieval=true
spring.datasource.username=root
spring.datasource.password=123456 mybatis-plus配置:
@Configuration
@MapperScan(basePackages = "com.tian.dao.mapper")
public class DataSourceConfig {@ConfigurationProperties(prefix = "spring.datasource")@Beanpublic DataSource dataSource() {return DataSourceBuilder.create().build();}@Beanpublic MybatisPlusInterceptor mybatisPlusInterceptor() {MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();//分页插件interceptor.addInnerInterceptor(new PaginationInnerInterceptor());//注册乐观锁插件interceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor());return interceptor;}@Beanpublic SqlSessionFactory sqlSessionFactory(DataSource dataSource, MybatisPlusInterceptor interceptor) throws Exception {MybatisSqlSessionFactoryBean ssfb = new MybatisSqlSessionFactoryBean();ssfb.setDataSource(dataSource);ssfb.setPlugins(interceptor);//到哪里找xml文件ssfb.setMapperLocations(new PathMatchingResourcePatternResolver().getResources("classpath*:mapper/*.xml"));return ssfb.getObject();}
} 实体类:
@TableName(value = "user_info")
@Data
public class UserInfo {/*** 主键ID*/@TableId(value = "id")private Long id;/*** 姓名*/@TableField(value = "user_name")private String userName;/*** 密码*/@TableField(value = "password")private String password;/*** 手机号*/@TableField(value = "phone")private String phone;/*** 性别,0:女,1:男*/@TableField(value = "gender")private Integer gender;/*** 状态,0:正常,1:已注销*/@TableField(value = "status")private Integer status;/*** 注册时间*/@TableField(value = "create_time")private Date createTime;@TableField(exist = false)private static final long serialVersionUID = 1L;
} mapper:
public interface UserInfoMapper extends BaseMapper<UserInfo> {
} service部分代码参照前面的代码来。
执行
GET http://localhost:8089/user/info/condition?userName=tiange&password=123456 返回
{"code": 200000,"message": "成功","data": [{"id": 1,"userName": "tiange","password": "123456","phone": "18257160375","gender": 0,"status": 0,"createTime": "2024-05-17T20:24:40.000+00:00"}]
} 到这里我们的项目就成功把mybatis-plus集成进来。
swagger
作为前后端分离项目,在团队开发中,一个好的 API 文档不但可以减少大量的沟通成本,还可以帮助一位新人快速上手业务。传统的做法是由开发人员创建一份 RESTful API文档来记录所有的接口细节,并在程序员之间代代相传。这种做法存在以下几个问题:
1)API 接口众多,细节复杂,需要考虑不同的HTTP请求类型、HTTP头部信息、HTTP请求内容等,想要高质量的完成这份文档需要耗费大量的精力;
2)难以维护。随着需求的变更和项目的优化、推进,接口的细节在不断地演变,接口描述文档也需要同步修订,可是文档和代码处于两个不同的媒介,除非有严格的管理机制,否则很容易出现文档、接口不一致的情况;
Swagger2 的出现就是为了从根本上解决上述问题。它作为一个规范和完整的框架,可以用于生成、描述、调用和可视化 RESTful 风格的 Web 服务:
接口文档在线自动生成,文档随接口变动实时更新,节省维护成本;
支持在线接口测试,不依赖第三方工具;
Swagger2 是一个规范和完整的框架,用于生成、描述、调用和可视化Restful风格的web服务,现在我们使用spring boot 整合它。作用:
接口的文档在线自动生成;
功能测试;
常用注解
| 注解 | 描述 |
|---|---|
| @Api | 将类标记为 Swagger 资源。 |
| @ApiImplicitParam | 表示 API 操作中的单个参数。 |
| @ApiImplicitParams | 允许多个 ApiImplicitParam 对象列表的包装器。 |
| @ApiModel | 提供有关 Swagger 模型的其他信息。 |
| @ApiModelProperty | 添加和操作模型属性的数据。 |
| @ApiOperation | 描述针对特定路径的操作或通常是 HTTP 方法。 |
| @ApiParam | 为操作参数添加额外的元数据。 |
| @ApiResponse | 描述操作的可能响应。 |
| @ApiResponses | 允许多个 ApiResponse 对象列表的包装器。 |
| @Authorization | 声明要在资源或操作上使用的授权方案。 |
| @AuthorizationScope | 描述 OAuth2 授权范围。 |
swagger配置
@Configuration //加入到容器里面
@EnableSwagger2 //开启Swagger
public class SwaggerConfig {@Beanpublic Docket docket() {return new Docket(DocumentationType.SWAGGER_2).apiInfo(apiInfo()).select().apis(RequestHandlerSelectors.basePackage("com.tian.controller")).build();}private ApiInfo apiInfo(){Contact contact = new Contact("web项目demo", "https://www.woaijava.cc/", "251965157@qq.com");return new ApiInfo("web项目demo的API文档","练手所用","v1.0","https://www.woaijava.cc/",contact,"Apache 2.0","http://www.apache.org/licenses/LICENSE-2.0",new ArrayList());}} 我们就可以在对应业务代码中标注上swagger:
@RestController
@RequestMapping("/user/info")
@Api(value = "用户信息接口",tags = "用户信息")
public class UserInfoController {@Resourceprivate UserInfoService userInfoService;@GetMapping("/{id}")@ApiOperation(value = "根据id查询用户信息", notes = "根据id查询用户信息",produces = "application/json",consumes = "application/json")@ApiImplicitParams({@ApiImplicitParam(name="id",value="用户id",required = true,dataType = "Integer")})public Result findById(@PathVariable("id") Integer id) {return Result.success(userInfoService.getById(id));}@GetMapping("/condition")@ApiOperation(value = "根据条件查询用户信息")public Result findByCondition(@Validated UserInfoReqDto userInfoReqDto) {return userInfoService.findByCondition(userInfoReqDto);}
} @Data
@AllArgsConstructor
@NoArgsConstructor
@ApiModel(value="用户信息查询条件")
public class UserInfoReqDto {@NotBlank(message = "姓名不能为空")@ApiModelProperty(value="姓名")private String userName;@NotBlank(message = "密码不能为空")@ApiModelProperty(value="密码")private String password;
} 效果
启动项目,访问:
http://localhost:8089/swagger-ui.html



也到这里,我们就基本形成了一个完整的demo级后端项目。
代码已上传到知识星球:


其他推荐
2024年最新面试总结,请查收!
充电桩项目如何部署?
应届生不会写简历?手把手教你怎么写简历
背八股文,不妨尝试这招!
相关文章:
花10分钟写个漂亮的后端API接口模板!
你好,我是田哥 在这微服务架构盛行的黄金时段,加上越来越多的前后端分离,导致后端API接口规范变得越来越重要了。 比如:统一返回参数形式、统一返回码、统一异常处理、集成swagger等。 目的主要是规范后端项目代码,以及…...
评估分类机器学习模型的指标
欢迎来到雲闪世界。一旦我们训练了一个监督机器学习模型来解决分类问题,如果这就是我们工作的结束,我们会很高兴,我们可以直接向他们输入新数据。我们希望它能正确地对所有内容进行分类。然而,实际上,模型做出的预测并…...
农机自动化:现代农业的未来趋势
随着人口的增长和农业生产的需求不断增加,提高农业生产效率成为现代农业的重要目标。农机自动化作为一种新兴技术,可以大幅度提升农机的使用效率和生产能力。农机自动化是指利用先进的传感技术、数据处理和人工智能技术,使农机能够自动完成农…...
25考研操作系统复习·1.1/1.2/1.3 操作系统的基本概念/发展历程/运行环境
目录 操作系统的基本概念 概念(定义) 功能和目标 资源的管理者 向上层提供服务 给普通用户的 给软件/程序员的 对硬件机器的拓展 操作系统的特征 操作系统的发展历程 操作系统的运行环境 操作系统的运行机制 中断和异常 中断的作用 中断的…...
如何培养学生的创新意识和实践能力
培养学生的创新意识和实践能力是一个复杂而系统的过程,涉及多个方面的努力和措施。以下是一些具体的做法: 一、培养学生的创新意识 提供创新环境: 为学生创造一个开放、自由、支持创新的学习环境,让他们能够自由地表达自己的想法…...
四、GD32 MCU 常见外设介绍(15)CAN 模块介绍
CAN是控制器局域网络(Controller Area Network)的简称,它是由研发和生产汽车电子产品著称的德国BOSCH公司开发的,并最终成为国际标准(ISO11519),是国际上应用最广泛的现场总线之一。 CAN总线协议已经成为汽车计算机控…...
AIGC大模型产品经理高频面试大揭秘‼️
近期有十几个学生在面试大模型产品经理(薪资还可以,详情见下图),根据他们面试(包括1-4面)中出现高频大于3次的问题汇总如下,一共32道题目(有答案)。 29.讲讲T5和Bart的区…...
【嵌入式笔记】【C语言】struct union
结构体(Struct)定义: struct 结构体名 {member1; // 成员1,可以是任何基本数据类型或复合类型member2; // 成员2... };//例如: struct Point {float x;float y;...
【初学人工智能原理】【9】深度学习:神奇的DeepLearning
前言 本文教程均来自b站【小白也能听懂的人工智能原理】,感兴趣的可自行到b站观看。 代码及工具箱 本专栏的代码和工具函数已经上传到GitHub:1571859588/xiaobai_AI: 零基础入门人工智能 (github.com),可以找到对应课程的代码 正文 深度…...
[RoarCTF 2019]Easy Calc1
打开题目 查看源码,看到 看到源代码有 calc.php,构造url打开 看到php审计代码, 由于页面中无法上传num,则输入 num,在num前加入一个空格可以让num变得可以上传,而且在进行代码解析时,php会把前…...
安卓APK安装包arm64-v8a、armeabi-v7a、x86、x86_64有何区别?如何选择?
在GitHub网站下载Android 安装包,Actions资源下的APK文件通常有以下版本供选择: 例如上图是某Android客户端的安装包文件,有以下几个版本可以选择: mobile-release.apk(通用版本,体积最大)mobi…...
【AI大模型】通义千问:开启语言模型新篇章与Function Call技术的应用探索
文章目录 前言一、大语言模型1.大模型介绍2.大模型的发展历程3.大模型的分类a.按内容分类b.按应用分类 二、通义千问1.通义千问模型介绍a.通义千问模型介绍b.应用场景c.模型概览 2.对话a.对话的两种方式通义千问API的使用 b.单轮对话Vue页面代码:Django接口代码 c.多…...
详细教程 MySQL 数据库 下载 安装 连接 环境配置 全面
数据库就是储存和管理数据的仓库,对数据进行增删改查操作,其本质是一个软件。 首先数据有两种,一种是关系型数据库,另一种是非关系型数据库。 关系型数据库是以表的形式来存储数据,表和表之间可以有很多复杂的关系&a…...
门控循环单元GRU
目录 一、GRU提出的背景:1.RNN存在的问题:2.GRU的思想: 二、更新门和重置门:三、GRU网络架构:1.更新门和重置门如何发挥作用:1.1候选隐藏状态H~t:1.2隐藏状态Ht: 2.GRU: 四、底层源码…...
程序员修炼之路
成为一名优秀的程序员,需要广泛而深入地学习多个领域的知识。这些课程不仅帮助建立扎实的编程基础,还培养了问题解决、算法设计、系统思维等多方面的能力。以下是一些核心的必修课: 计算机基础 计算机组成原理:理解计算机的硬件组…...
PHP时间相关函数
时间、日期 time()获取当前时间戳(10位)microtime(true)返回一个浮点时间戳data(格式,时间戳)日期格式化 $time time(); echo date(Y-m-d H:i:s, $time);strtotime&am…...
python进阶——python面向对象
前言 Python是一种面向对象的编程语言,可在Python中使用类和对象来组织和封装代码。面向对象编程(OOP)是一种编程范例,它将数据和操作数据的方法封装在一个对象内部,通过对象之间的交互来实现程序的功能。 1、面向对象…...
【无标题】vue2鼠标悬停(hover)时切换图片
在Vue 2中,要实现鼠标悬停(hover)时切换图片的功能,你不能直接在模板的:src绑定中处理这个逻辑,因为Vue的模板不支持条件渲染的复杂逻辑(如基于鼠标状态的动态图片切换)。但是,你可以…...
每天一个数据分析题(四百五十九)- 分析法
故障树分析法经常与哪些方法联合使用? A. 头脑风暴法 B. 五问法 C. 配对法 D. 引力法 数据分析认证考试介绍:点击进入 题目来源于CDA模拟题库 点击此处获取答案 数据分析专项练习题库 内容涵盖Python,SQL,统计学…...
英语:十、助动词和情态动词
1、助动词 (1)助动词be a、助动词be人称、数及时态的变化 be在作助动词时,也和系动词一样,有人称、数及时态的变化。 人称 数 现在时态 过去时态 现在分词 过去分词 第一人称 单数 am was being been 复数 are w…...
19c补丁后oracle属主变化,导致不能识别磁盘组
补丁后服务器重启,数据库再次无法启动 ORA01017: invalid username/password; logon denied Oracle 19c 在打上 19.23 或以上补丁版本后,存在与用户组权限相关的问题。具体表现为,Oracle 实例的运行用户(oracle)和集…...
使用VSCode开发Django指南
使用VSCode开发Django指南 一、概述 Django 是一个高级 Python 框架,专为快速、安全和可扩展的 Web 开发而设计。Django 包含对 URL 路由、页面模板和数据处理的丰富支持。 本文将创建一个简单的 Django 应用,其中包含三个使用通用基本模板的页面。在此…...
Admin.Net中的消息通信SignalR解释
定义集线器接口 IOnlineUserHub public interface IOnlineUserHub {/// 在线用户列表Task OnlineUserList(OnlineUserList context);/// 强制下线Task ForceOffline(object context);/// 发布站内消息Task PublicNotice(SysNotice context);/// 接收消息Task ReceiveMessage(…...
centos 7 部署awstats 网站访问检测
一、基础环境准备(两种安装方式都要做) bash # 安装必要依赖 yum install -y httpd perl mod_perl perl-Time-HiRes perl-DateTime systemctl enable httpd # 设置 Apache 开机自启 systemctl start httpd # 启动 Apache二、安装 AWStats࿰…...
STM32F4基本定时器使用和原理详解
STM32F4基本定时器使用和原理详解 前言如何确定定时器挂载在哪条时钟线上配置及使用方法参数配置PrescalerCounter ModeCounter Periodauto-reload preloadTrigger Event Selection 中断配置生成的代码及使用方法初始化代码基本定时器触发DCA或者ADC的代码讲解中断代码定时启动…...
【论文笔记】若干矿井粉尘检测算法概述
总的来说,传统机器学习、传统机器学习与深度学习的结合、LSTM等算法所需要的数据集来源于矿井传感器测量的粉尘浓度,通过建立回归模型来预测未来矿井的粉尘浓度。传统机器学习算法性能易受数据中极端值的影响。YOLO等计算机视觉算法所需要的数据集来源于…...
跨链模式:多链互操作架构与性能扩展方案
跨链模式:多链互操作架构与性能扩展方案 ——构建下一代区块链互联网的技术基石 一、跨链架构的核心范式演进 1. 分层协议栈:模块化解耦设计 现代跨链系统采用分层协议栈实现灵活扩展(H2Cross架构): 适配层…...
全志A40i android7.1 调试信息打印串口由uart0改为uart3
一,概述 1. 目的 将调试信息打印串口由uart0改为uart3。 2. 版本信息 Uboot版本:2014.07; Kernel版本:Linux-3.10; 二,Uboot 1. sys_config.fex改动 使能uart3(TX:PH00 RX:PH01),并让boo…...
3-11单元格区域边界定位(End属性)学习笔记
返回一个Range 对象,只读。该对象代表包含源区域的区域上端下端左端右端的最后一个单元格。等同于按键 End 向上键(End(xlUp))、End向下键(End(xlDown))、End向左键(End(xlToLeft)End向右键(End(xlToRight)) 注意:它移动的位置必须是相连的有内容的单元格…...
人工智能(大型语言模型 LLMs)对不同学科的影响以及由此产生的新学习方式
今天是关于AI如何在教学中增强学生的学习体验,我把重要信息标红了。人文学科的价值被低估了 ⬇️ 转型与必要性 人工智能正在深刻地改变教育,这并非炒作,而是已经发生的巨大变革。教育机构和教育者不能忽视它,试图简单地禁止学生使…...
