轻松打卡:使用Spring Boot和Redis Bitmap构建高效签到系统【redis实战 四】

欢迎来到我的博客,代码的世界里,每一行都是一个故事


在这里插入图片描述

轻松打卡:使用Spring Boot和Redis Bitmap构建高效签到系统【redis实战 四】

    • 引言(redis实战)
    • 前言
    • 回顾bitmap
      • 基本概念
      • 核心特性
      • 使用场景
    • 为什么使用redis中的bitmap实现?
      • 1. 存储效率
      • 2. 性能
      • 3. 扩展性和可用性
      • 4. 位操作的优势
      • 5. 灵活性和简易性
      • 6. 成本效益
    • 构思与实现逻辑
      • 基础逻辑
      • Key的组成
      • 与MySQL的交互
      • 优势和考虑
    • 签到系统实现
      • maven依赖
      • service实现
      • controller实现
      • 前端页面简化版
      • 功能展现图

引言(redis实战)

Redis List:打造高效消息队列的秘密武器【redis实战 一】

Redis Streams在Spring Boot中的应用:构建可靠的消息队列解决方案【redis实战 二】

Spring Boot和Redis Geo实现附近的人【redis实战 三】

前言

在数字化时代,签到系统已成为许多应用的标准功能,它不仅帮助我们追踪参与度,还激励着用户的日常活动。但如何在保证响应速度和数据准确性的同时处理成千上万的用户请求呢?这正是Spring Boot遇上Redis Bitmap的舞台。通过这篇文章,我们将踏上一个既充满挑战又充满创新的旅程,一起探索如何利用这两个强大工具打造出一个既高效又可靠的签到系统。

回顾bitmap

Redis Bitmap是一种特殊的数据结构,它使用连续的内存来存储和操作位数据(bit data),每个位可以是0或1。这种结构特别适合于那些需要处理大量布尔值的场景,比如在线状态、签到系统等。Bitmap由于其高效的存储和计算能力,在处理大规模数据时特别有优势。下面是Redis Bitmap的一些基本概念和特性:

基本概念

  1. 位操作:Bitmap的核心是位操作。它允许你设置、获取和计数单个位或一系列位的值。
  2. 存储效率:在Bitmap中,每个布尔值只占用一个位,而不是使用一个完整的字节或更多的空间,这使得它在存储大量布尔值时非常节省空间。
  3. 操作精确性:你可以精确地操作Bitmap中的每一位,这为处理复杂的数据结构提供了可能。

核心特性

  1. SETBIT & GETBIT

    • SETBIT 用于在指定偏移量处设置位的值。
    • GETBIT 用于获取指定偏移量处的位的值。
    • 这两个命令使得单个位的读写操作变得简单快速。
  2. BITCOUNT

    • BITCOUNT 命令用于计数Bitmap中设置为1的位的数量。
    • 可以对整个Bitmap或其特定范围进行计数。
    • 这对于统计和分析大量数据非常有用。
  3. BITOP

    • BITOP 命令用于对两个或多个Bitmap执行位运算(AND、OR、XOR和NOT)。
    • 这可以用来组合多个Bitmaps,计算交集、并集等。
  4. BITFIELD

    • BITFIELD 命令用于对Bitmap执行更复杂的操作,如将位作为整数进行递增。
    • 它提供了更高级别的控制,使得Bitmap可以用于更复杂的场景。
  5. 性能

    • Bitmap操作通常非常快,能够处理每秒数百万次的读写。
    • 它们特别适合高并发环境,如大型网站的在线状态跟踪。
  6. 空间效率

    • Bitmap在存储大量布尔值时极为节省空间。
    • 它特别适合那些值大部分为0的场景,因为Redis内部会进行优化以减少存储需求。

使用场景

  • 在线状态跟踪:跟踪数以百万计用户的在线或活跃状态。
  • 签到系统:记录用户每日的签到情况。
  • 访问统计:统计网站或应用的每日访问量。
  • 特征标记:记录用户或事物的特定特征,如权限、喜好等。

