网站建设平面要多少分辨率/国内做seo最好的公司

网站建设平面要多少分辨率,国内做seo最好的公司,javaweb网站首页怎么做,机械制造网站Spring Data作为Spring全家桶中重要的一员,在Spring项目全球使用市场份额排名中多次居前位,而在Spring Data子项目的使用份额排名中,Spring Data JPA也一直名列前茅。Spring Boot为Spring Data JPA提供了启动器,使Spring Data JPA…

Spring Data作为Spring全家桶中重要的一员,在Spring项目全球使用市场份额排名中多次居前位,而在Spring Data子项目的使用份额排名中,Spring Data JPA也一直名列前茅。Spring Boot为Spring Data JPA提供了启动器,使Spring Data JPA在Spring Boot项目中的使用更加便利。

Spring Data JPA概述

对象关系映射(Object Relational Mapping,ORM)框架在运行时可以参照映射文件的信息,把对象持久化到数据库中,可以解决面向对象与关系数据库存在的互不匹配的现象,常见的ORM框架有Hibernate、OpenJPA等。ORM框架的出现,使开发者从数据库编程中解脱出来,把更多的精力放在业务模型与业务逻辑上,但各ORM框架之间的API差别很大,使用了某种ORM框架的系统会严重受限于该ORM的标准,基于此,SUN公司提出JPA(Java Persistence API,Java持久化API)。

JPA是Sun官方提出的Java持久化规范,用于描述对象和关系表的映射关系,并将运行期的实体对象持久化到数据库中。JPA规范本质上是一套规范,它提供了一些编程的API接口,但具体实现则由服务厂商来提供基于JPA的数据访问。

Spring Data JPA是Spring基于ORM框架、JPA规范的基础上封装的一套JPA应用框架,它提供了增删改查等常用功能,使开发者可以用较少的代码实现数据操作,同时还易于扩展。

基于JPA的数据访问如下:

Spring Data JPA整体处理逻辑如下: 

 

Spring Data JPA快速入门

Spring Data JPA提供了很多模板代码,易于扩展,可以大幅提高开发效率,使开发者用极简的代码即可实现对数据的访问。使用Spring Data JPA可以通过Repository接口中的方法对数据库中的数据进行增删改查,也可以根据方法命名规则定义的方法、JPQL,以及原生SQL的方式进行操作。

Repository接口

自定义Repository接口,必须继承XXRepository<T, ID>接口,T:实体类,ID:实体类中主键对应的属性的数据类型。

Repository继承关系如下:

CrudRepository

在Spring Data JPA中,CrudRepository 是一个接口,它提供了基本的 CRUD(创建、读取、更新、删除)操作。虽然 CrudRepository 并没有直接提供一个专门用于更新的方法,但你可以通过 save() 方法来实现更新的功能。

当你调用 save() 方法并传递一个实体对象时,Spring Data JPA 会检查该实体是否已经存在于数据库中(通常是通过主键来判断,所以实体对象设置了主键值时会出现一条查询语句)。

  • 如果实体不存在,save() 方法会创建一个新的记录。
  • 如果实体已经存在,save() 方法会更新该实体的状态。 

JpaRepository接口提供的方法

使用Spring Data JPA进行数据操作的多种实现方式

1、如果自定义接口继承了JpaRepository接口,则默认包含了一些常用的CRUD方法。

2、自定义Repository接口中,可以使用@Query注解配合SQL语句进行数据的查、改、删操作。

3、自定义Repository接口中,可以直接使用关键字构成的方法名进行查询操作。

4、在自定义的Repository接口中,使用@Query注解方式执行数据变更操作(修改、删除)时,必须添加@Modifying注解表示数据变更和@Transactional注解表示事务管理

  • 在自定义的Repository接口中,针对数据的变更操作(修改、删除),无论是否使用了@Query注解,都必须在方法上方添加@Transactional注解进行事务管理,否则程序执行就会出现InvalidDataAccessApiUsageException异常。
  • 如果在调用Repository接口方法的业务层Service类上已经添加了@Transactional注解进行事务管理,那么Repository接口文件中就可以省略@Transactional注解。

5、使用Example实例进行复杂条件查询

