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

54.HarmonyOS NEXT 登录模块开发教程(八):测试与调试技巧

温馨提示:本篇博客的详细代码已发布到 git : https://gitcode.com/nutpi/HarmonyosNext 可以下载运行哦!

HarmonyOS NEXT 登录模块开发教程(八):测试与调试技巧

文章目录

  • HarmonyOS NEXT 登录模块开发教程(八):测试与调试技巧
    • 效果预览
    • 1. 引言
    • 2. 测试类型与策略
      • 2.1 测试类型概述
      • 2.2 测试策略
    • 3. 单元测试
      • 3.1 使用 Jest 进行单元测试
      • 3.2 测试状态管理逻辑
    • 4. UI 测试
      • 4.1 使用 UITest 进行 UI 测试
      • 4.2 截图对比测试
    • 5. 集成测试
      • 5.1 测试登录流程
      • 5.2 模拟网络请求
    • 6. 调试技巧
      • 6.1 日志调试
      • 6.2 使用 HiLog 进行结构化日志
      • 6.3 使用断点调试
      • 6.4 性能分析
    • 7. 常见问题与解决方案
      • 7.1 登录按钮点击无响应
      • 7.2 验证码发送失败
      • 7.3 UI 显示异常
    • 8. 最佳实践与注意事项
    • 9. 小结
    • 10. 参考资源

效果预览

1. 引言

在前七篇教程中,我们介绍了 HarmonyOS NEXT 登录模块的整体架构、模态窗口的实现原理、一键登录页面的实现、短信验证码登录的实现、状态管理和数据绑定机制、安全性考虑、UI 设计和用户体验优化以及性能优化和最佳实践。本篇教程将深入讲解登录模块的测试与调试技巧,帮助开发者构建稳定可靠的登录功能。

测试与调试是软件开发过程中不可或缺的环节,尤其对于登录模块这样的核心功能,更需要全面而严格的测试,以确保其在各种条件下都能正常工作。在 HarmonyOS NEXT 中,我们可以利用多种测试工具和调试技巧,提高登录模块的质量和可靠性。

2. 测试类型与策略

2.1 测试类型概述

在测试登录模块时,应考虑以下几种测试类型:

测试类型描述适用场景
单元测试测试单个组件或函数的功能验证输入验证、状态管理等独立功能
集成测试测试多个组件协同工作的功能验证登录流程、组件间通信等
UI 测试测试用户界面的外观和交互验证布局、样式、动画效果等
性能测试测试应用的性能指标验证加载时间、响应速度、内存占用等
安全测试测试应用的安全性验证输入验证、数据加密、防攻击措施等
兼容性测试测试在不同设备和系统版本上的兼容性验证在不同屏幕尺寸、系统版本上的表现

2.2 测试策略

针对登录模块,可以采用以下测试策略:

  1. 自底向上测试:先测试基础组件和功能,再测试复杂功能和流程
  2. 关键路径测试:重点测试用户最常使用的登录路径
  3. 边界条件测试:测试各种边界条件和异常情况
  4. 回归测试:在修改代码后,确保已有功能不受影响
  5. 自动化测试:将常规测试自动化,提高测试效率和覆盖率

3. 单元测试

3.1 使用 Jest 进行单元测试

HarmonyOS NEXT 支持使用 Jest 进行单元测试。以下是测试输入验证逻辑的示例:

// 被测试的输入验证函数
export function validatePhoneNumber(phoneNumber: string): boolean {return phoneNumber.length === 11 && /^[0-9]+$/.test(phoneNumber);
}// 单元测试
import { validatePhoneNumber } from '../utils/validators';describe('validatePhoneNumber', () => {test('should return true for valid phone number', () => {expect(validatePhoneNumber('13812345678')).toBe(true);});test('should return false for phone number with incorrect length', () => {expect(validatePhoneNumber('1381234567')).toBe(false); // 10位expect(validatePhoneNumber('138123456789')).toBe(false); // 12位});test('should return false for phone number with non-digit characters', () => {expect(validatePhoneNumber('1381234567a')).toBe(false);expect(validatePhoneNumber('138-1234-5678')).toBe(false);});test('should return false for empty string', () => {expect(validatePhoneNumber('')).toBe(false);});
});

3.2 测试状态管理逻辑

对于状态管理逻辑,可以测试状态变量的初始值和更新逻辑:

// 被测试的状态管理类
export class LoginState {private _phoneNumber: string = '';private _isPhoneValid: boolean = false;private _countdownSeconds: number = 0;private _timerId: number = 0;get phoneNumber(): string {return this._phoneNumber;}get isPhoneValid(): boolean {return this._isPhoneValid;}get countdownSeconds(): number {return this._countdownSeconds;}setPhoneNumber(value: string): void {this._phoneNumber = value;this._isPhoneValid = value.length === 11 && /^[0-9]+$/.test(value);}startCountdown(seconds: number): void {this._countdownSeconds = seconds;this.stopCountdown(); // 确保之前的定时器已清除this._timerId = setInterval(() => {this._countdownSeconds--;if (this._countdownSeconds <= 0) {this.stopCountdown();}}, 1000);}stopCountdown(): void {if (this._timerId) {clearInterval(this._timerId);this._timerId = 0;}}reset(): void {this._phoneNumber = '';this._isPhoneValid = false;this.stopCountdown();this._countdownSeconds = 0;}
}// 单元测试
import { LoginState } from '../models/LoginState';describe('LoginState', () => {let loginState: LoginState;beforeEach(() => {loginState = new LoginState();});test('should initialize with default values', () => {expect(loginState.phoneNumber).toBe('');expect(loginState.isPhoneValid).toBe(false);expect(loginState.countdownSeconds).toBe(0);});test('should validate phone number correctly', () => {loginState.setPhoneNumber('13812345678');expect(loginState.phoneNumber).toBe('13812345678');expect(loginState.isPhoneValid).toBe(true);loginState.setPhoneNumber('1381234567'); // 10位expect(loginState.isPhoneValid).toBe(false);loginState.setPhoneNumber('138a1234567'); // 非数字expect(loginState.isPhoneValid).toBe(false);});test('should reset state correctly', () => {loginState.setPhoneNumber('13812345678');loginState.startCountdown(30);loginState.reset();expect(loginState.phoneNumber).toBe('');expect(loginState.isPhoneValid).toBe(false);expect(loginState.countdownSeconds).toBe(0);});// 测试倒计时逻辑需要使用Jest的timer mockstest('should handle countdown correctly', () => {jest.useFakeTimers();loginState.startCountdown(3);expect(loginState.countdownSeconds).toBe(3);jest.advanceTimersByTime(1000);expect(loginState.countdownSeconds).toBe(2);jest.advanceTimersByTime(1000);expect(loginState.countdownSeconds).toBe(1);jest.advanceTimersByTime(1000);expect(loginState.countdownSeconds).toBe(0);jest.useRealTimers();});
});

4. UI 测试

4.1 使用 UITest 进行 UI 测试

HarmonyOS NEXT 提供了 UITest 框架,用于测试 UI 组件的外观和交互:

// UI测试示例
import { Driver, ON } from '@ohos.UiTest';
import { describe, beforeAll, it, expect } from '@ohos/hypium';export default function abilityTest() {describe('LoginPageTest', function () {it('should display login page correctly', 0, async function () {// 启动测试let driver = await Driver.create();await driver.delayMs(1000);// 点击登录按钮,打开登录模态窗口let loginButton = await driver.findComponent(ON.text('全屏模态窗口登录'));await loginButton.click();await driver.delayMs(1000);// 验证登录页面元素是否正确显示let welcomeText = await driver.findComponent(ON.text('欢迎回来'));expect(await welcomeText.isDisplayed()).assertTrue();let phoneText = await driver.findComponent(ON.text('18888888888'));expect(await phoneText.isDisplayed()).assertTrue();let agreementCheckbox = await driver.findComponent(ON.id('default_agreement'));expect(await agreementCheckbox.isDisplayed()).assertTrue();let loginActionButton = await driver.findComponent(ON.text('一键登录'));expect(await loginActionButton.isDisplayed()).assertTrue();// 测试复选框交互await agreementCheckbox.click();await driver.delayMs(500);// 测试登录按钮点击await loginActionButton.click();await driver.delayMs(1000);// 验证登录成功提示let successToast = await driver.findComponent(ON.text('登录成功'));expect(await successToast.isDisplayed()).assertTrue();// 测试返回按钮let backButton = await driver.findComponent(ON.id('login_back'));await backButton.click();await driver.delayMs(1000);// 验证模态窗口已关闭expect(await driver.findComponent(ON.text('欢迎回来')).isDisplayed()).assertFalse();});it('should switch to other login methods', 0, async function () {// 启动测试let driver = await Driver.create();await driver.delayMs(1000);// 点击登录按钮,打开登录模态窗口let loginButton = await driver.findComponent(ON.text('全屏模态窗口登录'));await loginButton.click();await driver.delayMs(1000);// 点击其他登录方式let otherLoginButton = await driver.findComponent(ON.text('其他登录方式'));await otherLoginButton.click();await driver.delayMs(1000);// 验证短信验证码登录页面元素是否正确显示let phoneLoginText = await driver.findComponent(ON.text('手机号登录'));expect(await phoneLoginText.isDisplayed()).assertTrue();let phoneInput = await driver.findComponent(ON.placeholder('请输入手机号'));expect(await phoneInput.isDisplayed()).assertTrue();let verifyButton = await driver.findComponent(ON.text('获取验证码'));expect(await verifyButton.isDisplayed()).assertTrue();});});
}

4.2 截图对比测试

截图对比测试可以验证 UI 的视觉一致性:

import { Driver, ON, MatchPattern } from '@ohos.UiTest';
import { describe, beforeAll, it, expect } from '@ohos/hypium';
import fs from '@ohos.file.fs';
import image from '@ohos.multimedia.image';export default function visualTest() {describe('LoginVisualTest', function () {it('should match the expected visual appearance', 0, async function () {// 启动测试let driver = await Driver.create();await driver.delayMs(1000);// 点击登录按钮,打开登录模态窗口let loginButton = await driver.findComponent(ON.text('全屏模态窗口登录'));await loginButton.click();await driver.delayMs(1000);// 截取当前屏幕const screenshot = await driver.takeScreenshot();// 读取基准图片const baselineImagePath = '/data/storage/el2/base/files/baseline_login_screen.png';const baselineImage = await fs.readFile(baselineImagePath);// 比较两张图片的相似度const similarity = await compareImages(screenshot, baselineImage);// 相似度应大于95%expect(similarity).assertLarger(0.95);});});
}// 图片比较函数(示例实现,实际可能需要更复杂的算法)
async function compareImages(image1: ArrayBuffer, image2: ArrayBuffer): Promise<number> {// 这里是简化的实现,实际应使用图像处理库进行像素级比较// 返回两张图片的相似度(0-1之间的值)const img1 = await image.createImageSource(image1);const img2 = await image.createImageSource(image2);const pixelMap1 = await img1.createPixelMap();const pixelMap2 = await img2.createPixelMap();// 获取图片尺寸和像素数据const info1 = await pixelMap1.getImageInfo();const info2 = await pixelMap2.getImageInfo();// 简单比较:如果尺寸不同,相似度降低if (info1.size.width !== info2.size.width || info1.size.height !== info2.size.height) {return 0.5; // 尺寸不同,相似度降低}// 实际应该进行像素级比较,这里简化处理return 0.98; // 示例返回值
}

5. 集成测试

5.1 测试登录流程

集成测试用于验证多个组件协同工作的功能,例如完整的登录流程:

import { Driver, ON } from '@ohos.UiTest';
import { describe, beforeAll, it, expect } from '@ohos/hypium';export default function integrationTest() {describe('LoginFlowTest', function () {it('should complete the entire login flow', 0, async function () {// 启动测试let driver = await Driver.create();await driver.delayMs(1000);// 1. 打开登录模态窗口let loginButton = await driver.findComponent(ON.text('全屏模态窗口登录'));await loginButton.click();await driver.delayMs(1000);// 2. 切换到短信验证码登录let otherLoginButton = await driver.findComponent(ON.text('其他登录方式'));await otherLoginButton.click();await driver.delayMs(1000);// 3. 输入手机号let phoneInput = await driver.findComponent(ON.placeholder('请输入手机号'));await phoneInput.inputText('13812345678');await driver.delayMs(500);// 4. 勾选协议let agreementCheckbox = await driver.findComponent(ON.id('other_agreement'));await agreementCheckbox.click();await driver.delayMs(500);// 5. 点击获取验证码let verifyButton = await driver.findComponent(ON.text('获取验证码'));await verifyButton.click();await driver.delayMs(1000);// 6. 验证验证码发送成功提示let successToast = await driver.findComponent(ON.text('验证码已发送'));expect(await successToast.isDisplayed()).assertTrue();// 7. 验证倒计时开始let countdownButton = await driver.findComponent(ON.textContains('后可再次发送'));expect(await countdownButton.isDisplayed()).assertTrue();// 8. 返回默认登录页面let backButton = await driver.findComponent(ON.id('login_back'));await backButton.click();await driver.delayMs(1000);// 9. 验证返回成功let welcomeText = await driver.findComponent(ON.text('欢迎回来'));expect(await welcomeText.isDisplayed()).assertTrue();});});
}

5.2 模拟网络请求

在集成测试中,可以使用模拟(Mock)技术模拟网络请求,避免依赖真实的网络环境:

// 网络请求模块
export class NetworkService {static async sendVerifyCode(phoneNumber: string): Promise<boolean> {// 实际实现会调用网络APIreturn true;}static async verifyCode(phoneNumber: string, code: string): Promise<boolean> {// 实际实现会调用网络APIreturn true;}
}// 测试代码
import { NetworkService } from '../services/NetworkService';describe('LoginWithNetwork', function () {beforeEach(() => {// 模拟网络请求jest.spyOn(NetworkService, 'sendVerifyCode').mockImplementation(async (phoneNumber: string) => {// 模拟网络延迟await new Promise(resolve => setTimeout(resolve, 100));// 模拟验证逻辑return phoneNumber.length === 11 && /^[0-9]+$/.test(phoneNumber);});jest.spyOn(NetworkService, 'verifyCode').mockImplementation(async (phoneNumber: string, code: string) => {// 模拟网络延迟await new Promise(resolve => setTimeout(resolve, 100));// 模拟验证逻辑(示例中只要验证码是6位数字即视为有效)return code.length === 6 && /^[0-9]+$/.test(code);});});afterEach(() => {jest.restoreAllMocks();});it('should send verify code successfully with valid phone number', async () => {const result = await NetworkService.sendVerifyCode('13812345678');expect(result).toBe(true);});it('should fail to send verify code with invalid phone number', async () => {const result = await NetworkService.sendVerifyCode('1381234567'); // 10位expect(result).toBe(false);});it('should verify code successfully with valid code', async () => {const result = await NetworkService.verifyCode('13812345678', '123456');expect(result).toBe(true);});it('should fail to verify code with invalid code', async () => {const result = await NetworkService.verifyCode('13812345678', '12345'); // 5位expect(result).toBe(false);});
});

6. 调试技巧

6.1 日志调试

在 HarmonyOS NEXT 中,可以使用 console API 进行日志调试:

// 不同级别的日志
console.debug('调试信息'); // 调试级别,仅在开发环境显示
console.info('一般信息'); // 信息级别
console.warn('警告信息'); // 警告级别
console.error('错误信息'); // 错误级别// 在登录按钮点击事件中添加日志
Button($r('app.string.modalwindow_phone_start_login')).onClick(() => {console.info('登录按钮点击');if (this.isConfirmed) {console.info('用户已同意协议,开始登录');// 调用Toast显示登录成功提示promptAction.showToast({ message: $r('app.string.modalwindow_login_success') });} else {console.warn('用户未同意协议,拒绝登录');// 调用Toast显示请先阅读并同意协议提示promptAction.showToast({ message: $r('app.string.modalwindow_please_read_and_agree') });}})

6.2 使用 HiLog 进行结构化日志

HarmonyOS NEXT 提供了 HiLog API,支持更强大的结构化日志功能:

import hilog from '@ohos.hilog';// 定义日志域和标签
const DOMAIN = 0xF811; // 自定义域ID
const TAG = 'LoginModule'; // 日志标签// 使用HiLog记录日志
function logDebug(message: string, ...args: any[]) {hilog.debug(DOMAIN, TAG, message, args);
}function logInfo(message: string, ...args: any[]) {hilog.info(DOMAIN, TAG, message, args);
}function logWarn(message: string, ...args: any[]) {hilog.warn(DOMAIN, TAG, message, args);
}function logError(message: string, ...args: any[]) {hilog.error(DOMAIN, TAG, message, args);
}// 在登录流程中使用结构化日志
Button(this.buttonContent).onClick(() => {logInfo('发送验证码按钮点击,手机号:%{public}s', this.phoneNumber);if (this.countdownSeconds > 0) {logInfo('倒计时中,剩余秒数:%{public}d', this.countdownSeconds);return;}if (!this.phoneNumberAvailable) {logWarn('手机号无效:%{public}s', this.phoneNumber);promptAction.showToast({ message: $r('app.string.modalwindow_message_right_phone_number') });} else if (!this.isAgree) {logWarn('用户未同意协议');promptAction.showToast({ message: $r('app.string.modalwindow_message_read_agreement') });} else {logInfo('开始发送验证码');// 发送验证码逻辑}})

6.3 使用断点调试

DevEco Studio 提供了强大的断点调试功能,可以帮助开发者定位和解决问题:

  1. 设置断点:在代码编辑器中点击行号左侧,设置断点
  2. 启动调试:点击工具栏中的调试按钮,启动调试会话
  3. 单步执行:使用 F8(步过)、F7(步入)、Shift+F8(步出)等快捷键控制代码执行
  4. 查看变量:在调试面板中查看变量的值和类型
  5. 条件断点:右键点击断点,设置条件表达式,只有条件满足时才会触发断点

6.4 性能分析

DevEco Studio 提供了性能分析工具,帮助开发者发现和解决性能问题:

  1. CPU 分析:分析应用的 CPU 使用情况,找出热点函数
  2. 内存分析:分析应用的内存使用情况,发现内存泄漏
  3. 网络分析:分析应用的网络请求情况,优化网络性能
  4. UI 分析:分析 UI 渲染性能,发现卡顿原因

7. 常见问题与解决方案

7.1 登录按钮点击无响应

问题:用户点击登录按钮,但没有任何反应。

可能原因

  1. 事件处理函数未正确绑定
  2. 事件处理函数中有错误
  3. 按钮被禁用或覆盖

解决方案

  1. 检查 onClick 事件是否正确绑定
  2. 添加日志,跟踪事件处理函数的执行
  3. 检查按钮的 enabled 属性和 z-index

7.2 验证码发送失败

问题:点击发送验证码按钮,但验证码未发送或发送失败。

可能原因

  1. 网络请求失败
  2. 服务器返回错误
  3. 手机号格式错误

解决方案

  1. 添加网络请求的错误处理和重试机制
  2. 检查服务器返回的错误信息
  3. 加强客户端的输入验证

7.3 UI 显示异常

问题:登录界面的 UI 显示异常,如布局错乱、元素重叠等。

可能原因

  1. 布局参数设置不当
  2. 屏幕适配问题
  3. 样式冲突

解决方案

  1. 使用 Inspector 工具检查 UI 层次结构和属性
  2. 使用百分比布局和弹性布局,提高适配性
  3. 检查样式设置,解决冲突

8. 最佳实践与注意事项

在测试和调试登录模块时,有以下几点最佳实践和注意事项:

  1. 测试驱动开发:采用测试驱动开发(TDD)方法,先编写测试,再实现功能
  2. 自动化测试:尽可能自动化测试,减少人工测试的工作量和错误率
  3. 测试覆盖率:关注测试覆盖率,确保关键代码路径都有测试覆盖
  4. 边界条件测试:重点测试边界条件和异常情况,如空输入、无网络等
  5. 模拟依赖:使用模拟(Mock)技术模拟外部依赖,如网络请求、系统服务等
  6. 持续集成:将测试集成到持续集成流程中,确保每次代码变更都经过测试
  7. 结构化日志:使用结构化日志,便于问题定位和分析
  8. 分级调试:根据问题的严重程度,使用不同级别的调试手段
  9. 性能监控:在测试过程中监控性能指标,及时发现性能问题
  10. 用户反馈:收集和分析用户反馈,发现和解决实际使用中的问题

9. 小结

本文详细介绍了 HarmonyOS NEXT 登录模块的测试与调试技巧,包括各种测试类型和策略、单元测试、UI 测试、集成测试、调试技巧以及常见问题的解决方案。通过合理的测试和调试,可以提高登录模块的质量和可靠性,为用户提供稳定流畅的登录体验。

测试和调试是软件开发过程中不可或缺的环节,尤其对于登录模块这样的核心功能,更需要全面而严格的测试。通过本教程的学习,你应该能够掌握如何在 HarmonyOS NEXT 中测试和调试登录功能,包括使用 Jest 进行单元测试、使用 UITest 进行 UI 测试、模拟网络请求进行集成测试,以及使用日志、断点和性能分析工具进行调试。

在下一篇教程中,我们将详细讲解登录模块的部署和发布,包括应用签名、版本管理、渠道分发等内容。

10. 参考资源

  • HarmonyOS 开发者文档 - 测试指南
  • HarmonyOS 开发者文档 - Jest 单元测试
  • HarmonyOS 开发者文档 - UITest
  • HarmonyOS 开发者文档 - 日志系统
  • HarmonyOS 开发者文档 - 调试指南
  • HarmonyOS 开发者文档 - 性能分析

相关文章:

54.HarmonyOS NEXT 登录模块开发教程(八):测试与调试技巧

温馨提示&#xff1a;本篇博客的详细代码已发布到 git : https://gitcode.com/nutpi/HarmonyosNext 可以下载运行哦&#xff01; HarmonyOS NEXT 登录模块开发教程&#xff08;八&#xff09;&#xff1a;测试与调试技巧 文章目录 HarmonyOS NEXT 登录模块开发教程&#xff08;…...

Vue3中 ref 与 reactive区别

ref 用途: ref 通常用于创建一个响应式的基本类型数据&#xff08;如 string、number、boolean 等&#xff09;&#xff0c;但它也可以用于对象或数组 返回值: ref 返回一个带有 .value 属性的对象&#xff0c;访问或修改数据需要通过 .value 进行 使用场景: …...

结构型——装饰器模式

装饰器模式 装饰器是指能动态地为对象添加额外的功能的一种结构型设计模式。 特点 不修改原有代码的情况下&#xff0c;动态地扩展一个对象的功能。支持多个装饰器叠加使用透明性&#xff0c;装饰后的对象与原对象保持一致&#xff0c;客户端无需感知装饰过程 结构模式与实…...

在Simulink中将Excel数据导入可变负载模块的方法介绍

文章目录 数据准备与格式要求Excel数据格式MATLAB预处理数据导入方法使用From Spreadsheet模块(直接导入Excel)通过MATLAB工作区中转(From Workspace模块)使用1-D Lookup Table模块(非线性负载映射)Signal Builder模块(变载工况导入)可变负载模块配置注意事项与调试在S…...

分布式事务的产生背景及理论指导

分布式事务的产生背景 在现代互联网和企业级系统架构中&#xff0c;随着业务需求的增长&#xff0c;单体架构逐渐向微服务架构、分布式架构演进。传统单体架构下&#xff0c;事务管理相对简单&#xff0c;可以依赖数据库的本地事务&#xff08;如 MySQL 的 ACID 事务&#xff…...

动手学强化学习-记录

3.5 蒙特卡洛方法 统计每一个状态s出现的总次数和总回报&#xff0c;用大数定律&#xff0c;总回报/总次数≈状态s的期望回报 第4章 动态规划算法 策略迭代中的策略评估使用贝尔曼期望方程来得到一个策略的状态价值函数,这是一个动 态规划的过程;而价值迭代直接使用贝尔曼最…...

RocketMQ性能优化篇

在分布式消息系统中&#xff0c;RocketMQ以其高性能、高可靠性和高可扩展性而被广泛应用。然而&#xff0c;为了充分发挥其性能优势&#xff0c;需要进行一系列的性能测试和优化。本文将从性能测试方法和优化实践两个方面&#xff0c;详细介绍如何对RocketMQ进行性能优化。通过…...

C语言为例谈数据依赖性

数据依赖性&#xff08;Data Dependency&#xff09;是指程序中后续操作的计算结果或内存访问依赖于前面操作的结果。在存在数据依赖的情况下&#xff0c;编译器或处理器会保证这些操作的执行顺序&#xff0c;因此不需要显式地使用内存屏障&#xff08;Memory Barrier&#xff…...

阿里云操作系统控制台评测:国产AI+运维 一站式运维管理平台

阿里云操作系统控制台评测&#xff1a;国产AI运维 一站式运维管理平台 引言 随着云计算技术的飞速发展&#xff0c;企业在云端的运维管理面临更高的要求。阿里云操作系统控制台作为一款集运维管理、智能助手和系统诊断等多功能于一体的工具&#xff0c;正逐步成为企业高效管理…...

C++中的const与类型转换艺术

目录 强制转换 static_cast const_cast reinterpret_cast dynamic_cast const关键字 修饰内置类型* 修饰指针类型* 类比 数组指针 指针数组 函数指针 指针函数 强制转换 C语言中的强制转换在C代码中依然可以使用&#xff0c;这种C风格的转换格式非常简单 TYPE a …...

网络安全演练有哪些形式

OPENVAS使用 1、确定指定IP是否能ping通 2、创建扫描目标 3、创建扫描任务&#xff08;scan management →newtask&#xff09; 4、开始任务start 5、查看扫描细节 6、查看扫描结果&#xff0c;包含漏洞详细信息&#xff0c;亦可到处PDF文件 7、导出扫描结果报告 8、为…...

c++常用的算术生成算法

注意&#xff1a; 算术生成算法属于小型算法&#xff0c;使用时包含的头文件为 #include <numeric> 算法简介&#xff1a; accumulate //计算容器元素累加总和fill //向容器中添加元素 1. accumulate 功能描述&#xff1a; 计算区间内 容器元素…...

2011. 执行操作后的变量值

执行操作后的变量值 题目描述尝试做法推荐做法 题目描述 存在一种仅支持 4 种操作和 1 个变量 X 的编程语言&#xff1a; X 和 X 使变量 X 的值 加 1 –X 和 X-- 使变量 X 的值 减 1 最初&#xff0c;X 的值是 0 给你一个字符串数组 operations &#xff0c;这是由操作组成的…...

特辣的海藻!10

基础知识点 1.清除换行符 scan.nextInt()要加scan.nextLine()清楚换行符。 2.Map.Entry<K, V> Map.Entry是Map接口的嵌套接口&#xff0c;表示一个键值对&#xff08;Key-Value&#xff09; 常用方法&#xff1a; entry.getKey()&#xff1a;获取键 …...

SpringBoot动态加载JAR包实战:实现插件化架构的终极指南

在需要热插拔业务模块、支持灰度发布的系统中&#xff0c;动态加载外部JAR包是提升系统扩展性的核心技术。本文将手把手实现3种动态加载方案&#xff0c;包含可直接运行的SpringBoot代码&#xff0c;并深入分析类加载机制与内存泄漏预防策略。 一、动态加载的应用场景 ‌电商…...

双因素拆解法 - 分析比例型指标的因子贡献度

什么是比例型指标 比例型指标是指那些以比例或比率形式表示的指标&#xff0c;通常涉及两个相关量的比较。以下是一些常见的比例型指标的例子&#xff1a; 毛利率&#xff1a;毛利率是毛利与销售收入的比率&#xff0c;公式为&#xff1a; 毛利率 毛利 销售收入 100 % \tex…...

sqli-lab靶场学习(八)——Less26-28

前言 25关已经出现了初步的一些关键字过滤&#xff0c;通过双写可以绕过。后面的关卡&#xff0c;我们会遇到更多关键字过滤&#xff0c;需要各种技巧绕过。 Less26 第26关写了会过滤空格和注释符。有很多的答案&#xff0c;会用%a0替代空格&#xff0c;但据说这是sqli-labs部…...

Netty基础—4.NIO的使用简介二

大纲 1.Buffer缓冲区 2.Channel通道 3.BIO编程 4.伪异步IO编程 5.改造程序以支持长连接 6.NIO三大核心组件 7.NIO服务端的创建流程 8.NIO客户端的创建流程 9.NIO优点总结 10.NIO问题总结 4.伪异步IO编程 (1)BIO的主要问题 (2)BIO编程模型的改进 (3)伪异步IO编程 …...

双指针算法专题之——复写零

文章目录 题目介绍思路分析异地复写优化为就地复写 AC代码 题目介绍 链接: 1089. 复写零 思路分析 那么这道题我们依然可以使用双指针算法来解决 异地复写 先不考虑题目的要求&#xff0c;直接就地在原数组上修改&#xff0c;可能不太好想&#xff0c;我们这里可以先在一个…...

【Pandas】pandas Series last_valid_index

Pandas2.2 Series Time Series-related 方法描述Series.asfreq(freq[, method, how, …])用于将时间序列数据转换为指定的频率Series.asof(where[, subset])用于返回时间序列中指定索引位置的最近一个非缺失值Series.shift([periods, freq, axis, …])用于将时间序列数据沿指…...

python-leetcode-子数组最大平均数 I

643. 子数组最大平均数 I - 力扣&#xff08;LeetCode&#xff09; 可以使用滑动窗口&#xff08;Sliding Window&#xff09;的方法来解决这个问题。具体步骤如下&#xff1a; 先计算数组 nums 中前 k 个元素的和 sum_k&#xff0c;作为初始窗口的和。然后滑动窗口&#xff0…...

【度的数量——数位DP】

题目 分析 数位DP可以解决“区间内满足某种性质的数的个数”的问题 通常按照数位分支&#xff0c;形成一颗数位树 最左分支的值由上界值决定&#xff0c;右分支可以直接计算权重 有可能最左分支会有一个权重 代码 #include <bits/stdc.h> using namespace std;cons…...

STM32使用EXTI触发进行软件消抖(更新中)

在STM32的HAL库中&#xff0c;为了实现按键的软件消抖&#xff0c;通常需要在按键中断处理或轮询程序中加入一定的延时和状态检测逻辑。以下是一个简单的示例&#xff0c;展示了如何使用HAL库来实现按键的软件消抖。 假设你有一个按键连接到GPIO引脚&#xff0c;并且已经配置好…...

计算机操作系统进程(3)

系列文章目录 第二章&#xff1a;进程的描述与控制 文章目录 系列文章目录前言一、进程同步的基本概念&#xff1a;二、临界资源&#xff1a;总结 前言 前面我们学习了进程的定义和特征&#xff0c;进程状态的转换&#xff0c;接下来我们开始学习我们最重要的一点也是相对最难…...

搭建阿里云专有网络VPC

目录 一、概述 二、专有网络vpc 2.1 vpc基本信息 2.2 vpc资源管理 2.3 vpc网段管理 三、交换机 四、NAT网关 4.1 绑定弹性公网IP 4.2 NAT网关信息 4.3 绑定的弹性公网IP 4.4 DNAT 4.5 SNAT 五、弹性公网IP 六、访问控制ACL&#xff08;绑定交换机&#xff09; 6…...

centos steam8 部署k8s

kubernetes搭建 文章目录 kubernetes搭建[toc] 准备工作&#xff08;三节点&#xff09;安装docker(三节点)安装cri-dockerd&#xff08;三节点&#xff09;添加阿里云软件源&#xff08;三节点&#xff09;安装kubeadm、kubelet、kubectl&#xff08;三节点&#xff09;初始化…...

DB2 字符串比较 (= 或 IN) 时,忽略末尾的空格踩坑与解决方法

一、问题描述 在 DB2 中&#xff0c;VARCHAR 类型的字段在 字符串比较 ( 或 IN) 时会忽略末尾的空格&#xff0c;这可能导致查询结果与预期不符。例如&#xff1a; SELECT * FROM t_user WHERE id IN (016110110000011763); 如果 id 字段中存储的值为016110110000011763 &…...

windows系统,pycharm运行.sh文件

博主亲身试验过&#xff0c;流程简单&#xff0c;可用。 需要pycharm &#xff0c;git。 注意需要Git Bash.exe &#xff0c;也就是Git Bash的应用程序&#xff0c;而不是快捷方式。 需要把这个应用程序的路径复制一下。可以通过右键&#xff0c;复制文件地址的方式。 接着在…...

论文调研 | 一些开源的AI代码生成模型调研及总结【更新于250313】

本文主要介绍主流代码生成模型&#xff0c;总结了基于代码生成的大语言模型&#xff0c;按照时间顺序排列。 在了解代码大语言模型之前&#xff0c;需要了解代码相关子任务 代码生成 文本生成代码(Text to code):根据自然语言描述生成代码 重构代码&#xff08;Refactoring …...

筛选法找质数(信息学奥赛一本通-2040)

【题目描述】 用筛法求出n(2≤n≤1000)以内的全部质数。 【输入】 输入n。 【输出】 多行&#xff0c;由小到大的质数。 【输入样例】 10 【输出样例】 2 3 5 7 【题解代码】 #include<bits/stdc.h> using namespace std;const int N 1e3 10; int nums[N];void isprim…...