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

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)

相关文章:

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

Android日历提醒是非常好的提醒功能&#xff0c;笔者在做的过程中&#xff0c;遇到的一些问题&#xff0c;现整理出来&#xff0c;以供参考。 一、申请日历的读写权限 <uses-permission android:name"android.permission.WRITE_CALENDAR" /> <uses-permiss…...

【力扣hot100】刷题笔记Day15

前言 今天要刷的是图论&#xff0c;还没学过&#xff0c;先看看《代码随想录》这部分的基础 深搜DFS理论基础 深搜三部曲 确认递归函数、参数确认终止条件处理目前搜索节点出发的路径 代码框架 void dfs(参数) {if (终止条件) {存放结果;return;}for (选择&#xff1a;本节点…...

vue-显示数据

​ v-text和v-html专门用来展示数据, 其作用和插值表达式类似。v-text和v-html可以避免插值闪烁问题. ​ 当网速比较慢时, 使用{{}}来展示数据, 有可能会产生插值闪烁问题。 ​ 插值闪烁: 在数据未加载完成时&#xff0c;页面会显示出原始的{{}}, 过一会才会展示正常数据.语法…...

kali linux常用命令

1. 网络扫描 功能&#xff1a;网络扫描是用来发现网络中的设备、服务和开放端口的过程。 命令&#xff1a;nmap 例子&#xff1a;nmap -sP 192.168.1.0/24 这个命令使用 Nmap 进行网络扫描&#xff0c;列出 192.168.1.0/24 网段中的所有活跃主机。 2. 密码破解 功能&#xf…...

HTML5:七天学会基础动画网页4

backgorund-size 值与说明 length(单位像素):设置背景图片高度和宽度&#xff0c;第一个值设置宽度&#xff0c;第二个值设置高度&#xff0c;如果只给出一个值&#xff0c;第二个是设置为auto。 percentage(百分比):以父元素的百分比来设置背景图像的宽度和高度&#xff0c…...

Web安全之接口鉴权

目录 接口鉴权定义 为什么会有cookie还有session还有token这种技术的存在?...

shardingsphere 集成springboot【水平分表】

创建sharding_sphere数据库 在数据库中创建两张表&#xff0c;t_order_1和t_order_2 分片规则&#xff1a;如果订单编号是偶数添加到t_order_1,如果是奇数添加到t_order_2 创建实体类 public class Order { private Integer id; private Integer orderType; private Int…...

GO 的 Web 开发系列(六)—— 遍历路径下的文件

文件 IO 处理是程序的基础功能&#xff0c;WEB 程序中通过文件 IO 实现附件的上传与下载。在 GO 中&#xff0c;有多种方式可以遍历文件目录&#xff0c;获取文件路径&#xff0c;本文从使用层面上论述这些函数。 预先准备一个包含子目录的目录&#xff0c;用于遍历测试&#…...

Flutter 处理异步操作并根据异步操作状态动态构建界面的方法FutureBuilder

概述 当界面的内容需要依靠网络请求的数据&#xff0c;就需要处理苦恼的&#xff0c;状态是空&#xff0c;非空的逻辑了&#xff0c;不然页面构建可能会报错&#xff0c;而FutureBuilder提供了一个非常好的解决方法&#xff0c;直接看代码 代码 异步操作函数 即网络请求函数…...

Git教程-Git的基本使用

Git是一个强大的分布式版本控制系统&#xff0c;它不仅用于跟踪代码的变化&#xff0c;还能够协调多个开发者之间的工作。在软件开发过程中&#xff0c;Git被广泛应用于协作开发、版本管理和代码追踪等方面。以下是一个详细的Git教程&#xff0c;我们将深入探讨Git的基本概念和…...

Java解决长度为K子的数组中的的最大和

Java解决长度为K子的数组中的的最大和 01 题目 给你一个整数数组 nums 和一个整数 k 。请你从 nums 中满足下述条件的全部子数组中找出最大子数组和&#xff1a; 子数组的长度是 k&#xff0c;且子数组中的所有元素 各不相同 。 返回满足题面要求的最大子数组和。如果不存在子…...