根据方法命名规则定义方法

Spring Data中按照框架的规范自定义了Repository接口,除了可以使用接口提供的默认方法外,还可以按特定规则来定义查询方法,只要这些查询方法的方法名遵守特定的规则,不需要提供方法实现体,Spring Data就会自动为这些方法生成查询语句。Spring Data对这种特定的查询方法的定义规范如下:

        以find、read、get、query、count开头。

        涉及查询条件时,条件的属性使用条件关键字连接,并且条件属性的首字母大写。

        支持属性的级联查询:

                若当前类有符合条件的属性,则优先使用,而不使用级联属性。

                若需要使用级联属性,则属性之间使用_连接。

条件关键字如下:

关键字

方法名示例

对应的JPQL片段

And

findByLastnameAndFirstname()

… where x.lastname = ?1 and x.firstname = ?2

Or

findByLastnameOrFirstname()

… where x.lastname = ?1 or x.firstname = ?2

Is,Equals

findByFirstname,findByFirstnameIs,findByFirstnameEquals()

… where x.firstname = ?1

Between

findByStartDateBetween()

… where x.startDate between ?1 and ?2

关键字

方法名示例

对应的JPQL片段

LessThan

findByAgeLessThan()

… where x.age < ?1

LessThanEqual

findByAgeLessThanEqual()

… where x.age <= ?1

GreaterThan

findByAgeGreaterThan()

… where x.age > ?1

GreaterThanEqual

findByAgeGreaterThanEqual()

… where x.age >= ?1

After

findByStartDateAfter()

… where x.startDate > ?1

Before

findByStartDateBefore()

… where x.startDate < ?1

IsNull

findByAgeIsNull()

… where x.age is null

IsNotNull

findByAgeIsNotNull()

… where x.ageisnot null

NotNull

findByAgeNotNull()

… where x.age not null

关键字

方法名示例

对应的JPQL片段

Like

findByFirstnameLike()

… where x.firstname like ?1

NotLike

findByFirstnameNotLike()

… where x.firstname not like ?1

StartingWith

findByFirstnameStartingWith()

… where x.firstname like ?1 (绑定参数 %)

EndingWith

findByFirstnameEndingWith()

… where x.firstname like ?1 (绑定参数 %)

Containing

findByFirstnameContaining()

… where x.firstname like ?1 (绑定参数 %)

OrderBy

findByAgeOrderByLastnameDesc()

… where x.age = ?1 order by x.lastname desc

关键字

方法名示例

对应的JPQL片段

Not

findByLastnameNot()

… where x.lastname <> ?1

In

findByAgeIn(Collection<Age> ages)

… where x.age in ?1

NotIn

findByAgeNotIn(Collection<Age> ages)

… where x.age not in ?1

True

findByActiveTrue()

… where x.active = true

False

findByActiveFalse()

… where x.active = false

IgnoreCase

findByFirstnameIgnoreCase()

… where UPPER(x.firstame) = UPPER(?1)

JPQL

使用Spring Data JPA提供的查询方法已经可以满足大部分应用场景的需求,但是有些业务需要更灵活的查询条件,这时就可以使用@Query注解,结合JPQL的方式来完成查询。 JPQL是JPA中定义的一种查询语言,此种语言旨在让开发者忽略数据库表和表中的字段,而关注实体类及实体类中的属性。 JPQL语句的写法和SQL语句的写法十分类似,但是要把查询的表名换成实体类名称,把表中的字段名换成实体类的属性名称。

JPQL支持命名参数和位置参数两种查询参数。

命名参数:在方法的参数列表中,使用@Param注解标注参数的名称,在@Query注解的查询语句中,使用“:参数名称” 匹配参数名称。

位置参数:在@Query注解的查询语句中,使用“?位置编号的数值” 匹配参数,查询语句中参数标注的编号需要和方法的参数列表中参数的顺序依次对应。

示例:

//命名参数绑定
@Query("from Book b where b.author=:author and b.name=:name") 
List<Book> findByCondition1(@Param("author") String author,@Param("name") String name); 
//位置参数绑定
@Query("from Book b where b.author=?1 and b.name=?2") 
List<Book> findByCondition2(String author, String name);

