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

kotlin图片合成和压缩

kotlin图片合成和压缩

之前的方法是继承AsyncTask 在doInBackground 里面去做压缩的操作,然后用 publishProgress 切到主线程里面更新
新方法是在协程里的去做

class ImageService {private fun getSumWidths(bitmaps: ArrayList<Bitmap>): Int {var sumWidth = 0for (b in bitmaps) {sumWidth = intValidCheck((sumWidth + b.width).toLong())}return sumWidth}private fun getMaxHeights(bitmaps: ArrayList<Bitmap>): Int {var max = bitmaps[0].heightfor (bitmap in bitmaps) {if (max < bitmap.height) {max = bitmap.height}}return max}/*** crop front rear left right View Image*/fun cropImage(src: Bitmap): Bitmap {Log.d("cropImage src width: " + src.width + ", height: " + src.height)//int x, int y, int width, int heightval result =Bitmap.createBitmap(src, 0, 0, CROP_IMAGE_WIDTH_SIZE, CROP_IMAGE_HEIGHT_SIZE)Log.d("cropImage result width: " + result.width + ", height: " + result.height)if (result != src) {src.recycle()}return result}fun cropOtherViewImage(src: Bitmap): Bitmap? {Log.d("cropTopViewImage width: " + src.width + ", height: " + src.height)//int x, int y, int width, int heightval result = Bitmap.createBitmap(src,CROP_OTHER_VIEW_IMAGE_X_POINT,0,CROP_OTHER_VIEW_IMAGE_WIDTH_SIZE,CROP_OTHER_VIEW_IMAGE_HEIGHT_SIZE)Log.d("cropTopViewImage result width: " + result.width + ", height: " + result.height)if (result != src) {src.recycle()}return result}fun mergeMultipleImages(bitmaps: ArrayList<Bitmap>): Bitmap? {Log.d("mergeMultipleImages")if (bitmaps.isEmpty()){return null}var totalWidth = 0var maxHeight = 0for (bitmap in bitmaps) {totalWidth += bitmap.widthif (bitmap.height > maxHeight) {maxHeight = bitmap.height}}val mergedBitmap = Bitmap.createBitmap(getSumWidths(bitmaps),getMaxHeights(bitmaps),Bitmap.Config.ARGB_8888)val canvas = Canvas(mergedBitmap)val paint = Paint()/*val mTextPaint = Paint()mTextPaint.color = Color.WHITEmTextPaint.isAntiAlias = true*/var currentX = 0for (i in bitmaps.indices) {val bitmap = bitmaps[i]Log.d("bitmaps.get( " + i + " ) result width: " + bitmap.width + ", height: " + bitmap.height)val srcRect = Rect(0, 0, bitmap.width, bitmap.height)val dstRect = Rect(currentX, 0, currentX + bitmap.width, maxHeight)canvas.drawBitmap(bitmap, srcRect, dstRect, paint)currentX += bitmap.width}Log.d("mergeMultipleImages result Bitmap width: " + mergedBitmap.width + ", height: " + mergedBitmap.height)return mergedBitmap}suspend fun compressImages(bitmap: Bitmap): ByteArray? {return withContext(Dispatchers.Default) {Log.d(" start compressImages: ")val outputStream = ByteArrayOutputStream()var quality = 100bitmap.compress(Bitmap.CompressFormat.JPEG, quality, outputStream)while (outputStream.size() / KB > MAX_IMAGE_SIZE && quality > 0) {outputStream.reset()quality -= 1bitmap.compress(Bitmap.CompressFormat.JPEG, quality, outputStream)}Log.d("compressImages:end  ${outputStream.size()}")val saveCompressImage = saveCompressImage(getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS),outputStream)Log.d("saveCompressImage: $saveCompressImage")outputStream.toByteArray()}}private fun saveCompressImage(f: File, bos: ByteArrayOutputStream?): Boolean {val file = File(f,"compress.jpeg")Log.d("saveCompressImage : " + file.path)return try {val fos = FileOutputStream(file)bos?.writeTo(fos)Log.d("saveCompressImage success")fos.flush()fos.close()true} catch (e: Exception) {Log.d("saveCompressImage error")e.printStackTrace()false}}companion object {private const val MAX_IMAGE_SIZE = 250private const val CROP_IMAGE_WIDTH_SIZE = 120  // 尺寸要保持一致private const val CROP_IMAGE_HEIGHT_SIZE = 121 // 尺寸要保持一致private const val CROP_OTHER_VIEW_IMAGE_X_POINT = 1288 // 其他的尺寸要保持一致 这个是额外添加的viewprivate const val CROP_OTHER_VIEW_IMAGE_WIDTH_SIZE = 632 // 其他的尺寸要保持一致private const val CROP_OTHER_VIEW_IMAGE_HEIGHT_SIZE = 720 其他的尺寸private const val KB: Long = 1024fun intValidCheck(value: Long): Int {if (value < Int.MIN_VALUE || value > Int.MAX_VALUE) {throw ArithmeticException("Integer overflow")}return value.toInt()}}}

相关文章:

kotlin图片合成和压缩

kotlin图片合成和压缩 之前的方法是继承AsyncTask 在doInBackground 里面去做压缩的操作&#xff0c;然后用 publishProgress 切到主线程里面更新 新方法是在协程里的去做 class ImageService {private fun getSumWidths(bitmaps: ArrayList<Bitmap>): Int {var sumWid…...

Java学习笔记004——接口概念理解及意义

一个类中有抽象方法&#xff0c;则必须声明为abstract&#xff08;做为抽象类&#xff09;&#xff0c;抽象类不能实例化。子类继承抽象类&#xff0c;必须对所有的抽象方法重写&#xff0c;否则依然有抽象方法&#xff0c;还是抽象的&#xff0c;无法实例化。故抽象类常做为基…...

MT笔试题

前言 某团硬件工程师的笔试题&#xff0c;个人感觉题目的价值还是很高的&#xff0c;分为选择题和编程题&#xff0c;选择题考的是嵌入式基础知识&#xff0c;编程题是两道算法题&#xff0c;一道为简单难度&#xff0c;一道为中等难度 目录 前言选择题编程题 选择题 C语言中变…...

50道SQL面试题

50道SQL面试题 有需要互关的小伙伴,关注一下,有关必回关,争取今年认证早日拿到博客专家 环境 -- ---------------------------- -- Table structure for teacher -- ---------------------------- DROP TABLE IF EXISTS teacher; CREATE TABLE teacher (t_id varchar(20) …...

2024蓝桥杯每日一题(双指针)

一、第一题&#xff1a;牛的学术圈 解题思路&#xff1a;双指针贪心 仔细思考可以知道&#xff0c;写一篇综述最多在原来的H指数的基础上1&#xff0c;所以基本方法可以是先求出原始的H指数&#xff0c;然后分类讨论怎么样提升H指数。 【Python程序代码】 n,l map(int,…...

Android 开发过程中常见的内存泄漏场景分析

场景1 Static变量存储上下文环境Context public class ClassName {// 定义1个静态变量private static Context mContext;//... // 引用的是Activity的contextmContext context; // 当Activity需销毁时&#xff0c;由于mContext 静态 & 生命周期 应用程序的生命周期&…...

Codeforces-1935E:Distance Learning Courses in MAC(思维)

E. Distance Learning Courses in MAC time limit per test 2 seconds memory limit per test 256 megabytes input standard input output standard output The New Year has arrived in the Master’s Assistance Center, which means it’s time to introduce a new feature…...

ZooKeeper和Diamond有什么不同

本文主要是讨论下两个类似产品&#xff1a;ZooKeeper和Diamond在配置管理这个应用场景上的异同点。 Diamond&#xff0c;顾名思义&#xff0c;寄寓了开发人员对产品稳定性的厚望&#xff0c;希望它像钻石一样&#xff0c;提供稳定的配置访问。Diamond是淘宝网Java中间件团队的核…...

三、N元语法(N-gram)

为了弥补 One-Hot 独热编码的维度灾难和语义鸿沟以及 BOW 词袋模型丢失词序信息和稀疏性这些缺陷&#xff0c;将词表示成一个低维的实数向量&#xff0c;且相似的词的向量表示是相近的&#xff0c;可以用向量之间的距离来衡量相似度。 N-gram 统计语言模型是用来计算句子概率的…...

QML 3D入门知识路线

目前使用的版本 v5.14.0 模块导入 使用QML 3D时需要 import Qt3D.Core 2.14 核心模块类 V6以上的版本已经发布&#xff0c;所以有很多module会发生变化&#xff0c;主要有核心module、输入、逻辑、渲染、动画和扩展module&#xff0c;以及2D/3D场景模块 类名 能…...

蓝牙系列五:开源蓝牙协议BTStack框架代码阅读(1)

蓝牙学习系列,借鉴卫东上老师的蓝牙视频教程。 BTStack协议栈学习。首先来看一下,对于硬件操作,它是如何来进行处理的。在上篇文章中曾说过,在main函数里面它会调用硬件相关的代码,调用操作系统相关的代码。在BTStack中,可以搜索一下main.c,将会发现有很多main.c,都是…...

c++ 类内可以定义引用数据成员吗?

在C中&#xff0c;类内是可以定义引用数据成员的&#xff0c;但是在初始化对象时&#xff0c;必须在构造函数的成员初始化列表中对引用进行初始化&#xff0c;因为引用必须在创建时被初始化&#xff0c;并且不能在其生存期内引用不同的对象。下面是一个简单的示例&#xff1a; …...

MacBook2024苹果免费mac电脑清理垃圾软件CleanMyMac X

CleanMyMac X是一款专业的Mac清理软件&#xff0c;具备多种强大功能。首先&#xff0c;它能够智能清理Mac磁盘上的垃圾文件和多余语言安装包&#xff0c;从而快速释放电脑内存。其次&#xff0c;CleanMyMac X可以轻松管理和升级Mac上的应用&#xff0c;同时强力卸载恶意软件并修…...

Vue.js计算属性:实现数据驱动的利器

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…...

10-ARM gicv3/gicv4的总结-基础篇

目录 1、gic的版本2、GICv3/gicv4的模型图3、gic中断号的划分4、GIC连接方式5、gic的状态6、gic框架7、gic Configuring推荐 本文转自 周贺贺&#xff0c;baron&#xff0c;代码改变世界ctw&#xff0c;Arm精选&#xff0c; armv8/armv9&#xff0c;trustzone/tee&#xff0c;s…...

数据库系统概论(超详解!!!) 第三节 关系数据库

1.基本概念 1. 域&#xff08;Domain&#xff09; 域是一组具有相同数据类型的值的集合。 2. 笛卡尔积&#xff08;Cartesian Product&#xff09; 给定一组域D1&#xff0c;D2&#xff0c;…&#xff0c;Dn&#xff0c;允许其中某些域是相同的。 D1&#xff0c;D2…...

Springboot 集成kafka 消费者实现ssl方式连接监听消息实现消费

证书准备&#xff1a;springboot集成kafka 消费者实现 如何配置是ssl方式连接的时候需要进行证书的转换。原始的证书是pem, 或者csr方式 和key方式的时候需要转换&#xff0c;因为kafka里面是jks 需要通过openssl进行转换。 证书处理&#xff1a; KeyStore 用于存储客户端的证…...

spark 实验二 RDD编程初级实践

目录 一. pyspark交互式编程示例&#xff08;学生选课成绩统计&#xff09; 该系总共有多少学生&#xff1b; 该系DataBase课程共有多少人选修&#xff1b; 各门课程的平均分是多少&#xff1b; 使用累加器计算共有多少人选了DataBase这门课。 二.编写独立应用程序实现数…...

【MySQL】not in遇上null的坑

今天遇到一个问题&#xff1a; 1、当 in 内的字段包含 null 的时候&#xff0c;正常过滤&#xff1b; 2、当 not in 内的字段包含 null 的时候&#xff0c;不能正常过滤&#xff0c;即使满足条件&#xff0c;最终结果也为 空。 测试如下&#xff1a; select * from emp e;当…...

鸿蒙4.0-DevEco Studio界面工程

DevEco Studio界面工程 DevEco Studio 下载与第一个工程新建的第一个工程界面回到Project工程结构来看 DevEco Studio 下载与第一个工程 DevEco Studio 下载地址&#xff1a;点击跳转 https://developer.harmonyos.com/cn/develop/deveco-studio#download 学习课堂以及文档地址…...

Rust高性能编程:Yi-Coder-1.5B所有权模型解析

Rust高性能编程&#xff1a;Yi-Coder-1.5B所有权模型解析 1. 引言 如果你刚开始学习Rust&#xff0c;可能会被所有权这个概念搞得有点懵。别担心&#xff0c;这很正常。Rust的所有权系统是它最独特的特性&#xff0c;也是保证内存安全的关键所在。今天我们就用Yi-Coder-1.5B这…...

怎样批量给文件重命名?这三个方法拿走不谢

日常办公或学习中&#xff0c;我们经常会遇到大量文件命名杂乱无章的情况&#xff0c;比如从相机导出的照片、批量下载的文档、项目相关的素材等&#xff0c;逐个手动重命名不仅耗时费力&#xff0c;还容易出现序号错乱、命名不统一的问题。今天就给大家分享3种实用的批量重命名…...

crontab——你的自动化打工人

咕嘎讲堂&#xff1a;crontab——你的自动化打工人 “人类最大的进步&#xff0c;就是学会了让机器帮自己干活。”——咕嘎 &#x1f4cc; crontab 是什么&#xff1f; crontab cron table&#xff0c;是 Linux 系统中用于定时执行任务的工具。 简单说&#xff1a;你想让系…...

JavaScript+WebGL可视化LingBot-Depth点云数据

JavaScriptWebGL可视化LingBot-Depth点云数据 1. 引言 想象一下&#xff0c;你手里有一个深度相机&#xff0c;它能捕捉到周围环境的3D信息&#xff0c;但原始数据往往充满了噪声和缺失区域。这就是LingBot-Depth发挥作用的地方——它能将不完整、有噪声的深度数据转换为高质…...

UniApp实战:Android原生插件实现动态时间水印踩坑全记录(附完整代码)

UniApp实战&#xff1a;Android原生插件实现动态时间水印的深度优化方案 在移动应用开发中&#xff0c;视频处理一直是技术难点之一&#xff0c;特别是需要实时添加动态时间水印的场景。本文将分享在UniApp中开发Android原生插件时&#xff0c;如何高效实现动态时间水印功能&am…...

SeaTunnel + SeaTunnel-Web 安装部署

下载SeaTunnel-Web 下载seatunnel-web安装包&#xff0c;安装包的版本在RENAME.md中有介绍。根据对应的版本号下载相应的软件包 https://mirrors.aliyun.com/apache/seatunnel/seatunnel-web/1.0.2/?spma2c6h.25603864.0.0.42d217c3AzltQh下载SeaTunnel 下载seatunnel安装包&a…...

我实测过的9个AI Agent Skills(用过就再也离不开)

智能体技能正成为打造实用AI智能体的全新黄金标准&#xff0c;但没人告诉你这个生态系统究竟有多混乱。找到安全又好用的技能就像碰运气&#xff1b;大多数仓库看起来惊艳无比……可一上手就原形毕露。我深有体会&#xff0c;因为我翻遍了几十个仓库。我一头扎进这个领域&#…...

小白友好!Llama-3.2V-11B-cot快速入门:上传图片提问,看AI推理全过程

小白友好&#xff01;Llama-3.2V-11B-cot快速入门&#xff1a;上传图片提问&#xff0c;看AI推理全过程 1. 引言&#xff1a;像聊天一样使用AI视觉推理 想象一下&#xff0c;你手头有一张图片——可能是旅游时拍的风景照&#xff0c;或是工作中遇到的图表&#xff0c;又或是孩…...

亲测口碑好的物联网开发生产厂家分享

亲测口碑好的物联网开发生产厂家分享行业痛点分析在当前物联网开发领域&#xff0c;存在着诸多技术挑战。首先&#xff0c;设备兼容性难题突出&#xff0c;不同品牌、型号的物联网设备通信协议和接口各异&#xff0c;导致系统集成困难。数据表明&#xff0c;约 60%的物联网项目…...

ICML 2025 | 时间序列预测与生成模型前沿进展全景解读

1. 时间序列预测与生成模型的2025技术风向标 ICML 2025收录的63篇时间序列相关论文&#xff0c;清晰地勾勒出该领域三大技术演进路径&#xff1a;扩散模型的高阶应用、基础模型的领域适配以及多模态融合的范式创新。从工业界实际应用的角度来看&#xff0c;今年最显著的变化是研…...