【手机端测试】adb基础命令

一、什么是adb adb&#xff08;Android Debug Bridge&#xff09;是android sdk的一个工具 adb是用来连接安卓手机和PC端的桥梁&#xff0c;要有adb作为二者之间的维系&#xff0c;才能让用户在电脑上对手机进行全面的操作。 Android的初衷是用adb这样的一个工具来协助开发人…...

【数据结构】深入探讨二叉树的遍历和分治思想(一)

&#x1f6a9;纸上得来终觉浅&#xff0c; 绝知此事要躬行。 &#x1f31f;主页&#xff1a;June-Frost &#x1f680;专栏&#xff1a;数据结构 &#x1f525;该文章主要讲述二叉树的递归结构及分治算法的思想。 目录&#xff1a; &#x1f30d;前言&#xff1a;&#x1f30d;…...

jQuery AJAX get() 和 post() 方法—— W3school 详解 简单易懂(二十四)

jQuery get() 和 post() 方法用于通过 HTTP GET 或 POST 请求从服务器请求数据。 HTTP 请求&#xff1a;GET vs. POST 两种在客户端和服务器端进行请求-响应的常用方法是&#xff1a;GET 和 POST。 GET - 从指定的资源请求数据POST - 向指定的资源提交要处理的数据 GET 基本…...

Linux中如何进行LVM逻辑卷扩容?

#注意&#xff1a;如果lv所在的vg有空间直接扩容就ok了&#xff01; 1.创建pv pvcreate /dev/sdb 执行以上命令得到以下内容&#xff1a; Physical volume "/dev/sdb" successfully created. 2.直接vgextend扩容 vgextend vg1 /dev/sdb #卷组名字&#xff0c;将…...

现代企业架构框架——应用架构

现代企业架构框架——应用架构。 现代企业架构中的应用架构是指企业在构建和维护应用系统时所采用的一种架构框架。应用架构旨在实现应用系统的可扩展性、灵活性、可维护性和可重用性,以满足企业在数字化时代对应用系统的快速交付和持续创新的需求。下面将详细介绍应用架构的…...

期货开户保证金保障市场正常运转

期货保证金是什么&#xff1f;在期货市场上&#xff0c;采取保证金交易制度&#xff0c;投资者只需按期货合约的价值&#xff0c;交一定比率少量资金即可参与期货合约买卖交易&#xff0c;这种资金就是期货保证金。期货保证金&#xff08;以下简称保证金〕按性质与作用的不同。…...

WebGIS----wenpack

学习资料&#xff1a;https://webpack.js.org/concepts/ 简介&#xff1a; Webpack 是一个现代化的 JavaScript 应用程序的模块打包工具。它能够将多个 JavaScript 文件和它们的依赖打包成一个单独的文件&#xff0c;以供在网页中使用。 Webpack 还具有编译和转换其他类型文…...

【Maven】Maven 基础教程(二):Maven 的使用

《Maven 基础教程》系列&#xff0c;包含以下 2 篇文章&#xff1a; Maven 基础教程&#xff08;一&#xff09;&#xff1a;基础介绍、开发环境配置Maven 基础教程&#xff08;二&#xff09;&#xff1a;Maven 的使用 &#x1f60a; 如果您觉得这篇文章有用 ✔️ 的话&#…...

mirthConnect忽略HTTPS SSL验证

mirthConnect SSL忽略验证 1、下载https网站证书 点击不安全---->证书无效 2、查看mirth 秘钥库口令 在mirthConnect 的conf目录下面keystore.storepass 3、导入证书到本地 在jdk的bin目录下面执行 keytool -importcert -file "下载的网站证书路径" -keysto…...

多云管理“拦路虎”:深入解析网络互联、身份同步与成本可视化的技术复杂度​