总结来说,Redis的Bitmap提供了一种高效、灵活的方法来处理大量的位级数据。它在存储效率和性能方面都有显著优势,特别适合于那些需要处理大量布尔值的场景。

为什么使用redis中的bitmap实现?

使用Redis Bitmap相比于使用MySQL或其他关系型数据库实现签到系统有一系列的优势。这些优势不仅在于性能上的提升,还包括了存储效率、扩展性和特定场景下的操作便利性。下面详细说明这些优势:

1. 存储效率

  • Bitmap结构:在Redis中,Bitmap以非常紧凑的方式存储数据。每个签到只需要一个位,而在关系型数据库中,你可能需要一个完整的表行来记录同样的信息。这在用户量庞大时尤其节省空间。
  • 内存存储:Redis作为内存数据库,其读写速度远超基于磁盘的关系型数据库。

2. 性能

  • 快速读写:对于Bitmap,无论是设置还是读取位值,时间复杂度都是O(1),这意味着操作的速度极快,几乎不受数据规模影响。
  • 减少I/O操作:由于Redis是内存数据库,所有操作都在内存中完成,避免了磁盘I/O,这在高并发场景下尤为重要。

3. 扩展性和可用性

  • 水平扩展:Redis支持主从复制、持久化、分区等多种特性,使得它可以很好地扩展并保持高可用性。
  • 成熟的集群支持:Redis集群提供了一种易于扩展和具有容错能力的方式,可以在多个节点上平衡负载。

4. 位操作的优势

  • 内置位操作:Redis提供了丰富的位操作命令(如SETBITGETBITBITCOUNT等),使得处理像签到这样的布尔值数据更加方便和高效。
  • 批量操作和原子性:可以原子性地对多个位进行操作,而在关系型数据库中实现同样的效果可能需要复杂的事务管理。

5. 灵活性和简易性

  • 简单的数据模型:相比关系型数据库复杂的表结构和关系,Redis的数据模型简单直观,易于理解和使用。
  • 快速开发和部署:对于开发人员来说,使用Redis通常意味着更少的设置和配置,可以更快速地开发和部署应用。

6. 成本效益

  • 减少硬件需求:由于Redis的高效率,相同负载下可能需要更少的硬件资源。
  • 降低维护成本:Redis简单的架构和少量的依赖使得维护工作相对更简单。

构思与实现逻辑

在上面提到的实现中,签到系统的基础逻辑是使用Redis的Bitmap来跟踪用户的每日签到状态。这种实现方式利用了Bitmap的高效存储和快速位操作特性,适合处理大量用户和高频率的签到事件。以下是对这个实现的基础逻辑和关键组件的详细介绍:

基础逻辑

  1. 每日签到

    • 每当用户签到时,系统会在Bitmap中的特定位置设置一个位(将其设为1)。
    • 位置通常由日期决定,例如,可以使用年内的天数作为偏移量。
  2. 统计查询

    • 系统可以通过计算Bitmap中设置为1的位的数量来统计签到次数。
    • 可以计算总签到次数、指定日期范围内的签到次数等。

Key的组成

在实现中,关键的是如何构造用于Bitmap的key。一个好的key设计既能确保访问的效率,又能方便与其他系统(如MySQL中的用户信息)进行交互。

  1. 用户标识

    • 每个用户应有唯一标识,如用户ID。这个标识会成为key的一部分,确保每个用户的签到信息是隔离的。
  2. 时间信息

    • 为了便于管理和查询,通常会在key中包含时间信息,如年份。这样可以每年为用户创建新的Bitmap,也便于历史数据的归档和查询。
  3. 前缀或命名空间

    • 为了更好的组织和区分不同的数据,可以在key前加上前缀或命名空间,如signin:

示例Key: signin:userId:2023

与MySQL的交互

  1. 用户详细信息

    • 在MySQL中维护用户的详细信息,如姓名、邮箱、注册日期等。
    • 每个用户记录都有一个唯一的ID,这个ID与Redis中使用的用户ID相对应。
  2. 数据关联

    • 当需要获取用户的签到信息以及详细信息时,可以先从Redis获取签到数据,然后使用从Redis得到的用户ID去MySQL中查询详细信息。
    • 这样的关联查询使得我们既能利用Redis的高性能,又能保持复杂数据的完整性和丰富性。
  3. 数据一致性

    • 确保在用户信息更新时(如用户ID变更),Redis中相应的key也同步更新,以维护数据的一致性。

