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

Android 手写签名功能详解:从原理到实践

Android 手写签名功能详解

      • 1. 引言
      • 2. 手写签名核心实现:SignatureView 类
      • 3. 交互层实现:MainActivity 类
      • 4. 布局与配置
      • 5. 性能优化与扩展方向

1. 引言

    在电子政务、金融服务等移动应用场景中,手写签名功能已成为提升用户体验与业务合规性的关键需求。实现一个流畅、安全且符合用户习惯的签名功能,需要在交互设计、性能优化和存储方案等方面进行综合考量。本文将围绕核心需求,结合关键代码解析其实现方案。

2. 手写签名核心实现:SignatureView 类

(1)初始化绘图设置

private void setupDrawing() {drawPaint = new Paint();drawPaint.setColor(paintColor);drawPaint.setAntiAlias(true);drawPaint.setStrokeWidth(20);drawPaint.setStyle(Paint.Style.STROKE);drawPaint.setStrokeJoin(Paint.Join.ROUND);drawPaint.setStrokeCap(Paint.Cap.ROUND);canvasPaint = new Paint(Paint.DITHER_FLAG);
}

技术原理

  • 抗锯齿技术setAntiAlias(true) 通过边缘像素的灰度处理消除锯齿,提升线条平滑度。在高分辨率屏幕上效果尤为明显,其原理是在边缘区域生成半透明像素,通过颜色混合实现视觉上的平滑过渡。
  • 笔触优化ROUND 类型的 Join 和 Cap 使线条连接自然,避免尖锐棱角。这对于模拟真实书写体验至关重要,特别是在书写速度较快时,能有效避免线条断裂感。
  • 抖动处理Paint.DITHER_FLAG 通过随机噪声算法优化色彩显示,在低精度屏幕上减少色彩断层现象。当图像色彩深度高于显示设备时,抖动技术能通过图案化的方式模拟更多颜色。

(2) 视图大小变化处理

@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {canvasBitmap = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888);drawCanvas = new Canvas(canvasBitmap);
}

内存管理机制

  • ARGB_8888 配置:每个像素占用 4 字节(32 位),支持完整的 24 位色彩和 8 位透明度。这对于需要保留签名细节和背景透明度的场景至关重要,但同时也意味着较大的内存占用(例如 1080x1920 分辨率的 Bitmap 占用约 8MB 内存)。
  • 动态调整:当屏幕旋转或布局变化时,系统会调用 onSizeChanged 方法,此时需重新创建 Bitmap 以匹配新尺寸。为避免频繁创建导致的内存抖动,可考虑添加尺寸阈值判断,仅在尺寸变化超过一定比例时重新创建。

(3)触摸事件处理

@Override
public boolean onTouchEvent(MotionEvent event) {float touchX = event.getX();float touchY = event.getY();switch (event.getAction()) {case MotionEvent.ACTION_DOWN:drawPath.moveTo(touchX, touchY);break;case MotionEvent.ACTION_MOVE:drawPath.lineTo(touchX, touchY);break;case MotionEvent.ACTION_UP:drawCanvas.drawPath(drawPath, drawPaint);drawPath.reset();break;default:return false;}invalidate();return true;
}

事件处理流程

  1. ACTION_DOWN:记录触摸起点,初始化 Path 对象
  2. ACTION_MOVE:持续追踪手指轨迹,通过 lineTo() 方法连接路径点
  3. ACTION_UP:将最终路径绘制到 Bitmap 上,并重置 Path 准备下一次绘制
  4. invalidate():触发 onDraw() 方法重绘视图,确保用户能实时看到绘制结果

性能优化点

  • 事件过滤:在 ACTION_MOVE 中添加距离阈值判断(如 dx > 4 || dy > 4),过滤微小抖动,减少不必要的绘制操作
  • 批量处理:对于高频触摸事件(如 120Hz 屏幕),可采用采样策略,每 N 个事件处理一次,平衡响应速度与绘制性能

3. 交互层实现:MainActivity 类

(1)按钮事件绑定

@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);signatureView = findViewById(R.id.signature_view);clearButton = findViewById(R.id.clear_button);saveButton = findViewById(R.id.save_button);shareButton = findViewById(R.id.share_button);clearButton.setOnClickListener(v -> signatureView.clear());saveButton.setOnClickListener(v -> saveSignature());shareButton.setOnClickListener(v -> shareSignature());
}

