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

(二十五)安卓开发一个完整的登录页面-支持密码登录和手机验证码登录

下面将详细讲解如何在 Android 中开发一个完整的登录页面,支持密码登录和手机验证码登录。以下是实现过程的详细步骤,从布局设计到逻辑实现,再到用户体验优化,逐步展开。


1. 设计登录页面布局

首先,我们需要设计一个用户友好的登录页面布局,包含以下核心元素:

  • 用户名/手机号输入框:用于输入用户名或手机号。
  • 密码输入框:用于密码登录。
  • 验证码输入框:用于手机验证码登录,默认隐藏。
  • 登录按钮:触发登录操作。
  • 切换登录方式按钮:在密码登录和验证码登录之间切换。
  • 发送验证码按钮:发送验证码到用户手机,默认隐藏。

我们可以使用 LinearLayoutConstraintLayout 来布局这些元素。以下是一个简单的 XML 布局示例:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"android:padding="16dp"><EditTextandroid:id="@+id/et_username"android:layout_width="match_parent"android:layout_height="wrap_content"android:hint="用户名/手机号" /><EditTextandroid:id="@+id/et_password"android:layout_width="match_parent"android:layout_height="wrap_content"android:hint="密码"android:inputType="textPassword" /><EditTextandroid:id="@+id/et_verification_code"android:layout_width="match_parent"android:layout_height="wrap_content"android:hint="验证码"android:visibility="gone" /><Buttonandroid:id="@+id/btn_send_verification_code"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="发送验证码"android:visibility="gone" /><Buttonandroid:id="@+id/btn_login"android:layout_width="match_parent"android:layout_height="wrap_content"android:text="登录" /><Buttonandroid:id="@+id/btn_switch_login_type"android:layout_width="match_parent"android:layout_height="wrap_content"android:text="切换到手机验证码登录" /></LinearLayout>

说明

  • android:visibility="gone" 表示验证码输入框和发送验证码按钮默认隐藏。
  • android:inputType="textPassword" 确保密码输入框显示为密码格式。

2. 处理登录逻辑

接下来,我们在 Activity 中实现登录逻辑。这里以 Kotlin 为例,使用 AppCompatActivity 实现。

初始化视图和监听器
class LoginActivity : AppCompatActivity() {private lateinit var etUsername: EditTextprivate lateinit var etPassword: EditTextprivate lateinit var etVerificationCode: EditTextprivate lateinit var btnSendVerificationCode: Buttonprivate lateinit var btnLogin: Buttonprivate lateinit var btnSwitchLoginType: Buttonprivate var isPasswordLogin = true // 默认使用密码登录override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_login)// 初始化视图etUsername = findViewById(R.id.et_username)etPassword = findViewById(R.id.et_password)etVerificationCode = findViewById(R.id.et_verification_code)btnSendVerificationCode = findViewById(R.id.btn_send_verification_code)btnLogin = findViewById(R.id.btn_login)btnSwitchLoginType = findViewById(R.id.btn_switch_login_type)// 设置登录按钮点击事件btnLogin.setOnClickListener {if (isPasswordLogin) {performPasswordLogin()} else {performVerificationCodeLogin()}}// 设置切换登录方式按钮点击事件btnSwitchLoginType.setOnClickListener {switchLoginType()}// 设置发送验证码按钮点击事件btnSendVerificationCode.setOnClickListener {val phoneNumber = etUsername.text.toString()sendVerificationCode(phoneNumber)}}
}

说明

  • isPasswordLogin 是一个布尔变量,用于跟踪当前登录方式。
  • 点击登录按钮时,根据 isPasswordLogin 的值决定执行密码登录还是验证码登录。