优势和考虑

  • 高效存储:使用Bitmap存储签到信息极大地减少了所需的存储空间。
  • 快速操作:位操作的时间复杂度为O(1),即使是在大量数据面前也能保持高性能。
  • 灵活扩展:通过适当的key设计,系统可以灵活地进行扩展,如按年归档数据。
  • 与关系型数据库交互:通过在Redis和MySQL之间建立良好的数据关联,系统可以同时利用两者的优势。

通过以上的基础逻辑和key组成策略,你的签到系统将能够高效、灵活地处理用户的签到数据,并且能够方便地与存储在MySQL中的用户详细信息进行交互。

签到系统实现

maven依赖

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId>
</dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId>
</dependency>

service实现

package fun.bo.service;import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;import java.time.DayOfWeek;
import java.time.LocalDate;
import java.time.temporal.TemporalAdjusters;
import java.util.BitSet;/*** @author xiaobo*/
@Service
public class SignInService {// 这里可以不写构造方法,直接用@RequiredArgsConstructor实现,可参考<a href="https://blog.csdn.net/Mrxiao_bo/article/details/135113649">一行注解,省却百行代码:深度解析@RequiredArgsConstructor的妙用</>private final RedisTemplate<String, byte[]> redisTemplate;public SignInService(RedisTemplate<String, byte[]> redisTemplate) {this.redisTemplate = redisTemplate;}// 签到public void signIn(long userId) {LocalDate today = LocalDate.now();int dayOfYear = today.getDayOfYear();// 使用用户ID和年份作为key,确保每年的数据是独立的String key = "sign_in:" + userId + ":" + today.getYear();redisTemplate.opsForValue().setBit(key, dayOfYear, true);}// 获取今天签到的人数public long countSignInToday() {LocalDate today = LocalDate.now();return countSignInOnDate(today);}// 获取本周签到的人数public long countSignInThisWeek() {LocalDate today = LocalDate.now();// 获取本周的开始和结束日期LocalDate startOfWeek = today.with(TemporalAdjusters.previousOrSame(DayOfWeek.MONDAY));LocalDate endOfWeek = today.with(TemporalAdjusters.nextOrSame(DayOfWeek.SUNDAY));return countSignInBetweenDates(startOfWeek, endOfWeek);}// 获取本月签到的人数public long countSignInThisMonth() {// 获取本月的开始和结束日期LocalDate startOfMonth = LocalDate.now().with(TemporalAdjusters.firstDayOfMonth());LocalDate endOfMonth = LocalDate.now().with(TemporalAdjusters.lastDayOfMonth());return countSignInBetweenDates(startOfMonth, endOfMonth);}// 获取特定日期签到的人数private long countSignInOnDate(LocalDate date) {int dayOfYear = date.getDayOfYear();String keyPattern = "sign_in:*:" + date.getYear();return redisTemplate.keys(keyPattern).stream().filter(key -> redisTemplate.opsForValue().getBit(key, dayOfYear)).count();}// 获取日期范围内签到的人数private long countSignInBetweenDates(LocalDate start, LocalDate end) {long count = 0;// 遍历日期范围for (LocalDate date = start; !date.isAfter(end); date = date.plusDays(1)) {count += countSignInOnDate(date);}return count;}// 获取用户当月签到次数public long countUserSignInThisMonth(long userId) {LocalDate startOfMonth = LocalDate.now().with(TemporalAdjusters.firstDayOfMonth());LocalDate endOfMonth = LocalDate.now().with(TemporalAdjusters.lastDayOfMonth());String key = "sign_in:" + userId + ":" + LocalDate.now().getYear();byte[] bitmap = redisTemplate.opsForValue().get(key);BitSet bitSet = BitSet.valueOf(bitmap);long count = 0;for (int day = startOfMonth.getDayOfYear(); day <= endOfMonth.getDayOfYear(); day++) {count += bitSet.get(day) ? 1 : 0;}return count;}
}

