2311skia,06编解码图像上

1,API和自注册机制

Skia中只需要一行代码就可编解码图片:

SkBitmap bitmap;
SkImageDecoder::DecodeFile("test.xxx", &bitmap);//按文件名解码,自动推导`图片类型`//或按流解码
SkFILEStream stream("test.xxx");
SkImageDecoder::DecodeStream(stream, &bitmap);//由`输入流`解码,`自动推导`图片类型编码
//SkImageEncoder::EncodeFile("test.jpg", bitmap, SkImageEncoder::kJPEG_Type, 90/*编码图片质量,对`jpeg`格式和`webp`格式有用*/);

设计是用抽象工厂模式产生编码器,解码器实例,用自注册方式注入产生函数,使之在加载库时即初化完成.

template <typename T> class SkTRegistry : SkNoncopyable {
public:typedef T Factory;explicit SkTRegistry(T fact) : fFact(fact) {
#ifdef SK_BUILD_FOR_ANDROID//两次初化错误的变通{SkTRegistry* reg = gHead;while (reg) {if (reg == this) {return;}reg = reg->fChain;}}
#endiffChain = gHead;gHead  = this;}static const SkTRegistry* Head() { return gHead; }const SkTRegistry* next() const { return fChain; }const Factory& factory() const { return fFact; }
private:Factory      fFact;SkTRegistry* fChain;static SkTRegistry* gHead;
};
//调用者仍需要在某处声明此实例
template <typename T> SkTRegistry<T>* SkTRegistry<T>::gHead;

SkImageDecoder中规定的解码器工厂注册函数:

typedef SkTRegistry<SkImageDecoder*(*)(SkStreamRewindable*)>        SkImageDecoder_DecodeReg;

按如下方法注册gif解码器工厂:

static bool is_gif(SkStreamRewindable* stream) {char buf[GIF_STAMP_LEN];if (stream->read(buf, GIF_STAMP_LEN) == GIF_STAMP_LEN) {if (memcmp(GIF_STAMP,   buf, GIF_STAMP_LEN) == 0 ||memcmp(GIF87_STAMP, buf, GIF_STAMP_LEN) == 0 ||memcmp(GIF89_STAMP, buf, GIF_STAMP_LEN) == 0) {return true;}}return false;
}
static SkImageDecoder* sk_libgif_dfactory(SkStreamRewindable* stream) {if (is_gif(stream)) {return SkNEW(SkGIFImageDecoder);}return NULL;
}
static SkImageDecoder_DecodeReg gReg(sk_libgif_dfactory);

2,解码流程:

bool SkImageDecoder::DecodeStream(SkStreamRewindable* stream, SkBitmap* bm, SkColorType pref, Mode mode, Format* format) {SkASSERT(stream);SkASSERT(bm);bool success = false;SkImageDecoder* codec = SkImageDecoder::Factory(stream);if (NULL != codec) {success = codec->decode(stream, bm, pref, mode);if (success && format) {*format = codec->getFormat();if (kUnknown_Format == *format) {if (stream->rewind()) {*format = GetStreamFormat(stream);}}}delete codec;}return success;
}
bool SkImageDecoder::decode(SkStream* stream, SkBitmap* bm, SkColorType pref, Mode mode) {//调用`onDecode`前,重置为`false`fShouldCancelDecode = false;//赋值此值,供`getPrefColorType()`使用,以防`fUsePrefTable`为`false`fDefaultPref = pref;//传递一个临时位图,这样,如果返回`false`,就可确保`调用者`的位图不变.SkBitmap    tmp;if (!this->onDecode(stream, &tmp, mode)) {return false;}bm->swap(tmp);return true;
}

(1)遍历所有解码器,找到支持该文件解码器:
遍历代码见:

external/skia/src/images/SkImageDecoder_FactoryRegistrar.cpp
SkImageDecoder* SkImageDecoder::Factory(SkStreamRewindable* stream) {return image_decoder_from_stream(stream);
}
SkImageDecoder* image_decoder_from_stream(SkStreamRewindable* stream) {SkImageDecoder* codec = NULL;const SkImageDecoder_DecodeReg* curr = SkImageDecoder_DecodeReg::Head();while (curr) {codec = curr->factory()(stream);//在此倒带,因为承诺稍后当调用"解码"时,流在开始位置.bool rewindSuceeded = stream->rewind();//`图像解码器`要求支持倒带,否则如果得到不支持`倒带`的流,会提前失败.if (!rewindSuceeded) {SkDEBUGF(("Unable to rewind the image stream."));SkDELETE(codec);return NULL;}if (codec) {return codec;}curr = curr->next();}return NULL;
}

(2)解码器调用相应的解码库函数(简单的图片如bmp自行处理),解出原始图片,见各个解码器的onDecode方法
(3)onDecode方法中,一般需要用SkScaledBitmapSampler类,用来处理后续缩放图片及预乘透明度.
编码流程相对简单,就是根据类型取相应编码器并编码文件,不详述.

3,解码中流的设置:

框架层调Skia的解码,主要是以下几个函数:
nativeDecodeStream:以JavaInputStream类作为输入解码,创建JavaInputStreamAdaptor流(其实现是先回调JavaInputStream的读取方法,读其到缓存区,再从缓存区中读到目标内存.)

nativeDecodeFileDescriptor:以文件描述符输入解码,创建SkFILEStream,直接读取文件内容
nativeDecodeAsset:以Asset作为输入解码,创建AssetStreamAdaptor(frameworks/base/core/jni/android/graphics/Utils.cpp).
nativeDecodeByteArray:以ByteArray作为输入解码,创建SkMemoryStream,来直接从内存中读.

#define BYTES_TO_BUFFER 64
static jobject nativeDecodeStream(JNIEnv* env, jobject clazz, jobject is, jbyteArray storage,jobject padding, jobject options) {jobject bitmap = NULL;SkAutoTUnref<SkStream> stream(CreateJavaInputStreamAdaptor(env, is, storage));if (stream.get()) {SkAutoTUnref<SkStreamRewindable> bufferedStream( SkFrontBufferedStream::Create(stream, BYTES_TO_BUFFER));SkASSERT(bufferedStream.get() != NULL);bitmap = doDecode(env, bufferedStream, padding, options);}return bitmap;
}
static jobject nativeDecodeFileDescriptor(JNIEnv* env, jobject clazz, jobject fileDescriptor,jobject padding, jobject bitmapFactoryOptions) {NPE_CHECK_RETURN_ZERO(env, fileDescriptor);int descriptor = jniGetFDFromFileDescriptor(env, fileDescriptor);struct stat fdStat;if (fstat(descriptor, &fdStat) == -1) {doThrowIOE(env, "broken file descriptor");return nullObjectReturn("fstat return -1");}//在退出此函数时,恢复`描述符`的偏移.即使重复了描述符,`原始和重复`描述符都引用了相同的`打开文件描述`,且更改一个文件的`偏移`会影响另一个.AutoFDSeek autoRestore(descriptor);//在此`复制`描述符以`避免内存泄漏`.如果只关闭`文件描述符`而不关闭`创建`它的`文件对象`,就会泄漏.如果不`显式`清理文件(反之,又关闭了描述符),就会泄露`fseek`内部分配的缓冲. int dupDescriptor = dup(descriptor);FILE* file = fdopen(dupDescriptor, "r");if (file == NULL) {//清理重复描述符,因为`(fclose)`在`清理文件`时,不会关闭它.close(dupDescriptor);return nullObjectReturn("Could not open file");}SkAutoTUnref<SkFILEStream> fileStream(new SkFILEStream(file, SkFILEStream::kCallerPasses_Ownership));//使用缓冲流.尽管可倒带`SkFILEStream`,但这可确保`SkImageDecoder::Factory`在超出文件描述符的当前位置时,永远不倒带.SkAutoTUnref<SkStreamRewindable> stream(SkFrontBufferedStream::Create(fileStream, BYTES_TO_BUFFER));return doDecode(env, stream, padding, bitmapFactoryOptions);
}

因为以文件描述符InputStream为输入时,不能保证可倒带该流,因此加了一层SkFrontBufferedStream的包装,来缓存输入流中最前面的一段区域(#define BYTES_TO_BUFFER 64),以便在读取这段区域时可倒带(rewind),来让解码器读完文件头判断类型后可倒带.

详细见:
external/skia/src/utils/SkFrontBufferedStream.cpp

4,解码选项

external/skia/include/core/SkImageDecoder.h

class SkImageDecoder : SkNoncopyable {
private:Peeker*                 fPeeker;//仅在解`ninepatch`图片中使用,来提取子图片断,仅对`png`格式有效SkBitmap::Allocator*    fAllocator;//解码需要产生`SkBitmap`,这里是其像素的内存分配器int fSampleSize;//图片下采样率,为2的幂次,该设计主要针对的是`jpeg`格式的图像.`Jpeg`格式编码时以`8*8`像素单位为一个`块`作傅立叶变换,设成`2,4,8`时可针对性简化反傅立叶变换`(idct)`.SkColorType             fDefaultPref;//优先选取的解码出来的图片格式bool fDitherImage;//是否做抖动处理,主要针对`16`位色,在解码`png`和`jpeg`时生效bool                    fSkipWritingZeroes;//是否默认,结果图已清零并不写入像素值零的点.该选项仅在`SkScaledBitmapSampler`中有判断,如`(get_RGBA_to_8888_proc`函数),来节省计算`后续缩放`,意义不大.(真要优化不如用`simd`覆盖,如`neon`)mutable bool            fShouldCancelDecode;//比如用户在未解析完`A图片`时,已划过A图片所在区域,此时可设置`fShouldCancelDecode`,来取消解码`A图片`.bool fPreferQualityOverSpeed;//只对`jpeg`格式有用,决定反傅立叶变换`(idct)`的运算精度,速度优先时,以8位精度整数替代浮点运算,质量优先时,以`16`位精度整数替代浮点运算bool fRequireUnpremultipliedColors;//解带透明度的图片,如`png`时有用,表示是否用透明度预乘`图片像素`,默认是`预乘`,以便`渲染`时不用再乘.
};

这些私有变量对应BitmapFactory.Options中的相应的配置项,见

frameworks/base/graphics/java/android/graphics/BitmapFactory.java

设置的代码见doDecode方法:

frameworks/base/core/jni/android/graphics/BitmapFactory.cpp
static jobject doDecode(JNIEnv* env, SkStreamRewindable* stream, jobject padding, jobject options) {/*`......`*/SkImageDecoder* decoder = SkImageDecoder::Factory(stream);if (decoder == NULL) {return nullObjectReturn("SkImageDecoder::Factory returned null");}decoder->setSampleSize(sampleSize);decoder->setDitherImage(doDither);decoder->setPreferQualityOverSpeed(preferQualityOverSpeed);decoder->setRequireUnpremultipliedColors(requireUnpremultiplied);/*`......`*/
}

SkImageDecoder中并不反映全部Java层的配置项,而是如Mode解码模式,按函数参数传入.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/172734.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android之高级UI

系统ViewGroup原理解析 常见的布局容器: FrameLayout, LinearLayout,RelativeLayoout,GridLayout 后起之秀&#xff1a;ConstraintLayout,CoordinateLayout Linearlayout Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {if (mOrientation …

STK Components 二次开发-地面站传感器

上一篇我们说了创建地面站&#xff0c;那么这次我们在地面站添加一些特效。 1. 创建地面站 var locationPoint1 new PointCartographic(m_earth, new Cartographic(Trig.DegreesToRadians(117.17066), Trig.DegreesToRadians(31.84056), 240.359)); m_facility new Platfor…

【论文解读】Real-ESRGAN:使用纯合成数据训练真实世界的超分辨率图像

图一是4种超分方法的对比效果 。 0 摘要 尽管在盲超分辨率方面已经进行了许多尝试&#xff0c;以恢复具有未知和复杂退化的低分辨率图像&#xff0c;但它们仍然远远不能解决一般的真实世界退化图像。在这项工作中&#xff0c;我们将强大的 ESRGAN 扩展到一个实际的恢复应用程序…

Retrofit怎么返回一个JSON字符串?

项目用已经使用了 Retrofit&#xff0c;定义了接口方法&#xff0c;返回了 JSON 转换后的实体对象&#xff0c;炒鸡方便。但是总有意料之外的时候&#xff0c;比如我不需要返回实体对象&#xff0c;我要返回纯纯的 JSON 字符串&#xff0c;怎么办呢&#xff1f; 先看源码 通过…

什么是半监督学习

1 概述 1.1 定义 半监督学习&#xff08;Semi-Supervised Learning&#xff09;是机器学习中的一个重要分支&#xff0c;它介于监督学习和无监督学习之间。半监督学习利用少量标注数据和大量未标注数据共同训练模型&#xff0c;旨在充分挖掘未标注数据中潜在的信息和模式&…

git stash命令详解

git stash 工作区和暂存区的内容是公共的&#xff0c;不属于任何一个分支 所以在切换分支的时候务必保证工作区是干净的&#xff0c;否则切换过去的时候会导致那个分支也被修改了 如果不想add和commit&#xff0c;那可以用git stash将工作现场暂存一下 需要注意的是&#x…

【数据库】缓冲区管理器结构,几种常用替换策略分析,pin钉住缓冲区块防止错误的替换,以及缓冲区管理带来的代价优化

缓冲区管理 ​专栏内容&#xff1a; 手写数据库toadb 本专栏主要介绍如何从零开发&#xff0c;开发的步骤&#xff0c;以及开发过程中的涉及的原理&#xff0c;遇到的问题等&#xff0c;让大家能跟上并且可以一起开发&#xff0c;让每个需要的人成为参与者。 本专栏会定期更新&…

【算法】装备合成(二分)

链接&#xff1a;登录—专业IT笔试面试备考平台_牛客网 来源&#xff1a;牛客网 题目描述 牛牛有x件材料a和y件材料b&#xff0c;用2件材料a和3件材料b可以合成一件装备&#xff0c;用4件材料a和1件材料b也可以合成一件装备。牛牛想要最大化合成的装备的数量&#xff0c;于是…

重新开启GPT Plus充值通道——基于前端开发者工具

chatGPT PLUS充值通道的关闭 由于chatGPT用户激增&#xff0c;近日&#xff0c;OpenAI的CEO Sam Altman宣布需要暂停新用户对ChatGPT Plus的订阅。在X上&#xff0c;他表达了对于确保用户体验的承诺&#xff0c;同时也提到了用户可以通过应用程序内的通知功能来了解服务恢复的…

代码随想录算法训练营 ---第四十五天

前言&#xff1a; 昨天的题做过之后&#xff0c;今天的题基本上都很简单&#xff0c;但是要注重一下细节。 第一题&#xff1a; 简介&#xff1a; 动态规划五部曲&#xff1a; 1.确定dp数组的含义 dp[i]&#xff1a;爬到有i个台阶的楼顶&#xff0c;有dp[i]种方法 2.确定dp…

常见指令的数据通路和执行过程

作此篇的原因是17年19题&#xff1a; 本题选A&#xff0c;做的时候总感觉不够通透&#xff0c;因此把这题涉及到的内容全部看了一遍&#xff0c;顿时没有那种朦胧感了 零、五段式流水线&#xff1a; 以下均为MIPS设定&#xff1a;指令长度为32位&#xff0c;主存按字节编址&a…

gpt是如何进行训练的?

原理 gpt就是一个类似于成语接龙的游戏&#xff0c;根据之前的n个字符&#xff0c;预测下一个字符&#xff0c;那么gpt的输入和输出是如何构造的呢&#xff1f;比如给一个句子如下&#xff1a; sentence&#xff1a;如何理解gpt的原理。 构造gpt输入输入&#xff1a; input&am…

Echarts 最简单创建柱状图

设置容器 <div ref"myChart" style"width: 500px; height: 500px;"> </div>mounted() {//document渲染完成this.draw()}draw() {const myChart this.$echarts.init(this.$refs.myChart)//初始化对象myChart.setOption({ //参数配置项title: …

探索接口测试:SOAP、RestFul规则、JMeter及市面上的接口测试工具

引言 在当今软件开发领域&#xff0c;接口测试扮演着至关重要的角色。随着系统变得日益复杂和互联&#xff0c;对于内部和外部接口的测试变得愈发关键。接口测试不仅仅是验证接口的正确性&#xff0c;更是确保系统的稳定性、安全性和性能优越性的关键一环。 本篇博客将带您深入…

自动化任务:探索 Shell 脚本的实际应用

引言 在前一篇文章中&#xff0c;我们学习了 Shell 脚本的基础知识和语法。现在&#xff0c;让我们深入探讨 Shell 脚本在自动化任务中的实际应用。Shell 脚本能够帮助我们批量处理文件、定时执行任务等&#xff0c;提高工作效率。 批量处理文件 场景介绍&#xff1a; 假设我…

一次简单的 Http 请求异常处理 (请求的 url 太长, Nginx 直接返回 400, 导致请求服务异常)

1 结论 按照惯例直接说结论。 后台服务 A 有一个 Http 接口, 代码如下: RequestMapping(value "/user", method RequestMethod.GET) public List<UserInfoVo> getUserInfoByUserIds(RequestParam(value "userIds") List<String> userIds…

[Linux]进程等待

文章目录 3.进程等待3.1什么是进程等待3.2为什么要进程等待3.3如何进行进程等待?1.wait2.waitpid2.1函数的讲解2.2status的理解2.3代码理解 3.4学后而思1.直接用全局变量获取子进程退出码可以吗?如下2.进程具有独立性 退出码是子进程的数据 父进程是如何拿到退出码的3.对内存…

云匣子 FastJson反序列化RCE漏洞复现

0x01 产品简介 云匣子是租户连接云资源的安全管理工具&#xff0c;帮助云租户更加安全、精细的管理云上的虚拟机、数据库等资源。 云安宝结合多年的运维和安全实践&#xff0c;将云上的运维和安全有机结合&#xff0c;实现对运维过程的事前规划、事中控制和 事后审计。在此之上…

Handler系列-prepareMainLooper在哪里调用的

ActivityThread的main方法里调用prepareMainLooper public final class ActivityThread {public static void main(String[] args) {Looper.prepareMainLooper(); //创建sMainLooperLooper.loop();} } prepareMainLooper创建了sMainLooper public final class Looper {priva…

Redis常见面试问题

1、Redis分布式锁是怎么实现的 Redis实现分布式锁的7种方案-CSDN博客 2、Redis分布式锁会有什么问题 Redis实现分布式锁的7种方案-CSDN博客 3、Redis有哪些操作时间复杂度不是O(1)&#xff1f; Redis有一些操作的时间复杂度不是O(1)&#xff0c;例如&#xff0c;删除List、…