一、引言&#xff1a;多云环境的技术复杂性本质​​ 企业采用多云策略已从技术选型升维至生存刚需。当业务系统分散部署在多个云平台时&#xff0c;​​基础设施的技术债呈现指数级积累​​。网络连接、身份认证、成本管理这三大核心挑战相互嵌套&#xff1a;跨云网络构建数据…...

MPNet:旋转机械轻量化故障诊断模型详解python代码复现

目录 一、问题背景与挑战 二、MPNet核心架构 2.1 多分支特征融合模块(MBFM) 2.2 残差注意力金字塔模块(RAPM) 2.2.1 空间金字塔注意力(SPA) 2.2.2 金字塔残差块(PRBlock) 2.3 分类器设计 三、关键技术突破 3.1 多尺度特征融合 3.2 轻量化设计策略 3.3 抗噪声…...

[2025CVPR]DeepVideo-R1:基于难度感知回归GRPO的视频强化微调框架详解

突破视频大语言模型推理瓶颈,在多个视频基准上实现SOTA性能 一、核心问题与创新亮点 1.1 GRPO在视频任务中的两大挑战 ​安全措施依赖问题​ GRPO使用min和clip函数限制策略更新幅度,导致: 梯度抑制:当新旧策略差异过大时梯度消失收敛困难:策略无法充分优化# 传统GRPO的梯…...

日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする

日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする 1、前言(1)情况说明(2)工程师的信仰2、知识点(1) にする1,接续:名词+にする2,接续:疑问词+にする3,(A)は(B)にする。(2)復習:(1)复习句子(2)ために & ように(3)そう(4)にする3、…...

Zustand 状态管理库:极简而强大的解决方案

Zustand 是一个轻量级、快速和可扩展的状态管理库&#xff0c;特别适合 React 应用。它以简洁的 API 和高效的性能解决了 Redux 等状态管理方案中的繁琐问题。 核心优势对比 基本使用指南 1. 创建 Store // store.js import create from zustandconst useStore create((set)…...

java调用dll出现unsatisfiedLinkError以及JNA和JNI的区别

UnsatisfiedLinkError 在对接硬件设备中&#xff0c;我们会遇到使用 java 调用 dll文件 的情况&#xff0c;此时大概率出现UnsatisfiedLinkError链接错误&#xff0c;原因可能有如下几种 类名错误包名错误方法名参数错误使用 JNI 协议调用&#xff0c;结果 dll 未实现 JNI 协…...

智能在线客服平台:数字化时代企业连接用户的 AI 中枢

随着互联网技术的飞速发展&#xff0c;消费者期望能够随时随地与企业进行交流。在线客服平台作为连接企业与客户的重要桥梁&#xff0c;不仅优化了客户体验&#xff0c;还提升了企业的服务效率和市场竞争力。本文将探讨在线客服平台的重要性、技术进展、实际应用&#xff0c;并…...

OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别

OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别 直接训练提示词嵌入向量的核心区别 您提到的代码: prompt_embedding = initial_embedding.clone().requires_grad_(True) optimizer = torch.optim.Adam([prompt_embedding...

排序算法总结(C++)

目录 一、稳定性二、排序算法选择、冒泡、插入排序归并排序随机快速排序堆排序基数排序计数排序 三、总结 一、稳定性 排序算法的稳定性是指&#xff1a;同样大小的样本 **&#xff08;同样大小的数据&#xff09;**在排序之后不会改变原始的相对次序。 稳定性对基础类型对象…...

AxureRP-Pro-Beta-Setup_114413.exe (6.0.0.2887)

Name&#xff1a;3ddown Serial&#xff1a;FiCGEezgdGoYILo8U/2MFyCWj0jZoJc/sziRRj2/ENvtEq7w1RH97k5MWctqVHA 注册用户名&#xff1a;Axure 序列号&#xff1a;8t3Yk/zu4cX601/seX6wBZgYRVj/lkC2PICCdO4sFKCCLx8mcCnccoylVb40lP...