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

实战篇之基于二进制思想的用户标签系统(Mysql+SpringBoot)

        一: 计算机中的二进制

        计算机以二进制表示数据,以表示电路中的正反。在二进制下,一个位只有 0 和 1 。逢二进一 位。类似十进制下,一个位只有 0~9 。逢十进一位。

        二: 进制常用运算 (位运算)

  • 与运算(&):将两个二进制数的对应位进行与操作,只有当两个位都为1时,结果为1。
  • 或运算(|):将两个二进制数的对应位进行或操作,只要有一个位为1,结果就为1。
  • 非运算(~):对一个二进制数的每个位取反,将1变为0,将0变为1。
  • 异或运算(^):将两个二进制数的对应位进行异或操作,只有当两个位不同时,结果为1。
  • 左移运算(<<):将一个二进制数的所有位向左移动指定的位数,右边空出的位用0填充。
  • 右移运算(>>):将一个二进制数的所有位向右移动指定的位数,左边空出的位用原来的最高位填充。 这些二进制的运算在计算机的逻辑设计、编程和数据处理中经常使用。
  • 无符号右移( >>> , 无符号右移就是右移之后,无论该数为正还是为负,右移之后左边都是补上 0

        三: 标签记录的实现原理

        基于(或)|,与+取反(&~) 去实现:

        假设我们给用户添加的标签是一个数字 16 ,转换为二进制就是 10000
       1.设置标签  使用 或(|) (参加运算的两个位只要有一个为 1 ,其值为 1

        

        2.取消标签 与 + 取反(&~)(两位同时为 1,结果才为 1,否则为 0取消16这个标签

        四:一起动手实现用户标签系统 - 底层标签读写组件的实现

                4.1: 建立用户标签表SQL

CREATE TABLE `t_user_tag` (`user_id` bigint NOT NULL DEFAULT -1 COMMENT '用户 id',`tag_info_01` bigint NOT NULL DEFAULT '0' COMMENT '标签记录字段',`tag_info_02` bigint NOT NULL DEFAULT '0' COMMENT '标签记录字段',`tag_info_03` bigint NOT NULL DEFAULT '0' COMMENT '标签记录字段',`create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时
间',`update_time` datetime DEFAULT CURRENT_TIMESTAMP ON UPDATE 
CURRENT_TIMESTAMP COMMENT '更新时间',PRIMARY KEY (`user_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb3 COLLATE=utf8mb3_bin 
COMMENT='用户标签记录';

                4.2:service层接口

package com.laoyang.provider.service;import com.laoyang.constants.UserTagsEnum;/*** @author:Kevin* @create: 2023-08-01 09:53* @Description:*/public interface IUserTagService {/*** 设置标签 只能设置成功一次** @param userId* @param userTagsEnum* @return*/boolean setTag(Long userId, UserTagsEnum userTagsEnum);/*** 取消标签** @param userId* @param userTagsEnum* @return*/boolean cancelTag(Long userId, UserTagsEnum userTagsEnum);/*** 是否包含某个标签** @param userId* @param userTagsEnum* @return*/boolean containTag(Long userId,UserTagsEnum userTagsEnum);
}
                4.3: service实现接口
package com.laoyang.provider.service.impl;import com.laoyang.common.utils.ConvertBeanUtils;
import com.laoyang.constants.UserTagFieldNameConstants;
import com.laoyang.constants.UserTagsEnum;
import com.laoyang.dto.UserTagDTO;
import com.laoyang.framework.redis.key.UserProviderCacheKeyBuilder;
import com.laoyang.provider.dao.mapper.IUserTagMapper;
import com.laoyang.provider.dao.po.UserTagPO;
import com.laoyang.provider.service.IUserTagService;
import com.laoyang.usils.TagInfoUtils;
import jakarta.annotation.Resource;
import org.springframework.dao.DataAccessException;
import org.springframework.data.redis.connection.RedisConnection;
import org.springframework.data.redis.core.RedisCallback;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.RedisSerializer;import java.nio.charset.StandardCharsets;/*** @author:Kevin* @create: 2023-08-01 09:54* @Description:*/public class UserTagServiceImpl implements IUserTagService {@Resourceprivate IUserTagMapper userTagMapper;@Resourceprivate RedisTemplate<String, String> redisTemplate;private RedisTemplate<String, UserTagDTO> userTagDTORedisTemplate;@Resourceprivate UserProviderCacheKeyBuilder cacheKeyBuilder;@Overridepublic boolean setTag(Long userId, UserTagsEnum userTagsEnum) {boolean updateStatus = userTagMapper.setTag(userId,userTagsEnum.getFieldName(),userTagsEnum.getTag()) > 0;if (updateStatus){String redisKey = cacheKeyBuilder.buildtagInfoKey(userId);userTagDTORedisTemplate.delete(redisKey);return  true;}String key = cacheKeyBuilder.buildTagLockKey(userId);//TODO 分布式锁 实现多个线程之间对同一个资源的互斥访问,保证同一时间只有一个线程能够获取到锁并执行相应的操作String setNxResult = redisTemplate.execute(new RedisCallback<String>() {@Overridepublic String doInRedis(RedisConnection connection) throws DataAccessException {RedisSerializer keySerializer = redisTemplate.getKeySerializer();RedisSerializer valueSerializer = redisTemplate.getValueSerializer();String result = (String) connection.execute("set", keySerializer.serialize(key),valueSerializer.serialize("-1"),"NX".getBytes(StandardCharsets.UTF_8),"EX".getBytes(StandardCharsets.UTF_8),"3".getBytes(StandardCharsets.UTF_8));return result;}});if (!"OK".equals(setNxResult)){return false;}UserTagPO userTagPO = userTagMapper.selectById(userId);if (userTagPO!=null){return false;}userTagPO = new UserTagPO();userTagPO.setUserId(userId);userTagMapper.insert(userTagPO);updateStatus = userTagMapper.setTag(userId,userTagsEnum.getFieldName(),userTagsEnum.getTag()) > 0;redisTemplate.delete(key);return updateStatus;}@Overridepublic boolean cancelTag(Long userId, UserTagsEnum userTagsEnum) {boolean cancleStatus = userTagMapper.cancelTag(userId,userTagsEnum.getFieldName(),userTagsEnum.getTag()) > 0;if (cancleStatus){return false;}String redisKey = cacheKeyBuilder.buildtagInfoKey(userId);userTagDTORedisTemplate.delete(redisKey);return true;}@Overridepublic boolean containTag(Long userId, UserTagsEnum userTagsEnum) {UserTagDTO userTagDTO = this.queryByUserId(userId);if (userTagDTO == null) {return false;}String queryFieldName = userTagsEnum.getFieldName();if(UserTagFieldNameConstants.TAG_INFO_01.equals(queryFieldName)) {returnTagInfoUtils.isContain(userTagDTO.getTagInfo01(),userTagsEnum.getTag());} else if(UserTagFieldNameConstants.TAG_INFO_02.equals(queryFieldName)) {returnTagInfoUtils.isContain(userTagDTO.getTagInfo02(),userTagsEnum.getTag());} else if(UserTagFieldNameConstants.TAG_INFO_03.equals(queryFieldName)) {returnTagInfoUtils.isContain(userTagDTO.getTagInfo03(),userTagsEnum.getTag());}return false;}/*** 从redis查询用户标签* @param userId* @return*/private UserTagDTO queryByUserId(Long userId){String redisKey = cacheKeyBuilder.buildtagInfoKey(userId);UserTagDTO userTagDTO = userTagDTORedisTemplate.opsForValue().get(redisKey);if (userTagDTO != null){return userTagDTO;}UserTagPO userTagPO = userTagMapper.selectById(userId);if (userTagPO == null){return null;}userTagDTO = ConvertBeanUtils.convert(userTagPO,UserTagDTO.class);userTagDTORedisTemplate.opsForValue().set(redisKey, userTagDTO);return userTagDTO;}}

        说明:我们使用了redis作为缓存,mybatisplus, 并自行创建了redis业务主键生成工具类等等,会放在最后,先把核心代码呈现。这里说明下使用到了redis分布式实现

这段代码是使用RedisTemplate执行一个"set"命令,并设置了一些选项参数。下面对代码进行解释:

  1. 首先,通过redisTemplate.getKeySerializer()获取key的序列化器,通过redisTemplate.getValueSerializer()获取value的序列化器。
  2. 在RedisCallback的doInRedis方法中,通过RedisConnection的execute方法执行"set"命令。
  3. 参数中,keySerializer.serialize(key)将key序列化为字节数组,valueSerializer.serialize("-1")将value序列化为字节数组。
  4. "NX".getBytes(StandardCharsets.UTF_8)表示设置NX选项,即只有在key不存在时才进行set操作。
  5. "EX".getBytes(StandardCharsets.UTF_8)表示设置EX选项,即设置key的过期时间为3秒。
  6. "3".getBytes(StandardCharsets.UTF_8)表示设置key的过期时间为3秒。
  7. connection.execute方法返回的是一个Object类型的结果,需要将其转换为String类型并返回。 总体来说,这段代码的作用是在Redis中执行一个set命令,将key和value存储到Redis中,并设置了过期时间和NX选项,确保只有在key不存在时才进行set操作。

        

        当多个节点同时尝试执行set操作来设置同一个key时,只有一个节点能够成功设置,因为Redis中的set命令默认具有原子性。如果设置了NX选项,即只有在key不存在时才进行set操作,那么只有第一个节点能够成功设置该key,其他节点将无法设置。 通过利用这个特性,可以将某个共享资源对应的key作为锁的名称,多个节点试图通过set操作来竞争该锁。只有一个节点能够成功设置该锁的key,即获得了分布式锁。其他节点则在设置失败后,可以选择等待或者进行其他处理。 同时,为了避免因为某个节点获得锁后发生故障而导致锁一直无法释放,还可以为锁设置过期时间。当锁的持有者在一定时间后未能释放锁,锁将自动过期并被其他节点获取。 综上所述,通过使用Redis的set操作和一些选项参数,可以实现简单的分布式锁。多个节点可以通过竞争设置同一个key来获得锁,并通过设置过期时间来避免因为锁的持有者发生故障而导致锁一直无法释放。

                4.4: Mapper层
package com.laoyang.provider.dao.mapper;import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.laoyang.provider.dao.po.UserTagPO;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Update;/*** @author:Kevin* @create: 2023-08-01 09:54* @Description:*/
@Mapper
public interface IUserTagMapper extends BaseMapper<UserTagPO> {/*** 使用或的思路来设置标签,只能允许第一次设置成功* @param userId* @param fieldName* @param tag* @return*/@Update("update t_user_tag set ${fieldName}=${fieldName} | #{tag} where user_id=#{userId} and ${fieldName} & #{tag}=0")int setTag(Long userId, String fieldName, long tag);/*** 使用先取反在与的思路来取消标签,只能允许第一次删除成功* @param userId* @param fieldName* @param tag* @return*/@Update("update t_user_tag set ${fieldName}=${fieldName} &~ #{tag} where user_id=#{userId} and ${fieldName} & #{tag}=#{tag}")int cancelTag(Long userId, String fieldName, long tag);
}

        说明:这里的sql可以参考开头看到的实现原理

        4.5 工具类

        4.5.1:对象转换类

package com.laoyang.common.utils;import org.springframework.beans.BeanInstantiationException;
import org.springframework.beans.BeanUtils;import java.util.ArrayList;
import java.util.List;/*** @author:Kevin* @create: 2023-07-29 15:03* @Description:*/public class ConvertBeanUtils {/*** 将一个对象转成目标对象*/public static <T> T convert(Object source,Class<T> targetClass){if (source == null){return null;}T t = newInstance(targetClass);BeanUtils.copyProperties(source,t);return t;}/*** 将List对象转换成目标对象*/public static <K,T> List<T> convertList(List<K> sourceList, Class<T> targetClass){if (sourceList == null){return null;}List targetlist = new ArrayList((int) (sourceList.size() / 0.75) + 1);for (K source : sourceList) {targetlist.add(source);}return targetlist;}private static <T> T newInstance(Class<T> targetClass){try {return targetClass.newInstance();}catch (Exception e){throw new BeanInstantiationException(targetClass,"instantiation error",e);}}}

        调用实例:第一个参数:要转的对象  第二个参数:最终转换成的对象类

userTagDTO = ConvertBeanUtils.convert(userTagPO,UserTagDTO.class);

        4.5.2 redis业务封装key的工具类(继承实现)

                父类

package com.laoyang.framework.redis.key;import org.springframework.beans.factory.annotation.Value;/*** @author:Kevin* @create: 2023-07-30 16:37* @Description:*/public class RedisKeyBuilder {#获取到对应业务主题的名称@Value("${spring.application.name}")private String applicationName;private static final String SPLIT_ITEM = ":";public String getSplitItem() {return SPLIT_ITEM;}public String getRrefix(){return applicationName + SPLIT_ITEM;}
}

               子类

package com.laoyang.framework.redis.key;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Conditional;
import org.springframework.context.annotation.Configuration;/*** @author:Kevin* @create: 2023-07-30 16:41* @Description:    用户中台的redis的key的封装工具类,生成这个业务字段的key*/
@Configuration
@Conditional(RedisKeyLoadMatch.class)
public class UserProviderCacheKeyBuilder extends RedisKeyBuilder{private static String USER_INFO_KEY = "userInfo";private static String USER_TAG_LOCK_KEY = "userTagLock";private static String USER_TAG_KEY = "userTag";private static String USER_PHONE_LIST_KEY = "userPhoneList";private static String USER_PHONE_OBJ_KEY = "userPhoneObj";private static String USER_LOGIN_TOKEN_KEY = "userLoginToken";public String buildUserInfoKey(Long userId) {return super.getRrefix() + USER_INFO_KEY + super.getSplitItem() + userId;}public String buildTagLockKey(Long userId){return super.getRrefix() + USER_TAG_LOCK_KEY + super.getSplitItem() + userId;}public String buildtagInfoKey(Long userId){return super.getRrefix() + USER_TAG_KEY + super.getSplitItem() + userId;}public String buildUserPhoneListKey(Long userId) {return super.getRrefix() + USER_PHONE_LIST_KEY + super.getSplitItem() + userId;}public String buildUserPhoneObjKey(String phone) {return super.getRrefix() + USER_PHONE_OBJ_KEY + super.getSplitItem() + phone;}public String buildUserLoginTokenKey(String tokenKey) {return super.getRrefix() + USER_LOGIN_TOKEN_KEY + super.getSplitItem() + tokenKey;}}

                4.5.3 po类

package com.laoyang.provider.dao.po;import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;
import lombok.Data;import java.util.Date;/*** @author:Kevin* @create: 2023-08-01 09:56* @Description:*/
@Data
@TableName("t_user_tag")
public class UserTagPO {@TableId(type = IdType.INPUT)private Long userId;@TableField(value = "tag_info_01")private Long tagInfo01;@TableField(value = "tag_info_02")private Long tagInfo02;@TableField(value = "tag_info_03")private Long tagInfo03;private Date createTime;private Date updateTime;}

相关文章:

实战篇之基于二进制思想的用户标签系统(Mysql+SpringBoot)

一&#xff1a; 计算机中的二进制 计算机以二进制表示数据&#xff0c;以表示电路中的正反。在二进制下&#xff0c;一个位只有 0 和 1 。逢二进一 位。类似十进制下&#xff0c;一个位只有 0~9 。逢十进一位。 二&#xff1a; 进制常用运算 &#xff08;位运算&#xff09;…...

Ansible 进阶

Ansible 进阶 ⤴️Ansible 入门看这篇文章⤵️Ansible 实战看这篇文章 一.Ansible 中的 Playbook 1.1 Playbook 介绍 如下图&#xff0c;ansible 在整个管理过程中使用 playbook 的大体流程。 Playbook 中包含多个 role&#xff0c;每个 role 对应于在远程主机完成某个比较复…...

滴滴Ceph分布式存储系统优化之锁优化

摘自&#xff1a;https://mp.weixin.qq.com/s/oWujGOLLGItu1Bv5AuO0-A 2020-09-02 21:45 0.引言 Ceph是国际知名的开源分布式存储系统&#xff0c;在工业界和学术界都有着重要的影响。Ceph的架构和算法设计发表在国际系统领域顶级会议OSDI、SOSP、SC等上。Ceph社区得到Red Hat…...

flutter开发实战-MethodChannel实现flutter与iOS双向通信

flutter开发实战-MethodChannel实现flutter与iOS双向通信 最近开发中需要iOS与flutter实现通信&#xff0c;这里使用的MethodChannel 如果需要flutter与Android实现双向通信&#xff0c;请看 https://blog.csdn.net/gloryFlow/article/details/132218837 这部分与https://bl…...

华为、阿里巴巴、字节跳动 100+ Python 面试问题总结(七)

系列文章目录 个人简介&#xff1a;机电专业在读研究生&#xff0c;CSDN内容合伙人&#xff0c;博主个人首页 Python面试专栏&#xff1a;《Python面试》此专栏面向准备面试的2024届毕业生。欢迎阅读&#xff0c;一起进步&#xff01;&#x1f31f;&#x1f31f;&#x1f31f; …...

K8S系列一:概念入门

写在前面 本文组织方式&#xff1a; K8S的架构、作用和目的。需要首先对K8S整体有所了解。 K8S是什么&#xff1f; 为什么是K8S&#xff1f; K8S怎么做&#xff1f; K8S的重要概念&#xff0c;即K8S的API对象。要学习和使用K8S必须知道和掌握的几个对象。 Pod 实例 Volume 数…...

QT- QLineEdite设置自动补全功能,并修改自动补全的样式

#include <QApplication> #include <QLineEdit> #include <QCompleter> #include <QStringListModel> #include <QMainWindow>int main(int argc, char *argv[]) {QApplication app(argc, argv);// 创建一个字符串列表模型QStringListModel mode…...

解决Adobe Flash Player已被屏蔽

问题&#xff1a;该插件不支持 原因&#xff1a;现在浏览器默认禁用flash 博主当前使用的是谷歌浏览器Chrome 2个主要方法都已经失效 搜索一圈后&#xff0c;之前博客给出的2个主要方法都已经失效。 1、flash.cn 下载本地播放器 2、在chrome中打开flash的禁用开关 2023年解…...

【Spring专题】Spring之Bean的生命周期源码解析——阶段二(IOC之实例化)

目录 前言阅读准备阅读指引阅读建议 课程内容一、SpringIOC之实例化1.1 简单回顾1.2 概念回顾1.3 核心方法讲解 二、方法讲解2.1 AbstractBeanFactory#getMergedLocalBeanDefinition&#xff1a;合并BeanDefinition2.2 AbstractAutowireCapableBeanFactory#createBean&#xff…...

YOLOv8目标检测算法

YOLOv8目标检测算法相较于前几代YOLO系列算法具有如下的几点优势&#xff1a; 更友好的安装/运行方式速度更快、准确率更高新的backbone&#xff0c;将YOLOv5中的C3更换为C2FYOLO系列第一次尝试使用anchor-free新的损失函数 YOLOv8简介 YOLOv8 是 Ultralytics 公司继 YOLOv5…...

uniapp条件编译

uniapp条件编译 uni-app 将已经将可以跨平台兼容处理的组件及 API 等进行了封装&#xff0c;但是部分平台的特性无法跨平台。 由此&#xff0c;uni-app 提供了条件编译的方案&#xff0c;来处理不同平台的特定或差异。 写法 #ifdef&#xff1a; 仅在某平台存在#ifndef&#xf…...

2023年国赛数学建模思路 - 复盘:光照强度计算的优化模型

文章目录 0 赛题思路1 问题要求2 假设约定3 符号约定4 建立模型5 模型求解6 实现代码 建模资料 0 赛题思路 &#xff08;赛题出来以后第一时间在CSDN分享&#xff09; https://blog.csdn.net/dc_sinor?typeblog 1 问题要求 现在已知一个教室长为15米&#xff0c;宽为12米&…...

volte端到端问题分析(一)

1、MME专载保持功能验证 **描述&#xff1a;**当无线环境较差时&#xff0c;有可能由于“Radio_Connection_with_UE_Lost” 原因造成的VoLTE通话掉话&#xff0c;如果UE发生RRC重建成功&#xff0c;手机将不会掉话。 对MME1202进行功能验证&#xff1a;开启后&#xff0c;MME专…...

微信小程序(原生)搜索功能实现

一、效果图 二、代码 wxml <van-searchvalue"{{ keyword }}"shape"round"background"#000"placeholder"请输入关键词"use-action-slotbind:change"onChange"bind:search"onSearch"bind:clear"onClear&q…...

Android AOSP源码编译——AOSP整编(二)

切换到源码目录下执行下面命令 1、初始化环境 . build/envsetup.sh //清除缓存 make clobber2、选择编译目标 lunchAOSP 预制了很多 Product。这里为了简单我们先不用真机&#xff0c;而是选择模拟器的方式&#xff0c;对于 x86_64 模拟器&#xff0c;我们选择的是 aosp_x86…...

铁是地球科学争论的核心

一项新的研究调查了地球内部铁的形态。这些发现对理解内核的结构产生了影响。 一项新的研究探索了地球内核的铁结构&#xff0c;如图中的黄色和白色所示。 资料来源&#xff1a;地球物理研究快报 地球内核以铁为主&#xff0c;铁可以多种晶体形式作为固体材料存在。&#xff08…...

TX Text Control .NET Server for ASP.NET Crack

TX Text Control .NET Server for ASP.NET Crack TX Text Control.NET Server for ASP.NET是用于Web应用程序或服务的服务器端组件。它是一个完全可编程的ASP.NET文字处理引擎&#xff0c;提供了广泛的文字处理功能。使用TX Text Control.NET Server&#xff0c;程序员可以开发…...

工作纪实36-ES跨集群迁移

1.es数据备份、恢复 https://blog.csdn.net/andy_only/article/details/111319175 2.reindex命令 https://codeleading.com/article/40964498185/ 添加配置、重启ES cd bin sh elasticsearch -d3.开源工具 https://github.com/elasticsearch-dump/elasticsearch-dump 4.…...

【MFC】11.MFC文档和单文档架构-笔记

MFC文档 之前我们在写字符雨的时候&#xff0c;将数据都存储到了视图类中&#xff0c;这是不合理的&#xff0c;视图类只负责显示&#xff0c;不应该存储任何数据 文档&#xff1a;专门存储数据用的 CDocument 文档与视图的关系&#xff1a; 创建一个文档类 单文档四个类都…...

2023年“研究生科研素养提升”系列公益讲座在线测评题目与参考答案

一、单选题 1、关于参考文献的选择&#xff0c;说法错误的是 &#xff08; &#xff09; 参考文献的选择有原创性、必要性的原则 不能过度引用 不能故意隐藏引用来源 可以引用无关参考文献 您的答案&#xff1a;D 参考答案&#xff1a;D 答案解析&#xff1a;不可以引用无关参考…...

深入浅出Asp.Net Core MVC应用开发系列-AspNetCore中的日志记录

ASP.NET Core 是一个跨平台的开源框架&#xff0c;用于在 Windows、macOS 或 Linux 上生成基于云的新式 Web 应用。 ASP.NET Core 中的日志记录 .NET 通过 ILogger API 支持高性能结构化日志记录&#xff0c;以帮助监视应用程序行为和诊断问题。 可以通过配置不同的记录提供程…...

AtCoder 第409​场初级竞赛 A~E题解

A Conflict 【题目链接】 原题链接&#xff1a;A - Conflict 【考点】 枚举 【题目大意】 找到是否有两人都想要的物品。 【解析】 遍历两端字符串&#xff0c;只有在同时为 o 时输出 Yes 并结束程序&#xff0c;否则输出 No。 【难度】 GESP三级 【代码参考】 #i…...

JUC笔记(上)-复习 涉及死锁 volatile synchronized CAS 原子操作

一、上下文切换 即使单核CPU也可以进行多线程执行代码&#xff0c;CPU会给每个线程分配CPU时间片来实现这个机制。时间片非常短&#xff0c;所以CPU会不断地切换线程执行&#xff0c;从而让我们感觉多个线程是同时执行的。时间片一般是十几毫秒(ms)。通过时间片分配算法执行。…...

vue3+vite项目中使用.env文件环境变量方法

vue3vite项目中使用.env文件环境变量方法 .env文件作用命名规则常用的配置项示例使用方法注意事项在vite.config.js文件中读取环境变量方法 .env文件作用 .env 文件用于定义环境变量&#xff0c;这些变量可以在项目中通过 import.meta.env 进行访问。Vite 会自动加载这些环境变…...

Redis数据倾斜问题解决

Redis 数据倾斜问题解析与解决方案 什么是 Redis 数据倾斜 Redis 数据倾斜指的是在 Redis 集群中&#xff0c;部分节点存储的数据量或访问量远高于其他节点&#xff0c;导致这些节点负载过高&#xff0c;影响整体性能。 数据倾斜的主要表现 部分节点内存使用率远高于其他节…...

Maven 概述、安装、配置、仓库、私服详解

目录 1、Maven 概述 1.1 Maven 的定义 1.2 Maven 解决的问题 1.3 Maven 的核心特性与优势 2、Maven 安装 2.1 下载 Maven 2.2 安装配置 Maven 2.3 测试安装 2.4 修改 Maven 本地仓库的默认路径 3、Maven 配置 3.1 配置本地仓库 3.2 配置 JDK 3.3 IDEA 配置本地 Ma…...

Python Ovito统计金刚石结构数量

大家好,我是小马老师。 本文介绍python ovito方法统计金刚石结构的方法。 Ovito Identify diamond structure命令可以识别和统计金刚石结构,但是无法直接输出结构的变化情况。 本文使用python调用ovito包的方法,可以持续统计各步的金刚石结构,具体代码如下: from ovito…...

Vite中定义@软链接

在webpack中可以直接通过符号表示src路径&#xff0c;但是vite中默认不可以。 如何实现&#xff1a; vite中提供了resolve.alias&#xff1a;通过别名在指向一个具体的路径 在vite.config.js中 import { join } from pathexport default defineConfig({plugins: [vue()],//…...

Web后端基础(基础知识)

BS架构&#xff1a;Browser/Server&#xff0c;浏览器/服务器架构模式。客户端只需要浏览器&#xff0c;应用程序的逻辑和数据都存储在服务端。 优点&#xff1a;维护方便缺点&#xff1a;体验一般 CS架构&#xff1a;Client/Server&#xff0c;客户端/服务器架构模式。需要单独…...

OD 算法题 B卷【正整数到Excel编号之间的转换】

文章目录 正整数到Excel编号之间的转换 正整数到Excel编号之间的转换 excel的列编号是这样的&#xff1a;a b c … z aa ab ac… az ba bb bc…yz za zb zc …zz aaa aab aac…; 分别代表以下的编号1 2 3 … 26 27 28 29… 52 53 54 55… 676 677 678 679 … 702 703 704 705;…...