有坑注意:RedisTemplate<String, byte[]>,这里要留意,如果你的bean中序列化Value的时候用的非字节数组,可能会报错如下

2023-12-25 11:12:55.163 ERROR 56398 --- [nio-7004-exec-9] o.a.c.c.C.[.[.[/].[dispatcherServlet]    : Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processing failed; nested exception is org.springframework.data.redis.serializer.SerializationException: Could not read JSON: Illegal character ((CTRL-CHAR, code 0)): only regular white space (\r, \n, \t) is allowed between tokensat [Source: (byte[])"\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0001"; line: 1, column: 2]; nested exception is com.fasterxml.jackson.core.JsonParseException: Illegal character ((CTRL-CHAR, code 0)): only regular white space (\r, \n, \t) is allowed between tokensat [Source: (byte[])"\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0001"; line: 1, column: 2]] with root cause

controller实现

package fun.bo.controller;import fun.bo.service.SignInService;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;@RestController
public class SignInController {private final SignInService signInService;public SignInController(SignInService signInService) {this.signInService = signInService;}// ... 保留之前的signIn和count方法 ...@GetMapping("/signin/count/today")public String countSignInToday() {long count = signInService.countSignInToday();return "Total sign-ins today: " + count;}@GetMapping("/signin/count/week")public String countSignInThisWeek() {long count = signInService.countSignInThisWeek();return "Total sign-ins this week: " + count;}@GetMapping("/signin/count/month")public String countUserSignInThisMonth(@RequestParam long userId) {long count = signInService.countUserSignInThisMonth(userId);return "Total sign-ins this month for user " + userId + ": " + count;}@PostMapping("/signin")public String signIn(@RequestParam("userId") Long userId) {// 调用签到服务来处理签到signInService.signIn(userId);return "User " + userId + " signed in successfully.";}
}

前端页面简化版

<!DOCTYPE html>
<html>
<head><title>签到系统</title><meta charset="UTF-8"> <!-- 指定页面字符集为UTF-8 -->
</head>
<body>
<h1>欢迎来到签到系统</h1><div><h2>用户签到</h2>用户ID:<input type="number" id="userIdInput" placeholder="输入用户ID"><button onclick="signIn()">签到</button>
</div><div><h2>今日签到人数</h2><button onclick="countSignInToday()">查询</button><p id="todayCount"></p>
</div><div><h2>本周签到人数</h2><button onclick="countSignInThisWeek()">查询</button><p id="weekCount"></p>
</div><div><h2>本月用户签到次数</h2>用户ID:<input type="number" id="userId"><button onclick="countUserSignInThisMonth()">查询</button><p id="monthCount"></p>
</div><script>function signIn() {// 请替换为你的实际用户ID和API端点let userId = document.getElementById('userIdInput').value;if (!userId) {alert("请先输入用户ID!");return;}fetch('/signin?userId=' + userId, { method: 'POST' }).then(response => alert('签到成功!')).catch(error => console.error('Error:', error));}function countSignInToday() {fetch('/signin/count/today').then(response => response.text()).then(data => document.getElementById('todayCount').innerText = data).catch(error => console.error('Error:', error));}function countSignInThisWeek() {fetch('/signin/count/week').then(response => response.text()).then(data => document.getElementById('weekCount').innerText = data).catch(error => console.error('Error:', error));}function countUserSignInThisMonth() {let userId = document.getElementById('userId').value;fetch('/signin/count/month?userId=' + userId).then(response => response.text()).then(data => document.getElementById('monthCount').innerText = data).catch(error => console.error('Error:', error));}
</script>
</body>
</html>

功能展现图

在这里插入图片描述

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

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

相关文章

紫光展锐M6780丨超分辨率技术——画质重构还原经典

上一期&#xff0c;我们揭秘了让画质更加炫彩的AI-PQ技术。面对分辨率较低的老电影&#xff0c;光有高饱和度的色彩是不够的&#xff0c;如何能够提高视频影像的分辨率&#xff0c;使画质更加清晰&#xff0c;实现老片新看&#xff1f; 本期带大家揭晓紫光展锐首颗AI8K超高清智…

【分布式技术专题】「分布式技术架构」 探索Tomcat技术架构设计模式的奥秘(Server和Service组件原理分析)

探索Tomcat技术架构设计模式的奥秘 Tomcat系统架构分析Tomcat 整体结构Tomcat总体结构图以 Service 作为“婚姻”1) Service 接口方法列表 2) StandardService 的类结构图方法列表 3) StandardService. SetContainer4) StandardService. addConnector 以 Server 为“居”1) Ser…