切换登录方式
private fun switchLoginType() {if (isPasswordLogin) {// 切换到手机验证码登录etPassword.visibility = View.GONEetVerificationCode.visibility = View.VISIBLEbtnSendVerificationCode.visibility = View.VISIBLEbtnSwitchLoginType.text = "切换到密码登录"isPasswordLogin = false} else {// 切换到密码登录etPassword.visibility = View.VISIBLEetVerificationCode.visibility = View.GONEbtnSendVerificationCode.visibility = View.GONEbtnSwitchLoginType.text = "切换到手机验证码登录"isPasswordLogin = true}
}

说明

  • 通过改变视图的可见性(View.VISIBLEView.GONE)和按钮文本,实现登录方式的切换。

3. 实现密码登录

密码登录需要向服务器发送用户名和密码,并处理响应。我们使用 Retrofit 库进行网络请求。

定义 API 接口
interface ApiService {@POST("login")fun login(@Body request: LoginRequest): Call<LoginResponse>
}data class LoginRequest(val username: String, val password: String)
data class LoginResponse(val token: String)
实现密码登录逻辑
private fun performPasswordLogin() {val username = etUsername.text.toString()val password = etPassword.text.toString()val retrofit = Retrofit.Builder().baseUrl("https://your-api-url.com/") // 替换为实际 API 地址.addConverterFactory(GsonConverterFactory.create()).build()val apiService = retrofit.create(ApiService::class.java)val request = LoginRequest(username, password)apiService.login(request).enqueue(object : Callback<LoginResponse> {override fun onResponse(call: Call<LoginResponse>, response: Response<LoginResponse>) {if (response.isSuccessful) {val token = response.body()?.tokentoken?.let {saveToken(it)goToMainActivity()}} else {showError("登录失败,请检查用户名或密码")}}override fun onFailure(call: Call<LoginResponse>, t: Throwable) {showError("网络错误,请稍后重试")}})
}

说明

  • saveTokengoToMainActivity 用于保存 token 并跳转到主界面,具体实现见后文。
  • showError 用于显示错误提示。

4. 实现手机验证码登录

手机验证码登录分为两步:发送验证码和验证验证码。

定义 API 接口
interface ApiService {@POST("send_verification_code")fun sendVerificationCode(@Body request: SendCodeRequest): Call<SendCodeResponse>@POST("verify_code")fun verifyCode(@Body request: VerifyCodeRequest): Call<VerifyCodeResponse>
}data class SendCodeRequest(val phoneNumber: String)
data class SendCodeResponse(val success: Boolean)data class VerifyCodeRequest(val phoneNumber: String, val code: String)
data class VerifyCodeResponse(val token: String)
发送验证码
private fun sendVerificationCode(phoneNumber: String) {val retrofit = Retrofit.Builder().baseUrl("https://your-api-url.com/").addConverterFactory(GsonConverterFactory.create()).build()val apiService = retrofit.create(ApiService::class.java)val request = SendCodeRequest(phoneNumber)apiService.sendVerificationCode(request).enqueue(object : Callback<SendCodeResponse> {override fun onResponse(call: Call<SendCodeResponse>, response: Response<SendCodeResponse>) {if (response.isSuccessful && response.body()?.success == true) {Toast.makeText(this@LoginActivity, "验证码已发送", Toast.LENGTH_SHORT).show()startCountdown() // 开始倒计时} else {showError("验证码发送失败")}}override fun onFailure(call: Call<SendCodeResponse>, t: Throwable) {showError("网络错误,请稍后重试")}})
}
验证验证码
private fun performVerificationCodeLogin() {val phoneNumber = etUsername.text.toString()val verificationCode = etVerificationCode.text.toString()val retrofit = Retrofit.Builder().baseUrl("https://your-api-url.com/").addConverterFactory(GsonConverterFactory.create()).build()val apiService = retrofit.create(ApiService::class.java)val request = VerifyCodeRequest(phoneNumber, verificationCode)apiService.verifyCode(request).enqueue(object : Callback<VerifyCodeResponse> {override fun onResponse(call: Call<VerifyCodeResponse>, response: Response<VerifyCodeResponse>) {if (response.isSuccessful) {val token = response.body()?.tokentoken?.let {saveToken(it)goToMainActivity()}} else {showError("验证码错误")}}override fun onFailure(call: Call<VerifyCodeResponse>, t: Throwable) {showError("网络错误,请稍后重试")}})
}

5. 优化用户体验

为了提升用户体验,可以添加以下功能:

发送验证码倒计时
private fun startCountdown() {btnSendVerificationCode.isEnabled = falsevar remainingTime = 60 // 60秒倒计时val handler = Handler(Looper.getMainLooper())val runnable = object : Runnable {override fun run() {if (remainingTime > 0) {btnSendVerificationCode.text = "重新发送($remainingTime)"remainingTime--handler.postDelayed(this, 1000)} else {btnSendVerificationCode.text = "发送验证码"btnSendVerificationCode.isEnabled = true}}}handler.post(runnable)
}

说明

  • sendVerificationCode 成功后调用 startCountdown(),防止用户频繁发送验证码。
输入验证

可以在登录前检查输入是否合法,例如:

private fun isValidPhoneNumber(phone: String): Boolean {return phone.matches(Regex("^1[3-9]\\d{9}$")) // 简单验证中国手机号
}private fun performPasswordLogin() {val username = etUsername.text.toString()val password = etPassword.text.toString()if (username.isEmpty() || password.isEmpty()) {showError("用户名或密码不能为空")return}// 继续登录逻辑
}private fun performVerificationCodeLogin() {val phoneNumber = etUsername.text.toString()val code = etVerificationCode.text.toString()if (!isValidPhoneNumber(phoneNumber)) {showError("请输入有效的手机号")return}if (code.isEmpty()) {showError("请输入验证码")return}// 继续验证逻辑
}

6. 处理登录成功和失败

保存 Token 并跳转
private fun saveToken(token: String) {val sharedPreferences = getSharedPreferences("app_prefs", Context.MODE_PRIVATE)sharedPreferences.edit().putString("token", token).apply()
}private fun goToMainActivity() {val intent = Intent(this, MainActivity::class.java)startActivity(intent)finish()
}
显示错误提示
private fun showError(message: String) {Toast.makeText(this, message, Toast.LENGTH_SHORT).show()
}

7. 总结

通过以上步骤,我们实现了一个完整的登录页面,支持密码登录和手机验证码登录。主要包括:

  • 布局设计:使用 XML 创建用户界面。
  • 逻辑处理:通过 Kotlin 实现切换登录方式和网络请求。
  • 网络请求:使用 Retrofit 与服务器交互。
  • 用户体验:添加倒计时和输入验证。

在实际开发中,还需考虑:

  • 安全性:密码加密传输、验证码防刷。
  • 界面美化:使用 Material Design 组件。
  • 错误处理:更细致的网络错误和服务器响应处理。

相关文章:

(二十五)安卓开发一个完整的登录页面-支持密码登录和手机验证码登录

下面将详细讲解如何在 Android 中开发一个完整的登录页面&#xff0c;支持密码登录和手机验证码登录。以下是实现过程的详细步骤&#xff0c;从布局设计到逻辑实现&#xff0c;再到用户体验优化&#xff0c;逐步展开。 1. 设计登录页面布局 首先&#xff0c;我们需要设计一个用…...

【过程控制系统】PID算式实现,控制系统分类,工程应用中控制系统应该注意的问题

目录 1-1 试简述过程控制的发展概况及各个阶段的主要特点。 1-2 与其它自动控制相比&#xff0c;过程控制有哪些优点&#xff1f;为什么说过程控制的控制过程多属慢过程&#xff1f; 1-3 什么是过程控制系统&#xff0c;其基本分类是什么&#xff1f; 1-4 何为集散控制系统…...

测试第三课-------自动化测试相关

作者前言 &#x1f382; ✨✨✨✨✨✨&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f382; ​&#x1f382; 作者介绍&#xff1a; &#x1f382;&#x1f382; &#x1f382; &#x1f389;&#x1f389;&#x1f389…...

关于数据清洗和数据处理实践学习笔记

一些可能想要知道的&#xff1a; pandas是一个模板&#xff0c;它读取的数据都是dataframe的格式&#xff0c;即df Matplotlib是一个用于数据可视化的Python库&#xff0c;能够生成各种静态、动态和交互式图表 pyplot&#xff1a;Matplotlib 的核心接口模块&#xff0c;提供类…...

ubuntu学习day2

linux常用命令 3.文件查看及处理命令 3.1查看文件内容 cat[选项][文件] -b 对非空输出行编号 -E 在每行结束处显示$ -n 对输出的所有行编号 -s 不输出多行空行 标准输入、标准输出和标准错误 在 Linux 中&#xff0c;每个进程默认有三个文件描述符&#xff1a; 标准输入&…...

JavaScript `new Date()` 方法移动端 `兼容 ios`,ios环境new Date()返回NaN

在 iOS 环境下&#xff0c;new Date() 方法会返回 NaN&#xff0c;这通常是由于时间字符串的格式问题。iOS 的 Date 构造函数对时间字符串的格式要求比其他平台更严格。 原因&#xff1a;ios端不兼容“-”为连接符的时间。 解决办法&#xff1a; 替换时间格式 IOS 不支持某…...

考研408参考用书:计算机组成原理(唐朔飞)介绍,附pdf

我用夸克网盘分享了「《计算机组成原理》第2,3版 唐朔飞」&#xff0c; 链接&#xff1a;https://pan.quark.cn/s/6a87d10274a3 1. 书籍定位与适用对象 定位&#xff1a;计算机组成原理是计算机科学与技术、软件工程等专业的核心基础课程&#xff0c;涉及计算机硬件的底层工作原…...

案例-索引对于并发Insert性能优化测试

前言 最近因业务并发量上升&#xff0c;开发反馈对订单表Insert性能降低。应开发要求对涉及Insert的表进行分析并提供优化方案。   一般对Insert 影响基本都在索引&#xff0c;涉及表已按创建日期做了分区表&#xff0c;索引全部为普通索引未做分区索引。 优化建议&#xff…...

@Async 为什么要自定义线程池,使用默认线程池风险

为什么要自定义线程池而非使用默认线程池 使用Spring的Async注解时&#xff0c;如果不自定义线程池而使用默认线程池&#xff0c;可能会带来一些风险和问题。以下是主要原因&#xff1a; 默认线程池的风险 无限制的资源消耗 默认线程池使用SimpleAsyncTaskExecutor&#xff0…...

Spark-SQL简介与编程

1. Spark-SQL是什么 Spark SQL 是 Spark 用于结构化数据(structured data)处理的 Spark 模块。 Hadoop与Spark的对比 Hadoop的局限性 Hadoop无法处理结构化数据&#xff0c;导致一些项目无法推进。 例如&#xff0c;MySQL中的数据是结构化的&#xff0c;Hadoop无法直接处理。…...

如何分析 JVM OOM 内存溢出 Dump 快照日志

文章目录 1、需求背景2、OOM 触发3、Dump 日志分析 1、需求背景 企业开发过程中&#xff0c;如果系统服务客户量比较大&#xff0c;偶尔会出现OOM内存溢出问题&#xff0c;导致服务发生宕机&#xff0c;停止对外提供访问。 这种情况就需要排查定位内存溢出的原因&#xff08;…...

系统监控 | 简易多个内网服务器的CPU和内存使用率监控 system_moniter

效果图 原理 一台主机A上运行mysql数据库&#xff0c;接收数据。 其他主机设置定时任务&#xff0c;每6分钟发送一次自己的CPU和内存使用百分数到主机A。 主机A上提供flask为后台的可视化网页&#xff0c;见上图。 源码库 https://github.com/BioMooc/system_moniterhttps:/…...

【神经网络】python实现神经网络(四)——误差反向传播的基础理论

一.反向传播 本章将介绍能够高效计算权重参数的梯度的方法——误差反向传播法,这里简单介绍一下什么是反向传播,加入有个函数y = f(x),那么它的反向传播为图下这个样子: 反向传播的计算顺序是,将输入信号E乘以节点的局部导数,然后将结果传递给下一个节点。这里所…...

Django 开发服务器

$ python manage.py runserver $ python manage.py runserver 666 # 用 666 端口 $ python manage.py runserver 0.0.0.0:8000 # 让局域网内其他客户端也可访问 $ python manage.py runserver --skip-checks # 跳过检查自动检查 $ python manage.py runserver --…...

嵌入式基础(二)ARM基础

嵌入式基础&#xff08;二&#xff09;ARM基础 1.精简指令集和复杂指令集的区别⭐⭐⭐ 精简指令集 (RISC) 精简指令集 (Reduced Instruction Set Computing) 具有简洁、精简的指令集&#xff0c;每条指令执行的操作都很基础&#xff0c;使得处理器设计更简单。RISC 处理器通…...

RNA免疫共沉淀测序(RIP-seq)

技术简介 RNA免疫共沉淀测序&#xff08;RNA Immunoprecipitation Sequencing, RIP-seq&#xff09;是一种将RNA免疫共沉淀&#xff08;RIP&#xff09;与二代测序技术&#xff08;NGS&#xff09;相结合&#xff0c;用于研究细胞内RNA与蛋白相互作用的技术。 技术原理 利用目…...

期指跌对股市的影响是什么?

国内股指期货对大盘的影响&#xff0c;这种一般就是不想再买这种指数&#xff0c;大多数都在蓝筹股方面&#xff0c;题材股很少&#xff0c;股指期货是保证金交易&#xff0c;一手大概在15-18W&#xff0c;它的价格是根据指数&#xff08;如上证指数、深证成指&#xff09;来确…...

基于Python的​​LSTM、CNN中文情感分析系统

大家好&#xff0c;我是徐师兄&#xff0c;一个有着7年大厂经验的程序员&#xff0c;也是一名热衷于分享干货的技术爱好者。平时我在 CSDN、掘金、华为云、阿里云和 InfoQ 等平台分享我的心得体会。 &#x1f345;文末获取源码联系&#x1f345; 2025年最全的计算机软件毕业设计…...

Neovim安装及lazy配置

安装neovim 官网下载 配置lazy插件总成 lazy官网 一般在C盘里会有一个nvim-data&#xff0c;然后用官网里的命令会生成一个nvim 安装C编译器 参考此文 插件都放在目录’C:\Users\wnlea\AppData\Local\nvim\lua\plugins’中&#xff0c;所以新建一个插件&#xff0c;起名为vi…...

什么叫“架构”

我们学硬件架构的时候常常被一些名词和概念绕晕&#xff0c;这篇就来讲一讲“架构”这个概念&#xff0c;一种“架构”指的是什么&#xff0c;如何去学习一种新的架构。 1.架构&#xff1a;硬件设计与指令集的统一体 这里放上我大二下的手写笔记&#xff1a; 就是说硬件设计…...

【Python浅拷贝与深拷贝详解】

目录 前言&#xff1a;技术背景与价值当前技术痛点解决方案概述目标读者说明 一、技术原理剖析核心概念图解关键技术模块技术选型对比 二、实战演示环境配置要求核心代码实现&#xff08;10个案例&#xff09;案例1&#xff1a;列表嵌套列表案例2&#xff1a;字典嵌套列表案例3…...

numpy.ma.masked_where:屏蔽满足条件的数组

1.函数功能 屏蔽满足条件的数组内容&#xff0c;返回值为掩码数组 2.语法结构 np.ma.masked_where(condition, a, copyTrue)3. 参数 参数含义condition屏蔽条件a要操作的数组copy布尔值&#xff0c;取值为True时&#xff0c;结果复制数组(原始数据不变)&#xff0c;否则返回…...

力扣hot100_技巧_python版本

一、136. 只出现一次的数字 思路&#xff1a; 任何数和 0 做异或运算&#xff0c;结果仍然是原来的数&#xff0c;即 a⊕0a。任何数和其自身做异或运算&#xff0c;结果是 0&#xff0c;即 a⊕a0。异或运算满足交换律和结合律&#xff0c;即 a⊕b⊕ab⊕a⊕ab⊕(a⊕a)b⊕0b。 代…...

用队列实现栈

队列实现栈 用队列实现栈一、队列数据结构的基础定义与操作&#xff08;一&#xff09;队列节点与队列结构体定义&#xff08;二&#xff09;队列大小计算函数&#xff08;三&#xff09;队列初始化函数&#xff08;四&#xff09;队列销毁函数&#xff08;五&#xff09;队列元…...

Android WebView深度性能优化方案

一、启动阶段优化 预初始化策略 冷启动优化&#xff1a;在Application或后台线程提前初始化WebView new Thread(() -> {WebView preloadWebView new WebView(getApplicationContext());preloadWebView.loadUrl("about:blank"); }).start();WebView复用池 private…...

国标GB28181视频平台EasyCVR打造线下零售平台视频+AI全流程监管坚实防线

一、背景概述 在全球经济增长放缓、电商崛起、经营成本攀升的形势下&#xff0c;零售行业正经历深刻变革。数字化转型成为新零售发展的必由之路&#xff0c;但多数零售企业在信息化建设上困难重重&#xff0c;既缺乏足够重视&#xff0c;又因过高投入而犹豫。 随着大数据、人工…...

QML中打印Item的坐标

在 QML 中&#xff0c;你可以通过多种方式获取和打印 Item 的坐标信息。以下是几种常见的方法&#xff1a; 1. 打印相对坐标&#xff08;相对于父项&#xff09; qml Item {id: myItemx: 50y: 100width: 200height: 200Component.onCompleted: {// 打印相对于父项的坐标cons…...

基于【Lang Chain】构建智能问答系统的实战指南

&#x1f407;明明跟你说过&#xff1a;个人主页 &#x1f3c5;个人专栏&#xff1a;《深度探秘&#xff1a;AI界的007》 &#x1f3c5; &#x1f516;行路有良友&#xff0c;便是天堂&#x1f516; 目录 一、引言 1、什么是Lang Chain 2、LangChain在问答系统中的核心优…...

Vue使用axios实现:上传文件、下载文件

Vue 使用 axios 框架,系列文章: 《Vue使用axios实现Ajax请求》 《Vue使用axios二次封装、解决跨域问题》 《Vue使用axios实现:上传文件、下载文件》 在实际开发过程中,浏览器通常需要和服务器端进行数据交互。而 Vue.js 并未提供与服务器端通信的接口。Axios 提供了一些方便…...

泊松分布详解:从理论基础到实际应用的全面剖析

泊松分布详解&#xff1a;从理论基础到实际应用的全面剖析 目录 引言&#xff1a;事件的罕见性与随机计数泊松分布的历史源流泊松分布的数学定义与性质 概率质量函数 (PMF)累积分布函数 (CDF)期望、方差与其他矩矩生成函数 (MGF) 与特征函数 (CF) 泊松分布的严格推导 极限推导…...

PHP爬虫教程:使用cURL和Simple HTML DOM Parser

一个关于如何使用PHP的cURL和HTML解析器来创建爬虫的教程&#xff0c;特别是处理代理信息的部分。首先&#xff0c;我需要确定用户的需求是什么。可能他们想从某个网站抓取数据&#xff0c;但遇到了反爬措施&#xff0c;需要使用代理来避免被封IP。不过用户没有提到具体的目标网…...

# 更换手机热点后secureCRT无法连接centOS7系统

更换手机热点后secureCRT无法连接centOS7系统 一、问题描述 某些情况下&#xff0c;我们可能使用手机共享热点而给电脑联网。本来用一个手机热点共享网络时&#xff0c;SecureCRT可以正常连接到CentOS 7虚拟机&#xff0c;当更换一个手机热点时&#xff0c;突然发现SecureCR…...

【集成电路版图设计学习笔记】2. 基本绘制的layer层和电路失效机制

一、基本的版图层次 1. 金属层&#xff08;Metal Layers&#xff09; 金属层主要起到互连的作用&#xff0c;完成基本电路器件的连接金属线的材质通常是铝或者铜&#xff0c;一般在线条比较粗的情况下&#xff0c;即特征尺寸比较粗的&#xff0c;一般是用铝制作的。在先进工艺…...

SQL学习笔记-聚合查询

非聚合查询和聚合查询的概念及差别 1. 非聚合查询 非聚合查询&#xff08;Non-Aggregate Query&#xff09;是指不使用聚合函数的查询。这类查询通常用于从表中检索具体的行和列数据&#xff0c;返回的结果是表中的原始数据。 示例 假设有一个名为 employees 的表&#xff…...

Profibus DP主站转modbusTCP网关与dp从站通讯案例

Profibus DP主站转modbusTCP网关与dp从站通讯案例 在当前工业自动化的浪潮中&#xff0c;不同协议之间的通讯转换成为了提升生产效率和实现设备互联的关键。Profibus DP作为一种广泛应用的现场总线技术&#xff0c;与Modbus TCP的结合&#xff0c;为工业自动化系统的集成带来了…...

【Linux】41.网络基础(2.3)

文章目录 2.3 TCP协议2.3.5 理解TIME_WAIT状态2.3.6 解决TIME_WAIT状态引起的bind失败的方法(作业)2.3.7 理解 CLOSE_WAIT 状态2.3.8 滑动窗口2.3.9 流量控制 2.3 TCP协议 2.3.5 理解TIME_WAIT状态 现在做一个测试,首先启动server,然后启动client,然后用Ctrl-C使server终止,这…...

C++多态知识点梳理

多态 多态的概念&#xff1a; 多态就是多种形态&#xff0c;具体点就是去完成某个行为&#xff0c;当不同的对象去完成时会产生出不同的状态。 比如构成多态的俩个父子类&#xff0c;我们调用同一个函数&#xff0c;可能会产生不同的行为&#xff0c;比如普通人买票全价&…...

Python批量处理PDF图片详解(插入、压缩、提取、替换、分页、旋转、删除)

目录 一、概述 二、 使用工具 三、Python 在 PDF 中插入图片 3.1 插入图片到现有PDF 3.2 插入图片到新建PDF 3.3 批量插入多张图片到PDF 四、Python 提取 PDF 图片及其元数据 五、Python 替换 PDF 图片 5.1 使用图片替换图片 5.2 使用文字替换图片 六、Python 实现 …...

计算机网络分层模型:架构与原理

前言 计算机网络通过不同的层次结构来实现通信和数据传输&#xff0c;这种分层设计不仅使得网络更加模块化和灵活&#xff0c;也使得不同类型的通信能够顺利进行。在网络协议和通信体系中&#xff0c;最广为人知的分层模型有 OSI模型 和 TCP/IP模型。这两种模型分别定义了计算…...

算法-mysql笔记

寻找用户推荐人 mysql判断数据是空 IS null 非空 IS NOT null 584. 寻找用户推荐人 - 力扣&#xff08;LeetCode&#xff09; # Write your MySQL query statement below SELECTname FROMCustomer WHEREreferee_id ! 2 OR referee_id IS null 文章概览 当查询到有多…...

销售易CRM:技术架构与安全性能的深度解析

一、技术架构&#xff1a;云计算与微服务的完美结合 销售易CRM基于云计算架构&#xff0c;采用微服务设计理念&#xff0c;确保系统的高可用性和扩展性。这种架构不仅提高了系统的性能和稳定性&#xff0c;还为企业提供了灵活的定制化能力。 云计算架构的优势 高可用性&…...

Python用户管理系统深度解析(附源码):从类设计到安全实现的完整指南

目录 一、核心类结构全解 1.1 类定义与属性设计 代码解析&#xff1a; 二、注册功能代码逐行解析 2.1 用户名验证模块 功能实现&#xff1a; 2.2 密码设置流程 关键机制&#xff1a; 2.3 数据存储实现 文件操作要点&#xff1a; 三、登录安全机制全剖析 3.1 黑名单…...

【linux】使用LNMP环境+Discuz论坛源程序

我使用的版本是linux9.3、Discuz X3.5、nginx1.20、mariadb10.5、php8.0 整体结构 LNMP Linux Nginx mariadb PHP Nginx 最初于2004年10月4日为俄罗斯知名门户站点而开发的Nginx是一款轻量级的网站服务软件&#xff0c;因其稳定性和丰富的功能而深受信赖特点&#xff1a;…...

鸿蒙开发-动画

1. 动画-动画特效 // 定义接口 (每个列表项的数据结构) interface ImageCount {url: stringcount: number }// 需求1: 遮罩层显隐 透明度opacity 0-1 层级zIndex -1~99 // 需求2: 图片缩放 缩放scale 0-1Entry Component struct Index {// 基于接口, 准备数据State images…...

itext7 html2pdf 将html文本转为pdf

1、将html转为pdf需求分析 经常会看到爬虫有这样的需求&#xff0c;将某一个网站上的数据&#xff0c;获取到了以后&#xff0c;进行分析&#xff0c;然后将需要的数据进行存储&#xff0c;也有将html转为pdf进行存储&#xff0c;作为原始存档&#xff0c;当然这里看具体的需求…...

设计模式:模板模式 - 固定流程与灵活扩展的完美结合

一、为什么使用模板模式&#xff1f; 权限校验、数据处理、用例设计等流程虽然遵循固定步骤&#xff0c;但每个具体实现却总有不同。如果没有合适的设计&#xff0c;重复代码会堆积&#xff0c;导致系统复杂度增加&#xff0c;维护成本上升。那如何解决这个问题&#xff0c;让…...

Java 设计模式:组合模式详解

Java 设计模式&#xff1a;组合模式详解 组合模式&#xff08;Composite Pattern&#xff09;是一种结构型设计模式&#xff0c;它允许将对象组织成树形结构&#xff0c;以统一的方式处理单个对象和对象集合。组合模式适用于需要表示“部分-整体”层次结构的场景&#xff0c;例…...

使用命令打开电脑的[服务]窗口

1.首先打开[开始],找到[运行], 2.或者用快捷命令“windows键R键”命令打开运行, 3.然后输入命令“services.msc”, 4.点[确定]就可以进入电脑的[服务]窗口了...

语音识别——根据声波能量、VAD 和 频谱分析周围是否有人说话

语音活动检测&#xff08;Voice Activity Detection&#xff0c;简称VAD&#xff09;。简单来说&#xff0c;VAD就是用来判断一段音频里有没有人说话的技术。在实时语音识别的场景里&#xff0c;这个技术特别重要&#xff0c;因为它决定了什么时候把采集到的音频数据扔进大模型…...

C++算法优化实战:破解性能瓶颈,提升程序效率

C算法优化实战&#xff1a;破解性能瓶颈&#xff0c;提升程序效率 在现代软件开发中&#xff0c;算法优化是提升程序性能的关键手段之一。无论是在高频交易系统、实时游戏引擎&#xff0c;还是大数据处理平台&#xff0c;算法的高效性直接关系到整体系统的性能与响应速度。C作…...