架构设计

  • MVC 模式:Activity 作为控制器,负责处理用户交互并调用 Model(SignatureView)的方法
  • 单一职责:将签名绘制逻辑封装在 SignatureView 中,Activity 专注于业务流程控制
  • 事件驱动:通过接口回调机制实现组件间通信,保持代码松耦合

(2)保存签名功能

private void saveSignature() {Bitmap signatureBitmap = signatureView.getSignatureBitmap();if (isBitmapEmpty(signatureBitmap)) {Toast.makeText(this, "签名为空,无法保存", Toast.LENGTH_SHORT).show();return;}try {File photoFile = createImageFile();try (FileOutputStream fos = new FileOutputStream(photoFile)) {signatureBitmap.compress(Bitmap.CompressFormat.PNG, 100, fos);Toast.makeText(this, "签名已保存至相册", Toast.LENGTH_SHORT).show();}} catch (IOException e) {e.printStackTrace();Toast.makeText(this, "保存失败,请稍后再试", Toast.LENGTH_SHORT).show();}
}

文件存储技术

  • PNG 格式选择:无损压缩格式,支持透明度,适合保存精细的签名图像
  • 质量参数compress() 方法的第二个参数(0-100)对 PNG 无效(因其为无损格式),但对 JPEG 有效
  • 异常处理:使用 try-with-resources 自动关闭流,防止资源泄漏;捕获 IOException 处理文件操作失败场景

存储路径选择

  • 内部存储getFilesDir() 返回的路径,其他应用无法访问,适合存储敏感数据
  • 外部存储getExternalFilesDir() 返回的路径,应用卸载时会被删除
  • 公共目录:需申请 WRITE_EXTERNAL_STORAGE 权限,适合保存需要共享的文件

4. 布局与配置

(1)布局文件设计

<?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"><com.example.signatureapp.SignatureViewandroid:id="@+id/signature_view"android:layout_width="match_parent"android:layout_height="0dp"android:layout_weight="1"android:background="@android:color/white"android:layout_marginBottom="16dp"/><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:orientation="horizontal"android:weightSum="3"><Buttonandroid:id="@+id/clear_button"android:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:text="清除"android:layout_marginRight="8dp"/><Buttonandroid:id="@+id/save_button"android:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:text="保存"android:layout_marginRight="8dp"/><Buttonandroid:id="@+id/share_button"android:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:text="分享"/></LinearLayout>
</LinearLayout>

响应式设计

  • 权重系统:通过 layout_weight 属性动态分配空间,确保签名区域占据主要屏幕空间
  • 边距优化layout_marginRight 设置按钮间距,提升触控友好性(Android 推荐最小触控区域为 48dp×48dp)
  • 背景处理:白色背景提供清晰的签名对比,同时减少眼睛疲劳

(2)应用清单配置

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android"><external-files-path name="my_images" path="Pictures" />
</paths>

安全配置解析

  • FileProvider:Android 7.0+ 强制要求通过 ContentProvider 分享文件,避免直接暴露文件路径
  • 路径映射external-files-path 将应用外部存储目录映射为 content URI,格式为 content://<authority>/my_images/filename.png
  • 权限控制:通过 grantUriPermissions 动态授予临时访问权限,避免静态声明危险权限

在这里插入图片描述

5. 性能优化与扩展方向

(1)内存优化

  • Bitmap 复用:在不需要透明度时使用 Bitmap.Config.RGB_565(每个像素 2 字节),减少内存占用
  • 缓存策略:使用 LruCache 缓存最近使用的 Bitmap,避免重复创建
  • 内存泄漏检测:通过 LeakCanary 等工具检测 Bitmap 未释放问题

(2)绘制优化

  • 双缓冲技术:通过内存画布(Bitmap + Canvas)减少 UI 刷新频率,避免屏幕闪烁
  • 硬件加速:通过 setLayerType(LAYER_TYPE_HARDWARE, null) 启用 GPU 加速复杂绘制操作
  • 离屏渲染:对于频繁重绘区域,使用 setWillNotCacheDrawing(false) 开启离屏缓存

(3)扩展功能实现

  • 压力感应
    float pressure = event.getPressure();
    drawPaint.setStrokeWidth(BASE_WIDTH + pressure * PRESSURE_FACTOR);
    
  • 撤销/重做:使用两个栈分别保存历史状态和撤销操作
  • 缩放平移:通过 Matrix 实现签名区域的缩放和平移功能