JPQL中使用like模糊查询、排序查询、分页查询子句时,其用法与SQL中的用法相同,区别在于JPQL处理的类的实例不同。

示例:

//like模糊查询 
@Query("from Book b where b.name like %:name%")List<Book> findByCondition3(@Param("name") String name); 
//排序查询@Query("from Book b where b.name like %:name% order by id desc")
List<Book> findByCondition4(@Param("name") String name); 
//分页查询 
@Query("from Book b where b.name like %:name%") 
Page<Book> findByCondition5(Pageable pageable, @Param("name") String name);

JPQL中除了可以使用字符串和基本数据类型的数据作为参数外,还可以使用集合和Bean作为参数,传入Bean进行查询时可以在JPQL中使用SpEL表达式接收变量。

示例:

//传入集合参数查询 
@Query("from Book b where b.id in :ids") 
List<Book> findByCondition6(@Param("ids") Collection<String> ids); 
//传入Bean进行查询(使用SPEL表达式) 
@Query("from Book b where b.author=:#{#Book.author} and " +" b.name=:#{#Book.name}") 
Book findByCondition7(@Param("Book") Book Book); 

原生SQL 

如果出现非常复杂的业务情况,导致JPQL和其他查询都无法实现对应的查询,需要自定义SQL进行查询时,可以在@Query注解中定义该SQL。@Query注解中定义的是原生SQL时,需要在注解使用nativeQuery=true指定执行的查询语句为原生SQL,否则会将其当作JPQL执行。

示例:

@Query(value="SELECT * FROM book WHERE id = :id",nativeQuery=true)
Book findByCondition8(@Param("id") Integer id);

小提示

使用@Query注解可以执行JPQL和原生SQL查询,但是@Query注解无法进行DML数据操纵语言,主要语句有INSERT、DELETE和UPDATE操作,如果需要更新数据库中的数据,需要在对应的方法上标注@Modifying注解,以通知Spring Data当前需要进行的是DML操作。需要注意的是JPQL只支持DELETE和UPDATE操作,不支持INSERT操作。 

整合Spring Data JPA

0.在全局配置文件中添加数据库配置

server.port=8088spring.datasource.url=jdbc:mysql://localhost:3306/springbootdata?serverTimezone=UTC&characterEncoding=utf-8
spring.datasource.username=root
spring.datasource.password=zptc1234
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driverspring.jpa.hibernate.ddl-auto=update
spring.jpa.show-sql=true

spring.jpa.hibernate.ddl-auto 是 Spring Boot 应用程序中与 JPA(Java Persistence API)和 Hibernate 相关的配置属性。Hibernate 是一个流行的 JPA 实现,用于将 Java 对象映射到关系数据库中的表。

ddl-auto 属性用于控制 Hibernate 在启动时如何自动处理数据库架构(DDL,即数据定义语言)。具体来说,它定义了 Hibernate 是否应该基于实体类自动创建、更新或验证数据库表结构。

以下是 ddl-auto 的几个常用值:

  1. create:Hibernate 会在启动时创建数据库表。如果表已经存在,它会被删除并重新创建。这是一个非常危险的设置,因为它会丢失所有现有数据。通常,这仅用于开发环境或测试数据库。
  2. create-drop:在 create 的基础上,当 Hibernate 的 SessionFactory 关闭时,它会删除所有创建的表。这同样适用于开发或测试环境。
  3. update:Hibernate 会根据实体类更新数据库表结构。它只会添加、修改或删除必要的列,以保持与实体类的同步。这是一个相对安全的设置,但仍然建议在生产环境中谨慎使用,因为自动模式可能会引入难以预料的问题。
  4. validate:Hibernate 验证数据库表结构是否与实体类匹配。如果不匹配,它会抛出异常。这不会创建或修改任何表,只用于验证。

注意:尽管 ddl-auto 在开发过程中可能非常方便,但在生产环境中使用它通常是不推荐的。在生产环境中,建议使用迁移工具(如 Flyway 或 Liquibase)来管理数据库架构的版本控制。这样可以确保架构更改的清晰、可预测和可审计。

