实战篇之基于二进制思想的用户标签系统(Mysql+SpringBoot)
一: 计算机中的二进制
计算机以二进制表示数据,以表示电路中的正反。在二进制下,一个位只有 0 和 1 。逢二进一 位。类似十进制下,一个位只有 0~9 。逢十进一位。二: 进制常用运算 (位运算)
- 与运算(&):将两个二进制数的对应位进行与操作,只有当两个位都为1时,结果为1。
- 或运算(|):将两个二进制数的对应位进行或操作,只要有一个位为1,结果就为1。
- 非运算(~):对一个二进制数的每个位取反,将1变为0,将0变为1。
- 异或运算(^):将两个二进制数的对应位进行异或操作,只有当两个位不同时,结果为1。
- 左移运算(<<):将一个二进制数的所有位向左移动指定的位数,右边空出的位用0填充。
- 右移运算(>>):将一个二进制数的所有位向右移动指定的位数,左边空出的位用原来的最高位填充。 这些二进制的运算在计算机的逻辑设计、编程和数据处理中经常使用。
无符号右移( >>> , 无符号右移就是右移之后,无论该数为正还是为负,右移之后左边都是补上 0 )
三: 标签记录的实现原理
基于(或)|,与+取反(&~) 去实现:
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);
}
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"命令,并设置了一些选项参数。下面对代码进行解释:
- 首先,通过redisTemplate.getKeySerializer()获取key的序列化器,通过redisTemplate.getValueSerializer()获取value的序列化器。
- 在RedisCallback的doInRedis方法中,通过RedisConnection的execute方法执行"set"命令。
- 参数中,keySerializer.serialize(key)将key序列化为字节数组,valueSerializer.serialize("-1")将value序列化为字节数组。
- "NX".getBytes(StandardCharsets.UTF_8)表示设置NX选项,即只有在key不存在时才进行set操作。
- "EX".getBytes(StandardCharsets.UTF_8)表示设置EX选项,即设置key的过期时间为3秒。
- "3".getBytes(StandardCharsets.UTF_8)表示设置key的过期时间为3秒。
- 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来获得锁,并通过设置过期时间来避免因为锁的持有者发生故障而导致锁一直无法释放。
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)
一: 计算机中的二进制 计算机以二进制表示数据,以表示电路中的正反。在二进制下,一个位只有 0 和 1 。逢二进一 位。类似十进制下,一个位只有 0~9 。逢十进一位。 二: 进制常用运算 (位运算)…...
Ansible 进阶
Ansible 进阶 ⤴️Ansible 入门看这篇文章⤵️Ansible 实战看这篇文章 一.Ansible 中的 Playbook 1.1 Playbook 介绍 如下图,ansible 在整个管理过程中使用 playbook 的大体流程。 Playbook 中包含多个 role,每个 role 对应于在远程主机完成某个比较复…...
滴滴Ceph分布式存储系统优化之锁优化
摘自:https://mp.weixin.qq.com/s/oWujGOLLGItu1Bv5AuO0-A 2020-09-02 21:45 0.引言 Ceph是国际知名的开源分布式存储系统,在工业界和学术界都有着重要的影响。Ceph的架构和算法设计发表在国际系统领域顶级会议OSDI、SOSP、SC等上。Ceph社区得到Red Hat…...
flutter开发实战-MethodChannel实现flutter与iOS双向通信
flutter开发实战-MethodChannel实现flutter与iOS双向通信 最近开发中需要iOS与flutter实现通信,这里使用的MethodChannel 如果需要flutter与Android实现双向通信,请看 https://blog.csdn.net/gloryFlow/article/details/132218837 这部分与https://bl…...
华为、阿里巴巴、字节跳动 100+ Python 面试问题总结(七)
系列文章目录 个人简介:机电专业在读研究生,CSDN内容合伙人,博主个人首页 Python面试专栏:《Python面试》此专栏面向准备面试的2024届毕业生。欢迎阅读,一起进步!🌟🌟🌟 …...
K8S系列一:概念入门
写在前面 本文组织方式: K8S的架构、作用和目的。需要首先对K8S整体有所了解。 K8S是什么? 为什么是K8S? K8S怎么做? K8S的重要概念,即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已被屏蔽
问题:该插件不支持 原因:现在浏览器默认禁用flash 博主当前使用的是谷歌浏览器Chrome 2个主要方法都已经失效 搜索一圈后,之前博客给出的2个主要方法都已经失效。 1、flash.cn 下载本地播放器 2、在chrome中打开flash的禁用开关 2023年解…...
【Spring专题】Spring之Bean的生命周期源码解析——阶段二(IOC之实例化)
目录 前言阅读准备阅读指引阅读建议 课程内容一、SpringIOC之实例化1.1 简单回顾1.2 概念回顾1.3 核心方法讲解 二、方法讲解2.1 AbstractBeanFactory#getMergedLocalBeanDefinition:合并BeanDefinition2.2 AbstractAutowireCapableBeanFactory#createBeanÿ…...
YOLOv8目标检测算法
YOLOv8目标检测算法相较于前几代YOLO系列算法具有如下的几点优势: 更友好的安装/运行方式速度更快、准确率更高新的backbone,将YOLOv5中的C3更换为C2FYOLO系列第一次尝试使用anchor-free新的损失函数 YOLOv8简介 YOLOv8 是 Ultralytics 公司继 YOLOv5…...
uniapp条件编译
uniapp条件编译 uni-app 将已经将可以跨平台兼容处理的组件及 API 等进行了封装,但是部分平台的特性无法跨平台。 由此,uni-app 提供了条件编译的方案,来处理不同平台的特定或差异。 写法 #ifdef: 仅在某平台存在#ifndef…...
2023年国赛数学建模思路 - 复盘:光照强度计算的优化模型
文章目录 0 赛题思路1 问题要求2 假设约定3 符号约定4 建立模型5 模型求解6 实现代码 建模资料 0 赛题思路 (赛题出来以后第一时间在CSDN分享) https://blog.csdn.net/dc_sinor?typeblog 1 问题要求 现在已知一个教室长为15米,宽为12米&…...
volte端到端问题分析(一)
1、MME专载保持功能验证 **描述:**当无线环境较差时,有可能由于“Radio_Connection_with_UE_Lost” 原因造成的VoLTE通话掉话,如果UE发生RRC重建成功,手机将不会掉话。 对MME1202进行功能验证:开启后,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。这里为了简单我们先不用真机,而是选择模拟器的方式,对于 x86_64 模拟器,我们选择的是 aosp_x86…...
铁是地球科学争论的核心
一项新的研究调查了地球内部铁的形态。这些发现对理解内核的结构产生了影响。 一项新的研究探索了地球内核的铁结构,如图中的黄色和白色所示。 资料来源:地球物理研究快报 地球内核以铁为主,铁可以多种晶体形式作为固体材料存在。(…...
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文字处理引擎,提供了广泛的文字处理功能。使用TX Text Control.NET Server,程序员可以开发…...
工作纪实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文档 之前我们在写字符雨的时候,将数据都存储到了视图类中,这是不合理的,视图类只负责显示,不应该存储任何数据 文档:专门存储数据用的 CDocument 文档与视图的关系: 创建一个文档类 单文档四个类都…...
2023年“研究生科研素养提升”系列公益讲座在线测评题目与参考答案
一、单选题 1、关于参考文献的选择,说法错误的是 ( ) 参考文献的选择有原创性、必要性的原则 不能过度引用 不能故意隐藏引用来源 可以引用无关参考文献 您的答案:D 参考答案:D 答案解析:不可以引用无关参考…...
IDEA运行Tomcat出现乱码问题解决汇总
最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…...
边缘计算医疗风险自查APP开发方案
核心目标:在便携设备(智能手表/家用检测仪)部署轻量化疾病预测模型,实现低延迟、隐私安全的实时健康风险评估。 一、技术架构设计 #mermaid-svg-iuNaeeLK2YoFKfao {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg…...
在rocky linux 9.5上在线安装 docker
前面是指南,后面是日志 sudo dnf config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo sudo dnf install docker-ce docker-ce-cli containerd.io -y docker version sudo systemctl start docker sudo systemctl status docker …...
相机Camera日志分析之三十一:高通Camx HAL十种流程基础分析关键字汇总(后续持续更新中)
【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了:有对最普通的场景进行各个日志注释讲解,但相机场景太多,日志差异也巨大。后面将展示各种场景下的日志。 通过notepad++打开场景下的日志,通过下列分类关键字搜索,即可清晰的分析不同场景的相机运行流程差异…...
多模态大语言模型arxiv论文略读(108)
CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文标题:CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文作者:Sayna Ebrahimi, Sercan O. Arik, Tejas Nama, Tomas Pfister ➡️ 研究机构: Google Cloud AI Re…...
华为云Flexus+DeepSeek征文|DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建
华为云FlexusDeepSeek征文|DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建 前言 如今大模型其性能出色,华为云 ModelArts Studio_MaaS大模型即服务平台华为云内置了大模型,能助力我们轻松驾驭 DeepSeek-V3/R1,本文中将分享如何…...
SAP学习笔记 - 开发26 - 前端Fiori开发 OData V2 和 V4 的差异 (Deepseek整理)
上一章用到了V2 的概念,其实 Fiori当中还有 V4,咱们这一章来总结一下 V2 和 V4。 SAP学习笔记 - 开发25 - 前端Fiori开发 Remote OData Service(使用远端Odata服务),代理中间件(ui5-middleware-simpleproxy)-CSDN博客…...
20个超级好用的 CSS 动画库
分享 20 个最佳 CSS 动画库。 它们中的大多数将生成纯 CSS 代码,而不需要任何外部库。 1.Animate.css 一个开箱即用型的跨浏览器动画库,可供你在项目中使用。 2.Magic Animations CSS3 一组简单的动画,可以包含在你的网页或应用项目中。 3.An…...
C#中的CLR属性、依赖属性与附加属性
CLR属性的主要特征 封装性: 隐藏字段的实现细节 提供对字段的受控访问 访问控制: 可单独设置get/set访问器的可见性 可创建只读或只写属性 计算属性: 可以在getter中执行计算逻辑 不需要直接对应一个字段 验证逻辑: 可以…...
MinIO Docker 部署:仅开放一个端口
MinIO Docker 部署:仅开放一个端口 在实际的服务器部署中,出于安全和管理的考虑,我们可能只能开放一个端口。MinIO 是一个高性能的对象存储服务,支持 Docker 部署,但默认情况下它需要两个端口:一个是 API 端口(用于存储和访问数据),另一个是控制台端口(用于管理界面…...
