Android日历提醒增删改查事件、添加天数不对问题

Android日历提醒是非常好的提醒功能,笔者在做的过程中,遇到的一些问题,现整理出来,以供参考。

 一、申请日历的读写权限

<uses-permission android:name="android.permission.WRITE_CALENDAR" />
<uses-permission android:name="android.permission.READ_CALENDAR" />

如果需要读取日历数据,需要加入READ_CALENDAR权限。若需要对日历数据进行删改操作,需要WRITE_CALENDAR权限。

二、事件表

CalendarContract.Events表包含事件的信息,比如标题、描述、开始时间等,如需查看所支持字段的完整列表,请参阅 CalendarContract.Events。

常量描述
_ID事件ID,非常重要,更新删除数据都要用到。
CALENDAR_ID事件所属日历用户ID。
TITLE事件的名称。
DESCRIPTION事件的描述。
DTSTART事件开始时间,以从公元纪年开始计算的协调世界时毫秒数表示。
DTEND事件结束时间,以从公元纪年开始计算的协调世界时毫秒数表示。
EVENT_TIMEZONE事件的时区。
EVENT_END_TIMEZONE事件结束时间的时区。
DURATIONRFC5545 格式的事件持续时间。例如,值为 "PT1H" 表示事件应持续一小时,值为 "P2W" 表示持续 2 周。
ALL_DAY值为 1 表示此事件占用一整天(按照本地时区的定义)。值为 0 表示它是常规事件,可在一天内的任何时间开始和结束。
RRULE事件的重复发生规则格式。例如,"FREQ=WEEKLY;COUNT=10;WKST=SU"。您可以在此处找到更多示例。
RDATE事件的重复发生日期。RDATE 与 RRULE 通常联合用于定义一组存在聚合关系的重复实例。如需查看更详细的介绍,请参阅 RFC5545 规范。
AVAILABILITY将此事件视为忙碌时间还是可调度的空闲时间。

HAS_ALARM

设置是否有闹钟提醒,0-没有,1-有

三、日历用户

增加事件需要用户信息(CalendarContract.Events.CALENDAR_ID),相关代码如下:

private static final String CALENDER_URL = "content://com.android.calendar/calendars";private static final String CALENDARS_NAME = "XX";// 随便写
private static final String CALENDARS_ACCOUNT_NAME = "XX";// 随便写
private static final String CALENDARS_ACCOUNT_TYPE = CalendarContract.ACCOUNT_TYPE_LOCAL;
private static final String CALENDARS_DISPLAY_NAME = StringUtils.getString(R.string.app_name) + "账户";// 随便写/*** 检查是否已经添加了日历账户,如果没有添加先添加一个日历账户再查询* @return 获取账户成功返回账户id,否则返回-1*/
private static int checkAndAddCalendarAccount(Context context) {int oldId = checkCalendarAccount(context);if (oldId >= 0) {return oldId;} else {long addId = addCalendarAccount(context);if (addId >= 0) {return checkCalendarAccount(context);} else {return -1;}}
}/*** @return 检查是否存在现有账户,存在则返回账户id,否则返回-1*/
private static int checkCalendarAccount(Context context) {Cursor userCursor = context.getContentResolver().query(Uri.parse(CALENDER_URL), null, null, null, null);try {if (userCursor == null) { //查询返回空值return -1;}int columnIndex = userCursor.getColumnIndex(CalendarContract.Calendars._ID);int count = userCursor.getCount();if (count > 0) { //存在现有账户,取第一个账户的id返回userCursor.moveToFirst();return userCursor.getInt(columnIndex);} else {return -1;}} catch (Exception ex) {// do nothingreturn -1;} finally {if (userCursor != null) {userCursor.close();}}
}/*** @return 添加日历账户,账户创建成功则返回账户id,否则返回-1*/
private static long addCalendarAccount(Context context) {TimeZone timeZone = TimeZone.getDefault();ContentValues value = new ContentValues();value.put(CalendarContract.Calendars.NAME, CALENDARS_NAME);value.put(CalendarContract.Calendars.ACCOUNT_NAME, CALENDARS_ACCOUNT_NAME);value.put(CalendarContract.Calendars.ACCOUNT_TYPE, CALENDARS_ACCOUNT_TYPE);value.put(CalendarContract.Calendars.CALENDAR_DISPLAY_NAME, CALENDARS_DISPLAY_NAME);value.put(CalendarContract.Calendars.VISIBLE, 1);value.put(CalendarContract.Calendars.CALENDAR_COLOR, Color.BLUE);value.put(CalendarContract.Calendars.CALENDAR_ACCESS_LEVEL, CalendarContract.Calendars.CAL_ACCESS_OWNER);value.put(CalendarContract.Calendars.SYNC_EVENTS, 1);value.put(CalendarContract.Calendars.CALENDAR_TIME_ZONE, timeZone.getID());value.put(CalendarContract.Calendars.OWNER_ACCOUNT, CALENDARS_ACCOUNT_NAME);value.put(CalendarContract.Calendars.CAN_ORGANIZER_RESPOND, 0);Uri calendarUri = Uri.parse(CALENDER_URL);calendarUri = calendarUri.buildUpon().appendQueryParameter(CalendarContract.CALLER_IS_SYNCADAPTER, "true").appendQueryParameter(CalendarContract.Calendars.ACCOUNT_NAME, CALENDARS_ACCOUNT_NAME).appendQueryParameter(CalendarContract.Calendars.ACCOUNT_TYPE, CALENDARS_ACCOUNT_TYPE).build();Uri result = context.getContentResolver().insert(calendarUri, value);return result == null ? -1 : ContentUris.parseId(result);
}

四、增加日历事件

先创建日历事件,再将事件加入日历提醒,代码如下:

private static final String CALENDER_EVENT_URL = "content://com.android.calendar/events";
private static final String CALENDER_REMINDER_URL = "content://com.android.calendar/reminders";
private static final String[] PERMISSION_LIST = {Manifest.permission.READ_CALENDAR, Manifest.permission.WRITE_CALENDAR};/*** 真正添加日历事件** @param context:上下文* @param map:日历事件的信息* @return 返回事件id*/
private static long addCalendarEvent(Context context, @NonNull Map<String, Object> map) {if (context == null) {return 0;}if (!XXPermissions.isGranted(context, PERMISSION_LIST)) {return 0;}int calId = checkAndAddCalendarAccount(context); //获取日历账户的idif (calId < 0) { //获取账户id失败直接返回,添加日历事件失败return 0;}String title = (String) map.get("title");String description = (String) map.get("description");long start = 0;Object originStartTime = map.get("start_time");if (originStartTime instanceof Long) {start = (Long) originStartTime;}String rule = (String) map.get("rule");ContentValues event = new ContentValues();event.put(CalendarContract.Events.TITLE, title);event.put(CalendarContract.Events.DESCRIPTION, description);event.put(CalendarContract.Events.CALENDAR_ID, calId); //插入账户的idevent.put(CalendarContract.Events.DTSTART, start);//event.put(CalendarContract.Events.DTEND, end);// 与DURATION不能同时设置event.put(CalendarContract.Events.HAS_ALARM, 1);//设置有闹钟提醒event.put(CalendarContract.Events.EVENT_TIMEZONE, TimeZone.getDefault().getID());//这个是时区,必须有event.put(CalendarContract.Events.DURATION, "P0S");// 重复设置必须设置,否则日期有问题if (!StringUtils.isEmpty(rule)) {event.put(CalendarContract.Events.RRULE, rule);// 重复规则 rrule(Recurrence Rule)}Uri newEvent = context.getContentResolver().insert(Uri.parse(CALENDER_EVENT_URL), event); //添加事件if (newEvent == null) { //添加日历事件失败直接返回LogUtils.d("addCalendarEvent2 newEvent 添加日历事件失败直接返回");return 0;}//事件提醒的设定ContentValues values = new ContentValues();values.put(CalendarContract.Reminders.EVENT_ID, ContentUris.parseId(newEvent));values.put(CalendarContract.Reminders.MINUTES, 0);// 提前0分钟提醒values.put(CalendarContract.Reminders.METHOD, CalendarContract.Reminders.METHOD_ALERT);Uri reminderEvent = context.getContentResolver().insert(Uri.parse(CALENDER_REMINDER_URL), values);if (reminderEvent == null) {LogUtils.d("addCalendarEvent2 reminderEvent 添加日历事件失败直接返回");}return Long.parseLong(newEvent.getLastPathSegment());
}

总结下增加事件的规则:

  • 您必须加入 CALENDAR_ID 和 DTSTART。
  • 您必须加入 EVENT_TIMEZONE。如需获取系统中已安装时区 ID 的列表,请使用 getAvailableIDs()。请注意,如果您按使用 Intent 插入事件中所述通过 INSERT Intent 插入事件,则此规则不适用 — 在该情形下,系统会提供默认时区。
  • 对于非重复事件,您必须加入 DTEND。
  • 对于重复事件,您必须加入 DURATION,以及 RRULE 或 RDATE。请注意,如果您按使用 Intent 插入事件中所述通过 INSERT Intent 插入事件,则此规则不适用 — 在该情形下,您可以将 RRULE 与 DTSTART 和 DTEND 结合使用,日历应用会自动将其转换为持续时间。
  • RRULE规则可参考:iCalendar Recurrence Rule 规范翻译 - 简书 (jianshu.com)

碎碎念:

1、之前添加重复事件,只设置 DTEND ,导致添加的天数有问题,改成DURATION就好了。

2、_ID(事件ID)非常重要,增加日历提醒后需要保存起来,后续的删改查都需要这个信息。

五、查找日历事件

查找比较简单了,这里使用事件ID(_ID),代码如下:

/*** 检查日历事件** @param context:* @param eventId:* @return 返回事件是否存在。true-事件存在,false-事件不存在*/
public static boolean checkCalendarEvent(Context context, long eventId) {if (context == null || eventId <= 0) {return false;}if (!XXPermissions.isGranted(context, PERMISSION_LIST)) {return false;}Uri uri = ContentUris.withAppendedId(Uri.parse(CALENDER_EVENT_URL), eventId);Cursor eventCursor = context.getContentResolver().query(uri, null, null, null, null);if (eventCursor == null) {return false;}boolean result = eventCursor.getCount() > 0;eventCursor.close();return result;
}

六、修改日历事件 

public static void updateCalendarEvent(Context context, long eventId) {if (context == null || eventId <= 0) {return;}if (!XXPermissions.isGranted(context, PERMISSION_LIST)) {return;}if (!checkCalendarEvent(context, eventId)) {return;}try {Uri uri = ContentUris.withAppendedId(Uri.parse(CALENDER_EVENT_URL), eventId);ContentValues values = new ContentValues();values.put(CalendarContract.Events.TITLE, "XXXX");int rows = context.getContentResolver().update(uri, values, null, null);LogUtils.d("updateCalendarEvent rows=" + rows);} catch (Exception e) {e.printStackTrace();}
}

 七、删除日历事件

/*** 删除日历事件*/
public static void deleteCalendarEvent(Context context, long eventId) {if (context == null || eventId <= 0) {return;}if (!XXPermissions.isGranted(context, PERMISSION_LIST)) {return;}if (!checkCalendarEvent(context, eventId)) {return;}try {Uri uri = ContentUris.withAppendedId(Uri.parse(CALENDER_EVENT_URL), eventId);int rows = context.getContentResolver().delete(uri, null, null);LogUtils.d("deleteCalendarEventById rows=" + rows);} catch (Exception e) {e.printStackTrace();}
}

八、完整代码

import android.Manifest;
import android.content.ContentUris;
import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.graphics.Color;
import android.net.Uri;
import android.provider.CalendarContract;
import android.text.TextUtils;
import android.util.ArrayMap;import androidx.annotation.NonNull;
import androidx.annotation.Nullable;import com.blankj.utilcode.util.LogUtils;
import com.blankj.utilcode.util.StringUtils;
import com.hjq.permissions.XXPermissions;import java.util.ArrayList;
import java.util.Calendar;
import java.util.List;
import java.util.Map;
import java.util.TimeZone;/*** 日历提醒工具类*/
public class CalendarReminderUtils {private static final String CALENDER_URL = "content://com.android.calendar/calendars";private static final String CALENDER_EVENT_URL = "content://com.android.calendar/events";private static final String CALENDER_REMINDER_URL = "content://com.android.calendar/reminders";private static final String CALENDARS_NAME = "XXX";private static final String CALENDARS_ACCOUNT_NAME = "XXX";private static final String CALENDARS_ACCOUNT_TYPE = CalendarContract.ACCOUNT_TYPE_LOCAL;private static final String CALENDARS_DISPLAY_NAME = StringUtils.getString(R.string.app_name) + "账户";private static final String[] PERMISSION_LIST = {Manifest.permission.READ_CALENDAR, Manifest.permission.WRITE_CALENDAR};/*** 检查是否已经添加了日历账户,如果没有添加先添加一个日历账户再查询* @return 获取账户成功返回账户id,否则返回-1*/private static int checkAndAddCalendarAccount(Context context) {int oldId = checkCalendarAccount(context);if (oldId >= 0) {return oldId;} else {long addId = addCalendarAccount(context);if (addId >= 0) {return checkCalendarAccount(context);} else {return -1;}}}/*** @return 检查是否存在现有账户,存在则返回账户id,否则返回-1*/private static int checkCalendarAccount(Context context) {Cursor userCursor = context.getContentResolver().query(Uri.parse(CALENDER_URL), null, null, null, null);try {if (userCursor == null) { //查询返回空值return -1;}int columnIndex = userCursor.getColumnIndex(CalendarContract.Calendars._ID);int count = userCursor.getCount();if (count > 0) { //存在现有账户,取第一个账户的id返回userCursor.moveToFirst();return userCursor.getInt(columnIndex);} else {return -1;}} catch (Exception ex) {// do nothingreturn -1;} finally {if (userCursor != null) {userCursor.close();}}}/*** @return 添加日历账户,账户创建成功则返回账户id,否则返回-1*/private static long addCalendarAccount(Context context) {TimeZone timeZone = TimeZone.getDefault();ContentValues value = new ContentValues();value.put(CalendarContract.Calendars.NAME, CALENDARS_NAME);value.put(CalendarContract.Calendars.ACCOUNT_NAME, CALENDARS_ACCOUNT_NAME);value.put(CalendarContract.Calendars.ACCOUNT_TYPE, CALENDARS_ACCOUNT_TYPE);value.put(CalendarContract.Calendars.CALENDAR_DISPLAY_NAME, CALENDARS_DISPLAY_NAME);value.put(CalendarContract.Calendars.VISIBLE, 1);value.put(CalendarContract.Calendars.CALENDAR_COLOR, Color.BLUE);value.put(CalendarContract.Calendars.CALENDAR_ACCESS_LEVEL, CalendarContract.Calendars.CAL_ACCESS_OWNER);value.put(CalendarContract.Calendars.SYNC_EVENTS, 1);value.put(CalendarContract.Calendars.CALENDAR_TIME_ZONE, timeZone.getID());value.put(CalendarContract.Calendars.OWNER_ACCOUNT, CALENDARS_ACCOUNT_NAME);value.put(CalendarContract.Calendars.CAN_ORGANIZER_RESPOND, 0);Uri calendarUri = Uri.parse(CALENDER_URL);calendarUri = calendarUri.buildUpon().appendQueryParameter(CalendarContract.CALLER_IS_SYNCADAPTER, "true").appendQueryParameter(CalendarContract.Calendars.ACCOUNT_NAME, CALENDARS_ACCOUNT_NAME).appendQueryParameter(CalendarContract.Calendars.ACCOUNT_TYPE, CALENDARS_ACCOUNT_TYPE).build();Uri result = context.getContentResolver().insert(calendarUri, value);return result == null ? -1 : ContentUris.parseId(result);}/*** 添加日历事件--就寝提醒** @param context      上下文* @param title        提醒标题* @param description  提醒描述* @param reminderTime 就寝时间,格式HH:mm*/public static long addCalendarFromSleepAlarm(Context context, String title, String description, @Nullable String reminderTime) {Map<String, Object> map = new ArrayMap<>();map.put("title", title);map.put("description", description);map.put("rule", "FREQ=DAILY;COUNT=365");if (reminderTime == null) {reminderTime = "22:00";}String[] split = reminderTime.split(":");if (split.length != 2) {return 0;}Calendar calendar = Calendar.getInstance();calendar.set(Calendar.HOUR_OF_DAY, Integer.parseInt(split[0]));calendar.set(Calendar.MINUTE, Integer.parseInt(split[1]));calendar.set(Calendar.SECOND, 0);long start = calendar.getTime().getTime();map.put("start_time", start);try {return addCalendarEvent(context, map);} catch (Exception e) {ExceptionManager.get().report(e);}return 0;}/*** 真正添加日历事件** @param context:上下文* @param map:日历事件的信息* @return 返回事件id*/private static long addCalendarEvent(Context context, @NonNull Map<String, Object> map) {if (context == null) {return 0;}if (!XXPermissions.isGranted(context, PERMISSION_LIST)) {return 0;}int calId = checkAndAddCalendarAccount(context); //获取日历账户的idLogUtils.d("addCalendarEvent checkAndAddCalendarAccount calId=" + calId);if (calId < 0) { //获取账户id失败直接返回,添加日历事件失败return 0;}String title = (String) map.get("title");String description = (String) map.get("description");long start = 0;Object originStartTime = map.get("start_time");if (originStartTime instanceof Long) {start = (Long) originStartTime;}String rule = (String) map.get("rule");ContentValues event = new ContentValues();event.put(CalendarContract.Events.TITLE, title);event.put(CalendarContract.Events.DESCRIPTION, description);event.put(CalendarContract.Events.CALENDAR_ID, calId); //插入账户的idevent.put(CalendarContract.Events.DTSTART, start);//event.put(CalendarContract.Events.DTEND, end);// 与DURATION不能同时设置event.put(CalendarContract.Events.HAS_ALARM, 1);//设置有闹钟提醒event.put(CalendarContract.Events.EVENT_TIMEZONE, TimeZone.getDefault().getID());//这个是时区,必须有event.put(CalendarContract.Events.DURATION, "P0S");// 重复设置必须设置,否则日期有问题if (!StringUtils.isEmpty(rule)) {event.put(CalendarContract.Events.RRULE, rule);// 重复规则 rrule(Recurrence Rule)}Uri newEvent = context.getContentResolver().insert(Uri.parse(CALENDER_EVENT_URL), event); //添加事件if (newEvent == null) { //添加日历事件失败直接返回LogUtils.d("addCalendarEvent2 newEvent 添加日历事件失败直接返回");return 0;}//事件提醒的设定ContentValues values = new ContentValues();values.put(CalendarContract.Reminders.EVENT_ID, ContentUris.parseId(newEvent));values.put(CalendarContract.Reminders.MINUTES, 0);// 提前0分钟提醒values.put(CalendarContract.Reminders.METHOD, CalendarContract.Reminders.METHOD_ALERT);Uri reminderEvent = context.getContentResolver().insert(Uri.parse(CALENDER_REMINDER_URL), values);if (reminderEvent == null) {LogUtils.d("addCalendarEvent2 reminderEvent 添加日历事件失败直接返回");}LogUtils.d("addCalendarEvent 事件id=" + Long.parseLong(newEvent.getLastPathSegment()));return Long.parseLong(newEvent.getLastPathSegment());}/*** 检查日历事件** @param context:* @param eventId:* @return 返回事件是否存在。true-事件存在,false-事件不存在*/public static boolean checkCalendarEvent(Context context, long eventId) {if (context == null || eventId <= 0) {return false;}if (!XXPermissions.isGranted(context, PERMISSION_LIST)) {return false;}Uri uri = ContentUris.withAppendedId(Uri.parse(CALENDER_EVENT_URL), eventId);Cursor eventCursor = context.getContentResolver().query(uri, null, null, null, null);if (eventCursor == null) {return false;}boolean result = eventCursor.getCount() > 0;LogUtils.d("checkCalendarEvent eventCursor=" + eventCursor.getCount() + ",result=" + result);eventCursor.close();return result;}/*** 删除日历事件*/public static void deleteCalendarEvent(Context context, long eventId) {if (context == null || eventId <= 0) {return;}if (!XXPermissions.isGranted(context, PERMISSION_LIST)) {return;}if (!checkCalendarEvent(context, eventId)) {return;}try {Uri uri = ContentUris.withAppendedId(Uri.parse(CALENDER_EVENT_URL), eventId);int rows = context.getContentResolver().delete(uri, null, null);LogUtils.d("deleteCalendarEventById rows=" + rows);} catch (Exception e) {e.printStackTrace();}}public static void updateCalendarEvent(Context context, long eventId) {if (context == null || eventId <= 0) {return;}if (!XXPermissions.isGranted(context, PERMISSION_LIST)) {return;}if (!checkCalendarEvent(context, eventId)) {return;}try {Uri uri = ContentUris.withAppendedId(Uri.parse(CALENDER_EVENT_URL), eventId);ContentValues values = new ContentValues();values.put(CalendarContract.Events.TITLE, "XXXX");int rows = context.getContentResolver().update(uri, values, null, null);LogUtils.d("updateCalendarEvent rows=" + rows);} catch (Exception e) {e.printStackTrace();}}
}

参考文章:

Android 日历表事件表操作_android 读写系统日历 删除-CSDN博客

iCalendar Recurrence Rule 规范翻译 - 简书 (jianshu.com)

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

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

相关文章

Lua速成(5)table

一、table table 是 Lua 的一种数据结构用来帮助我们创建不同的数据类型&#xff0c;如&#xff1a;数组、字典等。 Lua table 使用关联型数组&#xff0c;你可以用任意类型的值来作数组的索引&#xff0c;但这个值不能是 nil。 Lua table 是不固定大小的&#xff0c;你可以…

springboot-基础-eclipse集成mybatis+使用方法+排错

备份笔记。所有代码都是2019年测试通过的&#xff0c;如有问题请自行搜索解决&#xff01; 目录 集成mybatis安装mybatis的jar包安装插件&#xff1a;mybatis-generator安装方法生成方法报错&#xff1a;java.lang.RuntimeException: Exception getting JDBC Driver mybatis注解…

数据之美:用山海鲸展现数据魅力

在数据分析领域&#xff0c;数据可视化是一个至关重要的环节。作为一名资深的数据分析师&#xff0c;我深知一个直观、易于理解的数据可视化看板对于传达信息、辅助决策的重要性。今天&#xff0c;我将以自己在用的山海鲸可视化软件为例&#xff0c;与大家分享如何制作一个高效…

LeetCode 热题 100 | 图论(上)

目录 1 200. 岛屿数量 2 994. 腐烂的橘子 2.1 智障遍历法 2.2 仿层序遍历法 菜鸟做题&#xff0c;语言是 C 1 200. 岛屿数量 解题思路&#xff1a; 遍历二维数组&#xff0c;寻找 “1”&#xff08;若找到则岛屿数量 1&#xff09;寻找与当前 “1” 直接或间接连接在…

项目登录方案选型

一.Cookie + Session 登录 大家都知道,HTTP 是一种无状态的协议。无状态是指协议对于事务处理没有记忆能力,服务器不知道客户端是什么状态。即我们给服务器发送 HTTP 请求之后,服务器根据请求返回数据,但不会记录任何信息。为了解决 HTTP 无状态的问题,出现了 Cookie。Co…

离线数仓(四)【数仓数据同步策略】

前言 今天来把数仓数据同步解决掉&#xff0c;前面我们已经把日志数据到 Kafka 的通道打通了。 1、实时数仓数据同步 关于实时数仓&#xff0c;我们的 Flink 直接去 Kafka 读取即可&#xff0c;我们在学习 Flink 的时候也知道 Flink 提供了 Kafka Source&#xff0c;所以这里不…

协议-http协议-基础概念02-请求应答过程-请求响应报文结构-头部字段-请求方法-响应方式

参考来源&#xff1a; 极客时间-透视HTTP协议(作者&#xff1a;罗剑锋)&#xff1b; web抓包实战课-陶辉&#xff1b; 01-HTTP协议请求-应答过程 最简单的浏览器 HTTP 请求过程 浏览器从地址栏的输入中获得服务器的 IP 地址和端口号&#xff1b;浏览器用 TCP 的三次握手与服…

CSS复合选择器(一)

CSS复合选择器&#xff08;一&#xff09; 1.交集选择器2. 并集选择器3. 后代选择器4. 子代选择器5.兄弟选择器5.1相邻兄弟选择器&#xff1a;5.2通用兄弟选择器&#xff1a; 6.属性选择器 1.交集选择器 作用&#xff1a;选中同时符合多个条件的元素。 交集有并且的含义&#…

基于springboot的4S店车辆管理系统源码和论文

随着信息技术和网络技术的飞速发展&#xff0c;人类已进入全新信息化时代&#xff0c;传统管理技术已无法高效&#xff0c;便捷地管理信息。为了迎合时代需求&#xff0c;优化管理效率&#xff0c;各种各样的管理系统应运而生&#xff0c;各行各业相继进入信息管理时代&#xf…

python中写monogo的uri参数里,为什么有多个IP,是连接多个服务器吗

问题来源&#xff1a; 代码如下&#xff1a; from pymongo import MongoClientmongo_uri "mongodb://admin:password192.168.93.23:32725,192.132.9.35:32740,192.22.16.42:30538/?" # 创建MongoClient实例 client MongoClient(mongo_uri)为什么uri中会存在多个…

win10开机黑屏,只有鼠标,解决方案

问题描述 win10进不去桌面&#xff0c;可以进去锁屏&#xff0c;只有鼠标&#xff0c;也能进去任务管理器&#xff08;ctrlwindelete&#xff09;, 问题分析 进入任务管理器->文件->运行新任务 然后输入 explorer.exe 发现找不到了 原因&#xff1a;误删explorer.exe …

Linux系统---nginx(1)服务

目录 一.Nginx概述 1.定义 2.Nginx模块作用 &#xff08;1&#xff09;main模块 &#xff08;2&#xff09;stream服务模块 &#xff08;3&#xff09;邮件服务模块 &#xff08;4&#xff09;第三方模块 &#xff08;5&#xff09;events模块 &#xff08;6&#xff0…

300分钟吃透分布式缓存-16讲:常用的缓存组件Redis是如何运行的?

Redis 基本原理 Redis 简介 Redis 是一款基于 ANSI C 语言编写的&#xff0c;BSD 许可的&#xff0c;日志型 key-value 存储组件&#xff0c;它的所有数据结构都存在内存中&#xff0c;可以用作缓存、数据库和消息中间件。 Redis 是 Remote dictionary server 即远程字典服务…

SpringCloud有哪些组件

什么是SpringCloud&#xff1f; Spring Cloud是基于Spring Boot的分布式系统开发工具&#xff0c;它提供了一系列开箱即用的、针对分布式系统开发的特性和组件&#xff0c;用于帮助开发人员快速构建和管理云原生应用程序。 Spring Cloud的主要目标是解决分布式系统中的常见问题…

linux使用 busybox microcom AT指令测试4G/5G模块

1、busybox microcom命令使用方法如下&#xff1a; 参数&#xff1a; -d&#xff1a;表示延时时间&#xff0c;一般不设置。 -t&#xff1a;表示超时时间&#xff0c;超多长时间后该命令自动退出。单位为ms。 -s&#xff1a;表示传输速度&#xff0c;即串口波特率。 -X&#…

阿里云oss工具ossutil使用手册(windows)

一、下载安装 下载地址和教程 安装ossutil。 单击下载链接下载Windows安装包。 将工具解压&#xff0c;并双击运行ossutil.bat文件。 配置ossutil。 输入配置命令。 ossutil config 根据提示设置配置文件路径。 请输入配置文件名&#xff0c;文件名可以带路径&#xff08;…

代码随想录算法训练营第二十七天补|39. 组合总和 ● 40.组合总和II ● 131.分割回文串

组合问题&#xff1a;集合内元素的组合&#xff0c;不同集合内元素的组合 分割问题&#xff1a;本质还是组合问题&#xff0c;注意一下如何分割字符串 回溯模板伪代码 void backtracking(参数) {if (终止条件) {存放结果;return;}for (选择&#xff1a;本层集合中元素&#xf…

HTML5 增加了辅助 DOM 焦点管理的功能

焦点管理 ​ HTML5 增加了辅助 DOM 焦点管理的功能。 ​ activeElement 可以用来查询文档&#xff0c;确定哪个元素拥有焦点&#xff0c;hasFocus() 方法可以查询文档是否获得了焦点&#xff0c; 而这对于保证 Web 应用程序的无障碍使用是非常重要的。无障碍 Web 应用程序的一…

算法提升——LeetCode第 386 场周赛总结

题目 分割数据 给你一个长度为偶数的整数数组nums。你需要将这个数组分割成nums1和nums2两部分&#xff0c;要求&#xff1a; nums1.lengthnums2.lengthnums.length/2。 nums1应包含互不相同的元素。 nums2也应包含互不相同的元素。 如果能够分割数组就返回true&#xff0c;…

vue2 项目 vscode 配置

安装node.js 下载https://nodejs.org/zh-cn/download/ 安装到目录D:\nodejs&#xff08;自定义&#xff0c;不能有空格、中文、特殊字符&#xff09;配置环境变量&#xff0c;将安装目录加入Path环境变量配置npm 在安装目录新建 node_cache 和 node_global 两个文件夹&#xff…