1.在pom文件中添加Spring Data JPA依赖启动器

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency><groupId>com.mysql</groupId><artifactId>mysql-connector-j</artifactId><scope>runtime</scope>
</dependency>

2.新建ORM实体类

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;@Entity(name = "t_comment")
public class Discuss {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Integer id;private String content;private String author;@Column(name = "a_id")private Integer aId;//补上get、set、toString方法}

3.新建Repository接口,添加增删改查等方法

创建子包repository,及接口DiscussRepository

import org.springframework.data.domain.Pageable;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Modifying;
import org.springframework.data.jpa.repository.Query;
import org.springframework.transaction.annotation.Transactional;public interface DiscussRepository extends JpaRepository<Discuss, Integer>{//查询author非空的Discuss评论信息public List<Discuss> findByAuthorNotNull();//通过文章id分页查询出Discuss评论信息。JPQL@Query("SELECT c FROM t_comment c WHERE c.aId = ?1")public List<Discuss> getDiscussPaged(Integer aid,Pageable pageable);  //通过文章id分页查询出Discuss评论信息。原生sql@Query(value = "SELECT * FROM t_comment  WHERE  a_Id = ?1",nativeQuery = true)public List<Discuss> getDiscussPaged2(Integer aid,Pageable pageable);//使用命名参数:bb@Query("SELECT c FROM t_comment c WHERE c.aId = :bb")public List<Discuss> getDiscussPaged3(@Param(value = "bb") Integer aid,Pageable pageable);//对数据进行更新和删除操作@Transactional@Modifying@Query("UPDATE t_comment c SET c.author = ?1 WHERE c.id = ?2")public int updateDiscuss(String author,Integer id);  @Transactional@Modifying@Query("DELETE t_comment c WHERE c.id = ?1")public int deleteDiscuss(Integer id);
}

注:

  • getDiscussPaged2与getDiscussPaged()方法的参数和作用完全一样。
  • 区别是该方法上方的@Query注解将nativeQuery属性设置为了true,用来编写原生SQL语句。 

4.编写单元测试

import java.util.List;
import java.util.Optional;import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.data.domain.Example;
import org.springframework.data.domain.ExampleMatcher;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;@SpringBootTest
class JpaTests {@Autowiredprivate DiscussRepository repository;//使用JpaRepository内部方法@Testpublic void selectComment() {Optional<Discuss>optional = repository.findById(1);if (optional.isPresent()) {System.out.println(optional.get());}}//使用方法名关键字进行数据操作@Testpublic void selectCommentByKeys() {List<Discuss>list = repository.findByAuthorNotNull();for (Discuss discuss : list) {System.out.println(discuss);}}//使用@Query注解@Testpublic void selectCommentPaged() {Pageable page = PageRequest.of(0, 3);List<Discuss>list = repository.getDiscussPaged(1, page);list.forEach(t -> System.out.println(t)); }//使用Example封装参数,精确匹配查询条件@Testpublic void selectCommentByExample() {Discuss discuss=new Discuss();discuss.setAuthor("张三");Example<Discuss> example = Example.of(discuss);List<Discuss> list = repository.findAll(example);list.forEach(t -> System.out.println(t));}//使用ExampleMatcher模糊匹配查询条件@Testpublic void selectCommentByExampleMatcher() {Discuss discuss=new Discuss();discuss.setAuthor("张");ExampleMatcher matcher = ExampleMatcher.matching().withMatcher("author",ExampleMatcher.GenericPropertyMatcher.of(ExampleMatcher.StringMatcher.CONTAINING));Example<Discuss> example = Example.of(discuss, matcher);List<Discuss> list = repository.findAll(example);System.out.println(list);}//保存评论@Testpublic void saveDiscuss() {Discuss discuss=new Discuss();discuss.setContent("张某的评论xxxx");discuss.setAuthor("张某");Discuss newDiscuss = repository.save(discuss);System.out.println(newDiscuss);}//更新作者@Testpublic void updateDiscuss() {int i = repository.updateDiscuss("更新者", 1);System.out.println("discuss:update:"+i);}//删除评论@Testpublic void deleteDiscuss() {int i = repository.deleteDiscuss(7);System.out.println("discuss:delete:"+i);}}

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

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

相关文章

Oracle归档配置及检查

配置归档位置到 USE_DB_RECOVERY_FILE_DEST&#xff0c;并设置存储大小 startup mount; !mkdir /db/archivelog ALTER SYSTEM SET db_recovery_file_dest_size100G SCOPEBOTH; ALTER SYSTEM SET db_recovery_file_dest/db/archivelog SCOPEBOTH; ALTER SYSTEM SET log_archive…

Four.meme是什么,一篇文章读懂

一、什么是Four.meme&#xff1f; Four.meme 是一个运行在 BNB 链的去中心化平台旨在为 meme 代币供公平启动服务。它允许用户以极低的成本创建和推出 meme 代币&#xff0c;无需预售或团队分配&#xff0c;它消除了传统的预售、种子轮和团队分配&#xff0c;确保所有参与者有…

Java 集合 List、Set、Map 区别与应用

一、核心特性对比 二、底层实现与典型差异 ‌List‌ ‌ArrayList‌&#xff1a;动态数组结构&#xff0c;随机访问快&#xff08;O(1)&#xff09;&#xff0c;中间插入/删除效率低&#xff08;O(n)&#xff09;‌‌LinkedList‌&#xff1a;双向链表结构&#xff0c;头尾操作…

欢迎来到未来:探索 Dify 开源大语言模型应用开发平台

欢迎来到未来&#xff1a;探索 Dify 开源大语言模型应用开发平台 如果你对 AI 世界有所耳闻&#xff0c;那么你一定听说过大语言模型&#xff08;LLM&#xff09;。这些智能巨兽能够生成文本、回答问题、甚至编写代码&#xff01;但是&#xff0c;如何将它们变成真正的实用工具…

python多线程和多进程的区别有哪些

python多线程和多进程的区别有七种&#xff1a; 1、多线程可以共享全局变量&#xff0c;多进程不能。 2、多线程中&#xff0c;所有子线程的进程号相同&#xff1b;多进程中&#xff0c;不同的子进程进程号不同。 3、线程共享内存空间&#xff1b;进程的内存是独立的。 4、同一…

【MySQL报错】:Column count doesn’t match value count at row 1

MySQL报错&#xff1a;Column count doesn’t match value count at row 1 意思是存储的数据与数据库表的字段类型定义不相匹配. 由于类似 insert 语句中&#xff0c;前后列数不等造成的 主要有3个易错点&#xff1a; 要传入表中的字段数和values后面的值的个数不相等。 由于类…

PostgreSQL 连接数超限问题

目录标题 **PostgreSQL 连接数超限问题解决方案****一、错误原因分析****二、查看连接数与配置****三、排查连接泄漏&#xff08;应用侧问题&#xff09;****四、服务侧配置调整****1. 调整最大连接数****2. 释放无效连接&#xff08;谨慎操作&#xff09;****3. 使用连接池工具…

2025最新-智慧小区物业管理系统

目录 1. 项目概述 2. 技术栈 3. 功能模块 3.1 管理员端 3.1.1 核心业务处理模块 3.1.2 基础信息模块 3.1.3 数据统计分析模块 3.2 业主端 5. 系统架构 5.1 前端架构 5.2 后端架构 5.3 数据交互流程 6. 部署说明 6.1 环境要求 6.2 部署步骤 7. 使用说明 7.1 管…

智能汽车图像及视频处理方案,支持视频智能包装能力

美摄科技的智能汽车图像及视频处理方案&#xff0c;通过深度学习算法与先进的色彩管理技术&#xff0c;能够自动调整图像中的亮度、对比度、饱和度等关键参数&#xff0c;确保在各种光线条件下&#xff0c;图像都能呈现出最接近人眼的自然色彩与细节层次。这不仅提升了驾驶者的…

React - LineChart组件编写(用于查看每日流水图表)

一、简单版本 LineChart.tsx // src/component/LineChart/LineChart.tsx import React, {useEffect,useRef,useImperativeHandle,forwardRef,useMemo,useCallback, } from react; import * as echarts from echarts/core; import type { ComposeOption } from echarts/core; …

Web前端考核 JavaScript知识点详解

一、JavaScript 基础语法 1.1 变量声明 关键字作用域提升重复声明暂时性死区var函数级✅✅❌let块级❌❌✅const块级❌❌✅ 1.1.1变量提升的例子 在 JavaScript 中&#xff0c;var 声明的变量会存在变量提升的现象&#xff0c;而 let 和 const 则不会。变量提升是指变量的声…

使用 Go 构建 MCP Server

一个互联网技术玩家&#xff0c;一个爱聊技术的家伙。在工作和学习中不断思考&#xff0c;把这些思考总结出来&#xff0c;并分享&#xff0c;和大家一起交流进步。 一、MCP 介绍 1. 基本介绍 MCP&#xff08;Model Context Protocol&#xff0c;模型上下文协议&#xff09;是…

CES Asia 2025赛逸展:科技浪潮中的创新与商贸盛会

在科技发展日新月异的当下&#xff0c;CES Asia 2025第七届亚洲消费电子技术贸易展&#xff08;赛逸展&#xff09;正积极筹备&#xff0c;将在北京举办&#xff0c;有望成为亚洲消费电子领域极具影响力的年度盛会。作为亚洲科技领域的重要展会&#xff0c;此次得到了数十家电子…

Windows桌面采集技术

在进入具体的方式讨论前&#xff0c;我们先看看 Windows 桌面图形界面的简化架构&#xff0c;如下图&#xff1a; 在 Windows Vista 之前&#xff0c;Windows 界面的复合画面经由 Graphics Device Interface&#xff08;以下简称 GDI&#xff09;技术直接渲染到桌面上。 在 Wi…

ElementPlus 快速入门

目录 前言 为什么要学习 ElementPlus&#xff1f; 正文 步骤 1 创建 一个工程化的vue 项目 ​2 安装 element-Plus :Form 表单 | Element Plus 1 点击 当前界面的指南 2 点击左边菜单栏上的安装&#xff0c;选择包管理器 3 运行该命令 demo(案例1 &#xff09; 步骤 …

[蓝桥杯 2023 省 A] 异或和之和

题目来自洛谷网站&#xff1a; 暴力思路&#xff1a; 先进性预处理&#xff0c;找到每个点位置的前缀异或和&#xff0c;在枚举区间。 暴力代码&#xff1a; #include<bits/stdc.h> #define int long long using namespace std; const int N 1e520;int n; int arr[N…

python学习笔记--实现简单的爬虫(二)

任务&#xff1a;爬取B站上最爱欢迎的编程课程 网址&#xff1a;编程-哔哩哔哩_bilibili 打开网页的代码模块&#xff0c;如下图&#xff1a; 标题均位于class_"bili-video-card__info--tit"的h3标签中&#xff0c;下面通过代码来实现&#xff0c;需要说明的是URL中…

windows清除电脑开机密码,可保留原本的系统和资料,不重装系统

前言 很久的一台电脑没有使用了&#xff0c;开机密码忘了&#xff0c;进不去系统 方法 1.将一个闲置u盘设置成pe盘&#xff08;注意&#xff0c;这个操作会清空原来u盘的数据&#xff0c;需要在配置前将重要数据转移走&#xff0c;数据无价&#xff0c;别因为配置这个丢了重…

5.4 位运算专题:LeetCode 137. 只出现一次的数字 II

1. 题目链接 LeetCode 137. 只出现一次的数字 II 2. 题目描述 给定一个整数数组 nums&#xff0c;其中每个元素均出现 三次&#xff0c;除了一个元素只出现 一次。请找出这个只出现一次的元素。 要求&#xff1a; 时间复杂度为 O(n)&#xff0c;空间复杂度为 O(1)。 示例&a…

C语言:扫雷

在编程的世界里&#xff0c;扫雷游戏是一个经典的实践项目。它不仅能帮助我们巩固编程知识&#xff0c;还能锻炼逻辑思维和解决问题的能力。今天&#xff0c;就让我们一起用 C 语言来实现这个有趣的游戏&#xff0c;并且通过图文并茂的方式&#xff0c;让每一步都清晰易懂 1. 游…