(4) 数据安全

  • 加密存储
    SecretKey secretKey = KeyGenerator.getInstance("AES").generateKey();
    Cipher cipher = Cipher.getInstance("AES/GCM/NoPadding");
    cipher.init(Cipher.ENCRYPT_MODE, secretKey);
    try (CipherOutputStream cos = new CipherOutputStream(fos, cipher)) {bitmap.compress(Bitmap.CompressFormat.PNG, 100, cos);
    }
    
  • 文件完整性校验:保存签名时计算并存储 SHA-256 哈希值,验证时重新计算比对
  • 水印技术:在签名图像中嵌入不可见水印,防止篡改

    Android手写签名功能通过自定义SignatureView基于CanvasPath捕捉绘制轨迹,利用双缓冲技术优化渲染性能,结合FileProvider实现安全存储与分享。开发中需注重抗锯齿、压力感应等体验优化,控制Bitmap内存占用以避免溢出,并通过加密存储、动态权限适配满足安全合规需求,模块化设计还可扩展撤销/重做等功能,适用于金融、医疗等多场景的数字化签名需求。

相关文章:

Android 手写签名功能详解:从原理到实践

Android 手写签名功能详解 1. 引言2. 手写签名核心实现&#xff1a;SignatureView 类3. 交互层实现&#xff1a;MainActivity 类4. 布局与配置5. 性能优化与扩展方向 1. 引言 在电子政务、金融服务等移动应用场景中&#xff0c;手写签名功能已成为提升用户体验与业务合规性的关…...

Level2.8蛇与海龟(游戏)

#小龟快跑游戏 输入难度(1-5),蛇追到龟&#xff0c;游戏结束 #分析问题&#xff1a;从局部>整体 #游戏画面&#xff1a;创建画笔(海龟蛇)>1.海龟移动(键盘控制)>2.蛇(自动追踪&#xff0c;海龟位置)>3.海龟(限定范围&#xff0c;防止跑出画布之外)>4.游戏&…...

【Android构建系统】如何在Camera Hal的Android.bp中选择性引用某个模块

背景描述 本篇文章是一个Android.bp中选择性引用某个模块的实例。 如果是Android.mk编译时期&#xff0c;在编译阶段通过某个条件判断是不是引用某个模块A, 是比较好实现的。Android15使用Android.bp构建后&#xff0c;要想在Android.bp中通过自定义的一个变量或者条件实现选…...

【Canvas与诗词】醉里挑灯看剑 梦回吹角连营

【成图】 【代码】 <!DOCTYPE html> <html lang"utf-8"> <meta http-equiv"Content-Type" content"text/html; charsetutf-8"/> <head><title>醉里挑灯看剑梦回吹角连营 Draft1</title><style type"…...

Hue面试内容整理-Hue 架构与前后端通信

Cloudera Hue 是一个基于 Web 的 SQL 助手,旨在为数据分析师和工程师提供统一的界面,以便与 Hadoop 生态系统中的各个组件(如 Hive、Impala、HDFS 等)进行交互。其架构设计强调前后端的分离与高效通信,确保系统的可扩展性和可维护性。以下是 Hue 架构及其前后端通信机制的…...

Linux搜索

假如我们要搜索 struct sockaddr_in 我们在命令终端输入 cd/usr/include/ //进入头文件目录地址 /usr/include/ grep " struct sockaddr_in { " *-nir &#xff08;*是在当前目录&#xff0c;n 是找出来显示行数…...

Git基础原理和使用

Git 初识 一、版本管理痛点 在日常工作和学习中&#xff0c;我们经常遇到以下问题&#xff1a; - 通过不断复制文件来保存历史版本&#xff08;如报告-v1、报告-最终版等&#xff09; - 版本数量增多后无法清晰记住每个版本的修改内容 - 项目代码管理存在同样问题 二、版本控…...

实现视频分片上传 OSS

访问 OSS 有两种方式&#xff0c;本文用到的是使用临时访问凭证上传到 OSS&#xff0c;不同语言版本的代码参考&#xff1a; 使用STS临时访问凭证访问OSS_对象存储(OSS)-阿里云帮助中心 1.安装并使用 首先我们要安装 OSS&#xff1a; npm install ali-oss --save 接着我们…...

网络I/O学习(一)