etcd技术解析:构建高可用分布式系统的利器

1. 引言 随着云原生技术的兴起&#xff0c;分布式系统的构建变得愈发重要。etcd作为一个高可用的分布式键值存储系统&#xff0c;在这个领域发挥着至关重要的作用。本文将深入探讨etcd的技术细节&#xff0c;以及如何利用它构建高可用的分布式系统。 2. etcd简介 etcd是一个开…

JVM系列-9.性能调优

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是爱吃芝士的土豆倪&#xff0c;24届校招生Java选手&#xff0c;很高兴认识大家&#x1f4d5;系列专栏&#xff1a;Spring原理、JUC原理、Kafka原理、分布式技术原理、数据库技术、JVM原理&#x1f525;如果感觉博主的文…

大数据安全 | 期末复习(中)

文章目录 &#x1f4da;感知数据安全⭐️&#x1f407;传感器概述&#x1f407;传感器的静态特性&#x1f407;调制方式&#x1f407;换能攻击&#x1f407;现有防护策略 &#x1f4da;AI安全⭐️&#x1f407;智能语音系统——脆弱性&#x1f407;攻击手段&#x1f407;AI的两…

探索IOC和DI:解密Spring框架中的依赖注入魔法

IOC与DI的详细解析 IOC详解1 bean的声明2 组件扫描 DI详解 IOC详解 1 bean的声明 IOC控制反转&#xff0c;就是将对象的控制权交给Spring的IOC容器&#xff0c;由IOC容器创建及管理对象。IOC容器创建的对象称为bean对象。 要把某个对象交给IOC容器管理&#xff0c;需要在类上…

【深度学习】sdxl中的 text_encoder text_encoder_2 区别

镜像问题是&#xff1a;https://editor.csdn.net/md/?articleId135867689 代码仓库&#xff1a; https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0/tree/main 截图&#xff1a; 为什么有两个CLIP编码器 text_encoder 和 text_encoder_2 &#xff1f; 在…

那些年与指针的爱恨情仇(一)---- 指针本质及其相关性质用法

关注小庄 顿顿解馋 (≧∇≦) 引言&#xff1a; 小伙伴们在学习c语言过程中是否因为指针而困扰&#xff0c;指针简直就像是小说女主&#xff0c;它逃咱追&#xff0c;我们插翅难飞…本篇文章让博主为你打理打理指针这个傲娇鬼吧~ 本节我们将认识到指针本质&#xff0c;何为指针和…

RHCE项目:使用LNMP搭建私有云存储

目录 一、准备工作 1、关闭防火墙、安全软件 2、搭建LNMP环境 3、上传软件 4、设置nextcloud安装命令权限 二、数据库 1、设置数据库 2、重启数据库 三、配置nginx 四、安装nextcloud 五、内网穿透 1、创建内网映射 2、linux系统安装花生壳客户端 3、重新打开浏览…

林浩然与极限的“无穷”约会

林浩然与极限的“无穷”约会 Lin Haoran’s Encounter with the Mathematical “Infinity” 在数学王国里&#xff0c;有一位名叫林浩然的大侠&#xff0c;他的江湖就是高等数学的殿堂。而他要挑战的终极Boss&#xff0c;便是那个既神秘又顽皮的“极限”。 In the kingdom of …

C# .Net6搭建灵活的RestApi服务器

