Spring Boot 的核心注解
一、引言
Spring Boot 作为一种流行的 Java 开发框架,以其简洁高效的开发方式受到广泛关注。其中,核心注解在 Spring Boot 应用的开发中起着至关重要的作用。理解这些注解的含义和用法,对于充分发挥 Spring Boot 的优势至关重要。本文将深入剖析 Spring Boot 的核心注解。
二、@SpringBootApplication 注解
(一)功能概述
- 组合注解
- @SpringBootApplication 是一个组合注解,它包含了 @Configuration、@EnableAutoConfiguration 和 @ComponentScan 三个注解的功能。
- 配置应用
- 该注解用于标识一个 Spring Boot 应用的主配置类,告诉 Spring Boot 框架如何进行自动配置和组件扫描。
(二)用法详解
- 定义主类
- 在 Spring Boot 应用中,通常会创建一个主类,并使用 @SpringBootApplication 注解进行标识。这个主类通常包含一个 main 方法,用于启动应用程序。
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;@SpringBootApplication
public class MyApplication {public static void main(String[] args) {SpringApplication.run(MyApplication.class, args);}
}
-
自动配置
- @SpringBootApplication 注解会触发 Spring Boot 的自动配置机制。Spring Boot 会根据项目中引入的依赖和类路径中的内容,自动配置应用程序所需的各种组件和配置。
-
组件扫描
- 该注解还会启动组件扫描功能,自动扫描主类所在的包及其子包中的所有带有 @Component、@Service、@Repository、@Controller 等注解的类,并将它们注册为 Spring 容器中的 Bean。
三、@Configuration 注解
(一)功能介绍
- 定义配置类
- @Configuration 注解用于标识一个类为 Spring 的配置类。在配置类中,可以使用 @Bean 注解定义 Bean,并进行各种配置。
- 替代 XML 配置
- 随着 Spring 的发展,越来越多的开发者倾向于使用基于 Java 的配置方式,而 @Configuration 注解就是实现这种配置方式的关键。
(二)用法示例
- 定义 Bean
- 在配置类中,可以使用 @Bean 注解定义 Bean。例如,定义一个名为 myService 的 Bean:
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@Configuration
public class AppConfig {@Beanpublic MyService myService() {return new MyServiceImpl();}
}
- 导入其他配置类
- 可以使用 @Import 注解导入其他配置类,实现配置的模块化。例如:
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Import;@Configuration
@Import(AnotherConfig.class)
public class MainConfig {
}
四、@EnableAutoConfiguration 注解
(一)自动配置原理
- 依赖分析
- @EnableAutoConfiguration 注解会根据项目中引入的依赖,自动配置应用程序所需的各种组件。例如,如果项目中引入了 Spring Data JPA 的依赖,Spring Boot 会自动配置数据库连接和 JPA 相关的组件。
- 排除特定自动配置
- 有时候,可能需要排除某些自动配置。可以使用 @SpringBootApplication 的 exclude 属性来排除特定的自动配置类。例如:
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class})
public class MyApplication {public static void main(String[] args) {SpringApplication.run(MyApplication.class, args);}
}
(二)使用场景
- 快速开发
- 在快速开发过程中,@EnableAutoConfiguration 注解可以大大减少配置的工作量,提高开发效率。
- 集成第三方库
- 当集成第三方库时,Spring Boot 的自动配置功能可以自动配置与该库相关的组件,使得集成更加简单快捷。
五、@ComponentScan 注解
(一)扫描机制
- 包扫描
- @ComponentScan 注解用于指定 Spring 容器进行组件扫描的包路径。默认情况下,它会扫描主类所在的包及其子包中的所有带有 @Component、@Service、@Repository、@Controller 等注解的类,并将它们注册为 Spring 容器中的 Bean。
- 自定义扫描路径
- 可以通过 basePackages 属性指定自定义的扫描路径。例如:
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;@Configuration
@ComponentScan(basePackages = {"com.example.myapp.service", "com.example.myapp.repository"})
public class AppConfig {
}
(二)与其他注解的配合
- 与 @Controller、@Service、@Repository 等注解配合
- @ComponentScan 注解与这些注解配合使用,可以实现自动注册 Bean 的功能。例如,当一个类被标注为 @Service 时,@ComponentScan 注解会将其扫描并注册为 Spring 容器中的一个服务 Bean。
- 与 @Configuration 注解配合
- 在配置类中,通常会使用 @ComponentScan 注解来指定组件扫描的范围,以便将其他带有注解的类注册为 Bean。
六、@Autowired 注解
(一)依赖注入原理
- 自动装配
- @Autowired 注解用于实现依赖注入。当一个类的字段、构造函数或方法被标注为 @Autowired 时,Spring 容器会自动将匹配的 Bean 注入到该字段、构造函数或方法中。
- 按类型匹配
- Spring 容器会根据 Bean 的类型进行匹配。如果有多个相同类型的 Bean,还可以使用 @Qualifier 注解指定具体要注入的 Bean 的名称。
(二)用法示例
- 字段注入
- 在类的字段上使用 @Autowired 注解,实现字段的自动注入。例如:
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class MyService {@Autowiredprivate MyRepository myRepository;public void doSomething() {// 使用 myRepository}
}
- 构造函数注入
- 在构造函数上使用 @Autowired 注解,实现构造函数的自动注入。例如:
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class MyService {private final MyRepository myRepository;@Autowiredpublic MyService(MyRepository myRepository) {this.myRepository = myRepository;}public void doSomething() {// 使用 myRepository}
}
- 方法注入
- 在方法上使用 @Autowired 注解,实现方法的自动注入。例如:
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class MyService {private MyRepository myRepository;@Autowiredpublic void setMyRepository(MyRepository myRepository) {this.myRepository = myRepository;}public void doSomething() {// 使用 myRepository}
}
七、@Value 注解
(一)属性注入
- 从配置文件中获取值
- @Value 注解可以用于从配置文件(如 application.properties 或 application.yml)中获取属性值,并注入到字段、构造函数参数或方法参数中。
- 表达式支持
- 还可以使用 SpEL(Spring Expression Language)表达式来获取更复杂的值。例如:
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;@Service
public class MyService {@Value("${my.property}")private String myProperty;@Value("#{T(java.lang.Math).random() * 100}")private int randomNumber;public void doSomething() {// 使用 myProperty 和 randomNumber}
}
(二)使用场景
- 动态配置
- 在需要根据不同环境进行动态配置的情况下,@Value 注解非常有用。可以将配置文件中的属性值注入到 Bean 中,实现灵活的配置。
- 参数化构造函数
- 可以在构造函数参数上使用 @Value 注解,实现参数化的构造函数,根据配置文件中的值来创建 Bean。
八、@RequestMapping 注解
(一)处理 HTTP 请求
- 映射请求路径
- @RequestMapping 注解用于将一个方法映射到特定的 HTTP 请求路径上。它可以用在类级别和方法级别上,用于定义控制器类和方法的请求映射。
- 支持多种 HTTP 方法
- 可以通过 value 属性指定请求路径,通过 method 属性指定支持的 HTTP 方法(如 GET、POST、PUT、DELETE 等)。
(二)用法示例
- 类级别映射
- 在控制器类上使用 @RequestMapping 注解,定义类级别的请求映射。例如:
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;@RestController
@RequestMapping("/api")
public class MyController {// 方法级别的请求映射将在此基础上进行
}
- 方法级别映射
- 在控制器方法上使用 @RequestMapping 注解,定义具体的请求映射。例如:
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;@RestController
@RequestMapping("/api")
public class MyController {@RequestMapping(value = "/users", method = RequestMethod.GET)public List<User> getUsers() {// 返回用户列表}
}
九、@RestController 注解
(一)构建 RESTful API
- 标识控制器
- @RestController 注解是 @Controller 和 @ResponseBody 的组合注解。它用于标识一个控制器类,该类中的方法将返回 JSON、XML 或其他格式的响应数据,用于构建 RESTful API。
- 自动序列化响应数据
- 当一个方法被标注为 @RestController 注解的类中的方法返回一个对象时,Spring Boot 会自动将该对象序列化为 JSON 或其他格式的响应数据,并返回给客户端。
(二)与其他注解的配合
- 与 @RequestMapping 注解配合
- @RestController 注解通常与 @RequestMapping 注解一起使用,用于定义 RESTful API 的请求映射和响应数据。
- 与 @Service、@Repository 等注解配合
- 在构建复杂的应用程序时,@RestController 注解可以与其他注解(如 @Service、@Repository)配合使用,实现业务逻辑和数据访问的分离。
十、实际案例分析
(一)案例背景
假设有一个简单的博客应用,需要实现文章的管理功能,包括创建、读取、更新和删除文章。
(二)技术选型
- 使用 Spring Boot 构建应用
- 选择 Spring Boot 作为开发框架,利用其快速开发、自动配置等优势。
- 数据库选择
- 选择一个关系型数据库(如 MySQL)来存储文章数据。
- 前端框架选择
- 可以选择一个前端框架(如 Vue.js 或 React)来构建用户界面,与后端的 Spring Boot 应用进行交互。
(三)核心注解的应用
- @SpringBootApplication 注解
- 在主类上使用 @SpringBootApplication 注解,标识这是一个 Spring Boot 应用的主配置类。
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;@SpringBootApplication
public class BlogApplication {public static void main(String[] args) {SpringApplication.run(BlogApplication.class, args);}
}
- @Configuration 注解
- 创建一个配置类,使用 @Configuration 注解标识。在这个配置类中,可以定义 Bean 和进行其他配置。
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.datasource.DriverManagerDataSource;@Configuration
public class AppConfig {@Beanpublic JdbcTemplate jdbcTemplate() {DriverManagerDataSource dataSource = new DriverManagerDataSource();dataSource.setDriverClassName("com.mysql.cj.jdbc.Driver");dataSource.setUrl("jdbc:mysql://localhost:3306/blogdb");dataSource.setUsername("root");dataSource.setPassword("password");return new JdbcTemplate(dataSource);}
}
- @Autowired 注解
- 在服务类和控制器类中,使用 @Autowired 注解进行依赖注入。例如,在文章服务类中注入 JdbcTemplate:
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class ArticleService {private final JdbcTemplate jdbcTemplate;@Autowiredpublic ArticleService(JdbcTemplate jdbcTemplate) {this.jdbcTemplate = jdbcTemplate;}public void createArticle(Article article) {// 使用 jdbcTemplate 执行数据库插入操作}public Article getArticleById(int id) {// 使用 jdbcTemplate 执行数据库查询操作}// 其他方法
}
- @RequestMapping 注解和 @RestController 注解
- 在控制器类中,使用 @RequestMapping 注解和 @RestController 注解定义 RESTful API 的请求映射。例如:
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;@RestController
@RequestMapping("/articles")
public class ArticleController {private final ArticleService articleService;@Autowiredpublic ArticleController(ArticleService articleService) {this.articleService = articleService;}@RequestMapping(method = RequestMethod.POST)public void createArticle(Article article) {articleService.createArticle(article);}@RequestMapping(value = "/{id}", method = RequestMethod.GET)public Article getArticleById(@PathVariable int id) {return articleService.getArticleById(id);}// 其他方法
}
(四)效果评估
- 开发效率
- 通过使用 Spring Boot 的核心注解,大大提高了开发效率。自动配置和依赖注入功能减少了手动配置的工作量,使得开发过程更加快捷。
- 代码可读性和可维护性
- 注解的使用使得代码更加清晰易读,易于理解和维护。例如,通过 @RequestMapping 注解和 @RestController 注解,可以很容易地看出控制器类的作用和请求映射关系。
- 功能实现
- 成功实现了博客应用的文章管理功能,包括创建、读取、更新和删除文章。通过 RESTful API 的方式,使得前端可以方便地与后端进行交互。
十一、总结
Spring Boot 的核心注解在开发过程中起着至关重要的作用。@SpringBootApplication、@Configuration、@EnableAutoConfiguration、@ComponentScan、@Autowired、@Value、@RequestMapping 和 @RestController 等注解分别在应用的配置、自动配置、组件扫描、依赖注入、属性注入、请求映射和构建 RESTful API 等方面发挥着重要作用。通过深入理解这些注解的功能和用法,并结合实际案例进行应用,可以提高开发效率,增强代码的可读性和可维护性,实现高效的 Java 开发。
相关文章:
Spring Boot 的核心注解
一、引言 Spring Boot 作为一种流行的 Java 开发框架,以其简洁高效的开发方式受到广泛关注。其中,核心注解在 Spring Boot 应用的开发中起着至关重要的作用。理解这些注解的含义和用法,对于充分发挥 Spring Boot 的优势至关重要。本文将深入剖…...
蓝桥杯备考——算法
一、排序 冒泡排序、选择排序、插入排序、 快速排序、归并排序、桶排序 二、枚举 三、二分查找与二分答案 四、搜索(DFS) DFS(DFS基础、回溯、剪枝、记忆化) 1.DFS算法(深度优先搜索算法) 深度优先搜…...
MutationObserver与IntersectionObserver的区别
今天主要是分享一下MutationObserver和IntersectionObserver的区别,希望对大家有帮助! MutationObserver 和 IntersectionObserver 的区别 MutationObserver 作用:用于监听 DOM 树的变动,包括:元素的属性、子元素列表或节点文本的…...
生产与配置
1.鲁滨孙克苏鲁经济 鲁滨孙克苏鲁经济是一种非常简单的自给自足的经济,劳动时间与休息时间总和为总的时间。 即 摘椰子的数量为劳动时间的函数 由于鲁滨孙喜欢椰子,厌恶劳动时间,因此无差异曲线表现为厌恶品的形态。 根据无差异曲线和生…...
Android Kotlin Flow 冷流 热流
在 Android 开发中,Flow 是 Kotlin 协程库的一部分,用于处理异步数据流的一个组件。本质上,Flow 是一个能够异步生产多个值的数据流,与 suspend 函数返回单个值的模式相对应。Flow 更类似于 RxJava 中的 Observable,但…...
订单日记助力“实峰科技”提升业务效率
感谢北京实峰科技有限公司选择使用订单日记! 北京实峰科技有限公司,成立于2022年,位于北京市石景区,是一家以从事生产、销售微特电机、输配电及控制设备等业务为主的企业。 在业务不断壮大的过程中,想使用一种既能提…...
如何安装和配置JDK17
教程目录 零、引言1、新特性概览2、性能优化3、安全性增强4、其他改进5、总结 一、下载安装二、环境配置三、测试验证 零、引言 JDK 17(Java Development Kit 17)是Java平台的一个重要版本,它带来了许多新特性和改进,进一步提升了…...
智能化温室大棚控制系统设计(论文+源码)
1 系统的功能及方案设计 本次智能化温室大棚控制系统的设计其系统整体结构如图2.1所示,整个系统在器件上包括了主控制器STC89C52,温湿度传感器DHT11,LCD1602液晶,继电器,CO2传感器,光敏电阻,按…...
面试题之---解释一下原型和原型链
实例化对象 和普调函数一样,只不过调用的时候要和new连用(实例化),不然就是一个普通函数调用 function Person () {} const o1 new Person() //能得到一个空对象 const o2 Person() //什么也得不到,这就是普通的…...
【Leecode】Leecode刷题之路第46天之全排列
题目出处 46-全排列-题目出处 题目描述 个人解法 思路: todo代码示例:(Java) todo复杂度分析 todo官方解法 46-全排列-官方解法 预备知识 回溯法:一种通过探索所有可能的候选解来找出所有的解的算法。如果候选解…...
自动驾驶革命:从特斯拉到百度,谁将主宰未来交通?
内容概要 自动驾驶技术正在经历一个前所未有的革命性变化,各大企业纷纷抢占这一充满潜力的新市场。以特斯拉和百度为代表的行业巨头,正利用各自的优势在这一技术的赛道上展开激烈竞争。特斯拉凭借其在电动汽车和自动驾驶领域的前瞻性设计与不断革新的技…...
Python __str__()方法
在Python中,str() 方法是一个特殊的方法(也称为魔术方法或双下方法),它定义了当对象需要被转换为字符串表示时应该如何做。 当你尝试打印对象(使用 print() 函数)或将对象插入到需要字符串表示的上下文中&…...
虚拟机的安装
添加映像文件 自动或者手动分配磁盘 添加密码 创建用户 创建快照...
HCIP快速生成树 RSTP
STP(Spanning Tree Protocol,生成树协议)和RSTP(Rapid Spanning Tree Protocol,快速生成树协议)都是用于在局域网中消除环路的网络协议。 STP(生成树协议) 基本概念: ST…...
Python基础学习-05元组 tuple
目录 1、元组的定义 2、元组的切片和索引 3、元组的函数 4、二维元组 5、本节总结 1、元组的定义 • 基本上可以理解为一个不可改变的列表 • 元组没有列表那么常用,但是它的关键是不可改变性 • 使用() 定义一个元组 1) T (1, 2, 3, 4, …...
vue3 基于element-plus进行的一个可拖动改变导航与内容区域大小的简单方法
1、先上个截图: 说明:拖动上面的分隔栏就可以实现,改变左右区域的大小。 2、上面的例子来自官网的: Container 布局容器 | Element Plus 3、拖动的效果来自: https://juejin.cn/post/7029640316999172104#heading-1…...
c++基础28函数的类型
函数的类型 基本用法例子usingfucntion 基本用法 在C中,函数类型是指函数的签名,包括返回类型、参数类型以及参数的数量。函数类型可以用来声明函数指针、函数引用或者作为模板参数。 函数也可当成一种数据类型 函数指针: 函数指针可以指向…...
Elasticsearch(四):query_string查询介绍
query_string查询介绍 1 概述2 基本概念3 数据准备4 query_string查询示例4.1 基本查询4.2 复杂查询解析4.3 高级过滤解析4.4 模糊查询解析4.5 高亮查询解析4.6 分页查询解析 5 总结 大家好,我是欧阳方超,可以我的公众号“欧阳方超”,后续内容…...
超好用shell脚本NuShell mac安装
利用管道控制任意系统 Nu 可以在 Linux、macOS 和 Windows 上运行。一次学习,处处可用。 一切皆数据 Nu 管道使用结构化数据,你可以用同样的方式安全地选择,过滤和排序。停止解析字符串,开始解决问题。 强大的插件系统 具备强…...
Vue禁止打开控制台/前端禁止打开控制台方法/禁用F12/禁用右键
代码片段展示了如何在前端页面中禁用右键菜单、禁止文本选择、阻止特定键盘操作(如F12键打开开发者工具),以及通过检测窗口尺寸变化来尝试阻止用户调试页面。 // 鼠标禁止右键禁止打开控制台及键盘禁用forbidden(){// 1.禁用右键菜单document…...
volatile关键字
1. 可见性 当一个变量被声明为 volatile 时,任何线程对该变量的写入操作都会立即对其他线程可见。这意味着: 当一个线程修改了 volatile 变量的值,其他线程在读取这个变量时会看到最新的值,而不是可能被缓存的旧值。 这解决了多线…...
[Linux] 共享内存
在Linux中,共享内存是一种允许不同进程之间直接交换数据的高效机制。它是IPC(Inter-Process Communication,进程间通信)的一种方式,允许多个进程通过映射同一块物理内存区域来实现数据共享,而无需使用内核来…...
网络的基础
学习地点(泷羽sec的个人空间-泷羽sec个人主页-哔哩哔哩视频 (bilibili.com)) HTTP协议介绍 HTTP,全称为超文本传输协议(HyperText Transfer Protocol),是用于万维网服务器向本地浏览器传输超文本ÿ…...
金融学期末速成笔记
【拯救者】金融学速成(基础习题) 重点: 市场经济是发达的商品经济。在市场经济条件下,市场机制作为资源配置方式,发挥基础性作用。 除具有商品经济的一般特征外,与商品经济相比,市场经济还具有一些新的特征…...
【Elasticsearch入门到落地】1、初识Elasticsearch
一、什么是Elasticsearch Elasticsearch(简称ES)是一款非常强大的开源搜索引擎,可以帮助我们从海量数据中快速找到需要的内容。它使用Java编写,基于Apache Lucene来构建索引和提供搜索功能,是一个分布式、可扩展、近实…...
电子版产品册代替纸质版产品册,开源节流!
在当今数字化时代,企业纷纷寻求创新手段以降低成本、提高效率。纸质版产品册作为传统宣传手段,虽然具有一定的宣传效果,但成本高昂、更新不便、环保压力等问题日益凸显。本文将为您详细解析如何通过采用电子版产品册替代纸质版产品册&#…...
npm i忽略依赖冲突
在使用npm安装依赖时,如果遇到依赖冲突,通常npm会提示错误并阻止安装。但是,如果你想要忽略这些依赖冲突,可以使用以下几种方法: 1.使用--force或-f参数:这个参数会强制npm忽略某些错误,包括依…...
商品,订单业务流程梳理一
业务架构梳理 业务系统介绍 业务商品流程 业务订单流程 业务售后流程 系统架构 技术栈...
Spring中的 bean 标签中的 factory-bean , factory-method
1.首先说说 factory-method 是指定创造实例的工厂方法,用法: factory-method 和 class 配合使用,这时 factory-method 必须是class所指定的类中的一个静态方法,也就是Spring会直接调用 class 所指定的类的静态工厂方法创建一个实例…...
车间管理|基于SprinBoot+vue工厂车间管理系统设计与实现(源码+数据库+文档)
车间管理系统系统 目录 基于SprinBootvue工厂车间管理系统设计与实现 一、前言 二、系统设计 三、系统功能设计 四、数据库设计 五、核心代码 六、论文参考 七、最新计算机毕设选题推荐 八、源码获取: 博主介绍:✌️大厂码农|毕设布道师&…...
手机网站开发 c/关键词歌曲
http://www.blogjava.net/Alpha/archive/2007/01/02.html转载于:https://blog.51cto.com/495798/1329292...
网站制作小常识/seo搜索引擎优化ppt
插上USB3.0的端口 在USB2.0的端口...
变态传奇手游/常用的seo工具
虽然Windows 10日臻完善和普及,可依然有很多用户在使用Windows 7,微软也在听取用户意见,前几天微软宣布DirectX 12下放到Windows 7,而微软也在准备Windows 7的新镜像。著名爆料WalkingCat在Twitter上爆料了微软新的Windows 7镜像的…...
网站免费观影怎么做/揭阳新站seo方案
系列文章目录 兼容Oracle与MySQL的那些事 兼容Oracle与MySQL的那些事(分页问题) 兼容Oracle与MySQL的一些事 兼容Oracle与MySQL的那些事【LISTAGG与GROUP_CONCAT】...
西安知名网站制作公司/360搜索首页网址是多少
习题1:一个列表,排重,不能用set,也不能用字典 算法: 1、生成一个空列表result,存储排重后的元素 2、遍历列表的每一个元素,判断是否在result列表里面,如果不在,就添加到result列表中 3、如果在,不做任何操作 4、打印result列表 l=[1,2,3,4,1,2,3,4,1,2] result=[] fo…...
wordpress首页访问慢/关于进一步优化
题目描述 经典编程。输入年号,判断并输出该年是否闰年。所谓闰年,是指能被4整除,但不能被100整除;或能被400整除的年份。 输入 输入年号 输出 是否为闰年,是输出yes,否则输出no 样例输入 2008 样例输出 …...