一、什么是网络IO&#xff1f; 就是客户端和服务端之间的进行通信的通道(fd)。 二、网络IO通信步骤 1、建立套接字 int socketfd socket(AF_INET, SOCK_STREAM, 0);struct sockaddr_in servaddr; servaddr.sin_family AF_INET; servaddr.sin_addr.s_addr htonl(INADDR_A…...

4:OpenCV—保存图像

将图像和视频保存到文件 在许多现实世界的计算机视觉应用中&#xff0c;需要保留图像和视频以供将来参考。最常见的持久化方法是将图像或视频保存到文件中。因此&#xff0c;本教程准备解释如何使用 OpenCV C将图像和视频保存到文件中。 将图像保存到文件 可以学习如何保存从…...

Selenium-Java版(css表达式)

css表达式 前言 根据 tag名、id、class 选择元素 tag名 #id .class 选择子元素和后代元素 定义 语法 根据属性选择 验证CSS Selector 组选择 按次序选择子节点 父元素的第n个子节点 父元素的倒数第n个子节点 父元素的第几个某类型的子节点 父元素的…...

产品更新丨谷云科技 iPaaS 集成平台 V7.5 版本发布

五月&#xff0c;谷云科技 iPaaS 集成平台保持月度更新&#xff0c; V7.5 版本于近日正式发布。我们一起来看看新版本有哪些升级和优化。 核心新增功能&#xff1a;深化API治理&#xff0c;释放连接价值 API网关&#xff1a;全链路可控&#xff0c;精准管控业务状态 业务状态…...

深度学习让鱼与熊掌兼得

通常,一个大的复杂的模型的loss会低,但是拟合方面不够,小的模型在拟合方面更好,但是loss高,我们可以通过深度学习来得到一个有着低loss的小模型 我们之前学过,peacewise linear可以用常数加上一堆这个阶梯型函数得到,然后因为peacewise linear可以逼近任何function,所以理论上…...

TDuckX 2.6 正式发布|API 能力开放,核心表单逻辑重构,多项实用功能上线。

大家好&#xff0c;TDuckX 2.6 已正式发布。 本次更新以可集成性提升、数据处理能力增强和交互体验优化为核心&#xff0c;新增了包括 新增OpenAPI 模块、表单数据批量修改、字段导出分列 等多个面向开发者和实际业务落地场景的功能。 我们也重构了部分底层逻辑模块&#xff…...

LeetCode Hot100刷题——除自身以外数组的乘积

238. 除自身以外数组的乘积 给你一个整数数组 nums&#xff0c;返回 数组 answer &#xff0c;其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法&a…...

JAVA EE(进阶)_进阶的开端

别放弃浸透泪水的昨天&#xff0c;晨光已为明天掀开新篇 ——陳長生. ❀主页&#xff1a;陳長生.-CSDN博客❀ &#x1f4d5;上一篇&#xff1a;JAVA EE_HTTP-CSDN博客 1.什么是Java EE Java EE&#xff08;Java Pla…...

PDF批量合并拆分+加水印转换 编辑 加密 OCR 识别

各位办公小能手们&#xff01;你们有没有遇到过被PDF文件折腾得晕头转向的时候呀&#xff1f;其实啊&#xff0c;有专门处理、编辑、管理和优化PDF文件的软件&#xff0c;那就是PDF工具。它功能老多了&#xff0c;有文档格式转换、内容编辑、页面管理、安全保护这些核心功能。下…...

Go语言交替打印问题及多种实现方法

Go语言交替打印问题及多种实现方法 在并发编程中&#xff0c;多个线程&#xff08;或 goroutine&#xff09;交替执行任务是一个经典问题。本文将以 Go 语言为例&#xff0c;介绍如何实现多个 goroutine 交替打印数字的功能&#xff0c;并展示几种不同的实现方法。 Go 语言相关…...

ArcGIS Pro调用多期历史影像

一、访问World Imagery Wayback&#xff0c;基本在我国范围 如下图&#xff1a; 二、 放大到您感兴趣的区域 三、 查看影像版本信息 点击第二步的按钮后&#xff0c;便可跳转至World Imagery (Wayback 2025-04-24)的相关信息。 四 、点击上图影像版本信息&#xff0c;页面跳转…...

10.11 LangGraph多角色Agent开发实战:生产级AI系统架构与性能优化全解析