1、准备 C# .Net6后支持顶级语句&#xff0c;更简单的RestApi服务支持&#xff0c;可以快速搭建一个极为简洁的Web系统。推荐使用Visual Studio 2022&#xff0c;安装"ASP.NET 和Web开发"组件。 2、创建工程 关键步骤如下&#xff1a; 包添加了“Newtonsoft.Json”&…

【Git】项目管理笔记

文章目录 本地电脑初始化docker报错.gitignoregit loggit resetgit statusgit ls-filesgit rm -r -f --cached拉取仓库文件更新本地的项目报错处理! [rejected] master -> master (fetch first)gitgitee.com: Permission denied (publickey).error: remote origin already e…

BACnet转IEC104网关BE104

随着电力系统信息化建设和数字化转型的进程不断加速&#xff0c;对电力能源的智能化需求也日趋增强。健全稳定的智慧电力系统能够为工业生产、基础设施建设以及国防建设提供稳定的能源支持。在此背景下&#xff0c;高性能的工业电力数据传输解决方案——协议转换网关应运而生&a…

研发日记,Matlab/Simulink避坑指南(六)——字节分割Bug

文章目录 前言 背景介绍 问题描述 分析排查 解决方案 总结归纳 前言 见《研发日记&#xff0c;Matlab/Simulink避坑指南&#xff08;一&#xff09;——Data Store Memory模块执行时序Bug》 见《研发日记&#xff0c;Matlab/Simulink避坑指南(二)——非对称数据溢出Bug》…

Arduino开发实例-DRV8833电机驱动器控制直流电机

DRV8833电机驱动器控制直流电机 文章目录 DRV8833电机驱动器控制直流电机1、DRV8833电机驱动器介绍2、硬件接线图3、代码实现DRV8833 使用 MOSFET,而不是 BJT。 MOSFET 的压降几乎可以忽略不计,这意味着几乎所有来自电源的电压都会传递到电机。 这就是为什么 DRV8833 不仅比基…

寒假思维训练day15 牛客练习赛121

牛客练习赛ABCD题解&#xff0c;更新一个题解作为今天的任务收尾。 寒假思维训练day15 摘要&#xff1a; Part1&#xff1a;B题&#xff0c;B-You Brought Me A Gentle Breeze on the Field_牛客练习赛121 (nowcoder.com) Part2: C题&#xff0c;C-氧气少年的水滴 2_牛客练…

职言圈 | 小伙年薪95w,女朋友父母却爱搭不理,如今上岸国家电网,人不到,叔叔阿姨吃饭都不动筷子...

在职场中&#xff0c;每个人都经历着不同的起伏和挑战。有时候&#xff0c;我们会面临着职业生涯上的起伏&#xff0c;但正是这些经历让我们成长&#xff0c;让我们更加坚韧。 就像这位网友一样&#xff0c;他在过去的一年里经历了美团L8的职位和年薪95W&#xff0c;却面临着女…

Sentinel解密:SlotChain中的SLot大揭秘

欢迎来到我的博客&#xff0c;代码的世界里&#xff0c;每一行都是一个故事 Sentinel解密&#xff1a;SlotChain中的SLot大揭秘 前言SlotChain简介&#xff1a;Sentinel的第一道防线入场仪式&#xff1a;SlotChain中的初始化SlotSlotChain的执行流程&#xff1a;从规则解析到流…

数据可视化练习

文章目录 试题示例 试题示例 绘制下图所示的表格 根据下表的数据&#xff0c;将班级名称一列作为x轴的刻度标签&#xff0c;将男生和女生两列的数据作为刻度标签对应的数值&#xff0c;使用bar()函数绘制下图所示的柱形图。 方式一 import numpy as np import matplotlib.p…

Kotlin快速入门系列3

Kotlin条件、循环控制 IF条件控制 与Java类似&#xff0c;一个if语句可包含布尔表达式和一条或多条语句。 fun compare(a:Int,b:Int) : Int{//常规传统用法var max aif (b>a) max breturn max//使用elsevar mMax : Intif(a>b){mMax a}else{mMax b}return mMax//使…