LangGraph 项目:High-level API for Multi-actor Agents 关键词:LangGraph 多角色 Agent, 状态管理, 持久化机制, 工作流编排, 生产级 AI 系统 1. LangGraph 设计哲学与架构演进 LangGraph 是 LangChain 生态中首个面向 多角色协作 Agent 的高阶 API 框架,其核心设计思想可…...

组态王|组态王中如何添加西门子1200设备

哈喽,你好啊,我是雷工! 最近使用组态王采集设备数据,设备的控制器为西门子的1214CPU, 这里边实施边记录,以下为在组态王中添加西门子1200PLC的笔记。 1、新建 在组态王工程浏览器中选择【设备】→点击【新建】。 2、选择设备 和设备建立通讯要通过对应的设备驱动。 在…...

发布时将多个bpl 打包成一个bpl的方法,或者说:不需要vcl60.bpl情况下 18.5K的exe 照常可以运行。

其实这种方式 就是把项目的逻辑和业务 和 依赖分开。 控件和IDE 相对来说一段时间内不会改变。 更新只是更新一些项目的逻辑&#xff0c;例如你在代码里多写了一个 if &#xff0c;这样就可以只更新这个极小的exe。 题&#xff1a;关于bpl发布时将vcl60.bpl&#xff0c;vcld…...

6.2.2邻接表法-图的存储

知识总览&#xff1a; 为什么要用邻接表 因为邻接矩阵的空间复杂度高(O(n))&#xff0c;且不适合边少的稀疏图&#xff0c;所以有了邻接表 用代码表示顶点、图 声明顶点图信息 声明顶点用一维数组存储各个顶点的信息&#xff0c;一维数组字段包括2个&#xff0c;每个顶点的…...

C++23 放宽范围适配器以允许仅移动类型(P2494R2)

文章目录 引言背景与动机提案内容与实现细节提案 P2494R2实现细节编译器支持 对开发者的影响提高灵活性简化代码向后兼容性 示例代码总结 引言 C23 标准中引入了许多重要的改进&#xff0c;其中一项值得关注的特性是放宽范围适配器&#xff08;range adaptors&#xff09;以允…...

【技海登峰】Kafka漫谈系列(十一)SpringBoot整合Kafka之消费者Consumer

【技海登峰】Kafka漫谈系列(十一)SpringBoot整合Kafka之消费者Consumer spring-kafka官方文档: https://docs.spring.io/spring-kafka/docs/2.8.10/reference/pdf/spring-kafka-reference.pdf KafkaTemplate API: https://docs.spring.io/spring-kafka/api/org/springframe…...

Spring Boot三层架构设计模式

Spring Boot 的三层架构设计模式是一种经典的软件分层设计模式&#xff0c;旨在将应用程序划分为 表现层&#xff08;Controller&#xff09;、业务逻辑层&#xff08;Service&#xff09;、数据访问层&#xff08;Repository/DAO&#xff09;&#xff0c;通过清晰的职责划分提…...

在Java中调用Ant命令

在Java中调用Ant命令 在Java程序中调用Ant命令有几种方法&#xff0c;下面介绍两种常用的方式&#xff1a; 1. 使用Runtime.exec()方法 import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader;public class AntRunner {public stat…...

WebRTC技术下的EasyRTC音视频实时通话SDK,助力车载通信打造安全高效的智能出行体验

一、方案背景​ 随着智能交通与车联网技术的飞速发展&#xff0c;车载通信在提升行车安全、优化驾驶体验以及实现智能交通管理等方面发挥着越来越重要的作用。传统的车载通信方式在实时性、稳定性以及多媒体交互能力上存在一定局限&#xff0c;难以满足现代车载场景日益复杂的…...

数据科学和机器学习的“看家兵器”——pandas模块 之二

目录 pandas 模块介绍 4.2 pandas 数据读取 4.2.1 课程目标 4.2.2 读取 Excel 文件中的数据 (一)读取某个工作表中的数据 (二)读取指定数据列的标签内容 (三)读取指定数据行的标签内容 (四)读取指定行或者列 4.2.3、读取 CSV 文件数据 4.2.4、课程总结回顾 4.2.5、课后…...

本地部署Firecrawl+Dify调用踩坑记录

最近自己研究Dify&#xff0c;使用到Firecrawl这个比较好用的工具。用Firecrawl官网的不知道为什么总是卡住得不到结果&#xff0c;于是我打算自己去本地部署一个。好家伙真给我人搞麻了&#xff0c;太多问题了。 我是在京东云上面租的一台服务器。 首先就是docker的安装&…...