原标题:Android高斯模糊你所不知道的坑

本文作者

作者:mandypig

链接:

https://www.jianshu.com/p/d29841b1a4d5

本文由作者授权发布。

如果你想了解如何封装一个高斯模糊,可以参考之前的文章:

如何封装个好用的高斯模糊组件

本文作者给大家分享了一个自己在制作高斯模糊效果时,遇到的问题以及分析的思路。

高斯模糊的UI效果相信大家多多少少都有接触过,只是可能并没有真正在实际项目中去使用过,这次产品需求中正好涉及到了高斯模糊的展示效果,随便去研究了下这块东西,先上一个最终的效果图。

实际真机效果还是流畅的,录屏出来有点卡顿的感觉。

从效果还是可以看出,当点击签到按钮的时候,会伴随签到页面和分享按钮的自下向上的弹出效果,这个动画直接使用属性动画就能实现没什么好说的,最主要的就是后面的高斯模糊效果,仔细看的话可以发现是有一个从清晰逐渐模糊的渐变效果,这个不是本文重点,这里想说的是在实现自定义高斯模糊效果的时候自己所遇到的一个坑。

1

基础知识

实际上高斯模糊效果本身是没什么好说的,只要你去网上搜索一下相信能找到不少相关的实现,其中对于android开发来说最主要的实现方法就是使用render方法去实现,自己实现模糊效果的时候也是参考这些代码来完成。

写这篇文章主要不是来说高斯模糊是如何实现,而是想把自己在实现这个高斯模糊效果中遇到的一个大坑和大家分享下,这个坑大家不一定遇到到,尤其是从github直接找一个现成的高斯模糊控件,通过看源码几乎不可能发现这个大坑,只有真正去实现过这个效果的人才有可能发现这个问题,很明显自己就属于后者,还正巧踩到了这个坑上,这里就说下自己踩坑埋坑的心路历程。

高斯模糊实现思路

还是简单说下高斯模糊实现的简单思路,就是得到原图bitmap然后通过render处理获取到模糊后的bitmap,自定义的view在ondraw中去draw该bitmap。思路就是这么简单,这里说下需要注意的几个点。

(1)如何得到模糊后的bitmap,这个就是通过render来实现的,模板代码如下:

//创建一个缩小后的bitmap

Bitmap inputBitmap = Bitmap.createScaledBitmap(image, width, height, false);

//创建将在ondraw中使用到的经过模糊处理后的bitmap

Bitmap outputBitmap = Bitmap.createBitmap(inputBitmap);

//创建Render,IntrinsicBlur固定写法

Render rs = Render.create(context);

IntrinsicBlur blur = IntrinsicBlur.create(rs, Element.U8_4(rs));

//根据inputBitmap,outputBitmap分别分配内存

Allocation tmpIn = Allocation.createFromBitmap(rs, inputBitmap);

Allocation tmpOut = Allocation.createFromBitmap(rs, outputBitmap);

//设置模糊半径取值0-25之间,不同半径得到的模糊效果不同

blur.setRadius( 15);

blur.setInput(tmpIn);

blur. forEach(tmpOut);

//得到最终的模糊bitmap

tmpOut.copyTo(outputBitmap);

高斯模糊的核心代码差不多都是这样大同小异,上述代码需要注意一下的有三个地方:

1. blur.setRadius参数取值在0-25之前,超过这个范围程序会崩溃掉

2. Render,IntrinsicBlur,Allocation这些类需要使用v8中的类,所以需要在你的build.gradle加入以下配置

defaultConfig{

minSdkVersion15

targetSdkVersion 27

versionCode 1

versionName "1.0"

renderTargetApi 19

renderSupportModeEnabled true

}

3 巧妙得到“原图”,高斯模糊就是通过对原图处理得到一张模糊图,但是这里所说的原图并不是真正意义上的原图,听着有点拗口.

举个简单的列子,比如你想对一个500x500大小的imageview进行高斯模糊处理,得到该bitmap后还需要对该原图进一步压缩得到一张例如200x200大小的图.

为什么要这么做呢,其实最主要的原因还是因为高斯模糊处理过程本身就比较消耗性能,所以说理论上你处理的原图分辨率越小那么消耗时间也就越少,对原图进行压缩处理,主要就是考虑到性能上的问题,尤其是当你的app需要支持动态高斯模糊效果的时候,需要不停的去得到模糊的bitmap,如果每次的处理量都很大就会造成页面卡顿的问题,所以使用压缩后的原图优势显而易见,另一个原因就是高斯模糊得到的最终bitmap就是模糊的效果,原图压缩失真对最后的结果并没有太大的影响。

明白了上面的道理那么高斯模糊的模板代码你也就掌握了。

2

处理imageview模糊效果

先来看下具体效果,如图原图如下

使用上述代码处理后可以得到如下效果

看起来效果还是不错的吧,修改 blur.setRadius(15);的参数可以得到不同模糊效果的展示,关于原图的bitmap如何得到可以直接通过view.draw方法得到,简单粗暴,如果你的高斯模糊只是用来处理imageview就可以的话那么关于我说的那个坑你是不会遇到的。

3

遇坑:处理整个屏幕的模糊效果

这才是可能会遇到坑的地方,如果你想模糊的是整个屏幕,就如我文章开头的那种效果,那么这个坑会弄得你晕头转向,至少我开始看到呈现出来的效果时真的就是一脸懵逼,先来看一下没有模糊时的布局demo

布局代码就不贴了,有点长,花点时间完全可以弄一个类似的出来,这里说一下我对该布局模糊的整体思路,首先得到布局文件中的根布局main,这里使用的是一个 Framelayout,通过调用Framelayout的draw方法得到bitmap,对该bitmap压缩后得到二次原图,最后通过render处理得到模糊的bitmap并在ondraw方法中绘制出来。

根据以上思路我预期得到的效果图应该长这样才对

然而实际效果却是这样的

看起来似乎一样,但是仔细对比一下就能发现上面的那张图片似乎只是经过了半模糊的处理,边缘部分还是清晰可见,这显然不是我想要的效果。

4

填坑过程

为什么会造成这种问题,当你模糊一个imageview的时候效果非常完美,而当你去模糊一个viewgroup的时候就出现上述诡异问题,起初是怀疑和图片有关,于是换了一张其他图片发现结果还是这样。

然后排查代码检查代码逻辑是否有问题,这是一个比较漫长的过程把可能引起问题的代码注释重新运行,最后折腾了半天发现并没有任何卵用。

图片没问题,代码看起来也似乎没有问题那到底是哪里出了问题,无奈只好网上搜相关问题,只能说网络这么大可惜没有我想要的答案。

不过从网上的一些文章也算间接找到可能解决的方案,去github上找一个高斯模糊的控件,将该控件应用到我的布局上去,然后奇迹发生了,居然没有发生边缘清晰的异常,这只能说明是我自己的代码有问题才会导致这个问题,剩下的问题就是排查到底是什么代码引起的这个诡异问题。

这里说一句题外话为什么不直接使用github上的开源库,而是自己重新造轮子。

第一github上的开源库不能完全满足需求即使使用也要进行二次开发,耗费的时间成本可能比自己写还要多。

第二高斯模糊本身模板代码不复杂完全可以自己自定义。

第三很多人都喜欢说不要重复造轮子,但我想说的是你也要有这个本事去造这个轮子,更多的时候给你这个时间你都不一定有本事造成这个轮子才是主要原因,很多人只不过是个代码的搬运工而已,github有个类似效果就直接拿到用也不去研究下如何实现。

回到问题本身,又是半天的摸索,经过大量对比终于将可疑点定位到了一个关键的地方,我自己代码中是通过得到你要模糊的view,然后调用该view的draw方法得到bitmap,而github上的高斯模糊控件是通过decorview的draw方法得到bitmap,当我将自己的高斯模糊控件对应逻辑改成decorview之后,整个世界都清净了,我得到了和github一样的高斯模糊效果,那一刻可以说困扰了我两天的问题终于得到了解决,这种成就感不是单单使用一个开源库所能得到的。

为什么??

开心没有多久,一个巨大的疑问又让我感到郁闷,为什么使用decorview就没问题,而使用其他viewgroup就会出问题,可能猜测到的原因就是decorview内部做了什么操作,但是做了什么操作,面对这么多的代码可以说真的就是无从下手去找,猜测的第一个原因是不是和硬件加速有关,测试后发现并没有用处,继续懵逼中......,换几张图片看看能不能发现一点蛛丝马迹,以下效果引起了我注意

上述效果是我调用布局中的根view的draw方法得到的模糊效果,可以发现一个奇怪的地方,似乎箭头模糊后原图悬浮到了模糊效果之上,为了验证猜测我在自定义高斯模糊的ondraw方法中绘制了一个白色的背景,如果原图悬浮在高斯模糊自定义控件之上的话那么将会出现白色背景不能覆盖原图的现象,反之白色背景将会遮盖掉原图。

运行看了下效果,真可谓无心插柳柳成荫,运行后的效果居然变成了:

这不正是我想要的最终效果吗!!

在给自定义高斯模糊控件设置一个白色背景居然就解决了困扰我两天的难题,接下来就是思考为什么这种方法是可行的,一个非常合理的解释就是如图所示:

原来的布局中红框圈起来的部分是没有背景色的,高斯模糊在处理模糊效果的时候将图中圆圈内部边缘的颜色值和圆圈外部的透明色进行处理最终得到的颜色值不变,这样就产生了一种边缘清晰可见的视觉效果,如图所示:

到此才可以说算是找到了导致我自定义高斯模糊效果异常的真正原因。

decorview什么时候设置背景色

最后一个困扰我的问题,为什么调用decorview的draw方法就没问题,经过上述我总结的原因,唯一一个解释就是decorview默认自带一个背景色!!

那么这个背景色是在什么时候设置的呢,只能在源码里面去找答案了,这次就不再是大海捞针的看源码了,在phonewindow--》installDecor--》generateLayout方法中最终找到了我想要的答案:

TypedArray a = getWindowStyle();

...

if(mBackgroundResource == 0) {

mBackgroundResource = a.getResourceId(

R.styleable.Window_windowBackground, 0);

}

...

final Drawable background;

if(mBackgroundResource != 0) {

background = getContext().getDrawable(mBackgroundResource);

} else{

background = mBackgroundDrawable;

}

mDecor.setWindowBackground(background);

关键代码都已经给出,可以看到最终decorview调用了setWindowBackground设置了一个背景色!!,到此所有的疑问都得到了非常完美的解答。

总结

总结下自定义高斯模糊所遇到的问题,由于viewgroup可能存在没有默认背景色的缘故,导致会发现模糊后的效果比较诡异的情况,而decorview默认在生成的时候就自带背景色所以使用decorview来实现高斯模糊不会出现问题,文章重点不是在说如何实现高斯模糊效果,而是解决问题的一个思路。

PS:有不少同学,通过这篇拥抱Android开发的变化,才能拥抱未来文章购买了绍文的课程,已经更新2讲了,记得学习和实践 github 上的课后作业,我会不定期提醒的。返回搜狐,查看更多

责任编辑:

android 开源 高斯模糊_Android高斯模糊你所不知道的坑相关推荐

  1. Android Context完全解析,你所不知道的Context的各种细节

    转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/47028975 前几篇文章,我也是费劲心思写了一个ListView系列的三部曲,虽然 ...

  2. android仿秒拍源码,你所不知道的程序员 程序员其实真的很…【Bus Weekly】三十六期...

    原标题:你所不知道的程序员 程序员其实真的很-[Bus Weekly]三十六期 快,点击蓝色"字体"关注这个公众号,一起涨姿势- 现如今,程序员在中国的科技 圈可以说已经达到了举足 ...

  3. android factory,“工厂映像”(factory image)是什么?你所不知道的Android问题

    趋势网(微博|微信)讯: "工厂映像"(factory image)是什么?你所不知道的Android问题. 从多种选择中挑选自己喜欢的终端,从而使高度可定制成为可能,这便是And ...

  4. 被低估的css滤镜,你所不知道的 CSS 滤镜技巧与细节

    本文主要介绍 CSS 滤镜的不常用用法,希望能给读者带来一些干货! 系列 CSS 文章汇总在我的 Github ,持续更新,欢迎点个 star 订阅收藏. OK,下面直接进入正文.本文所描述的滤镜,指 ...

  5. 互联网人群画像和你所不知道的真相

    作为新时代互联网营销的关键部分,人群画像引起了诸多兴趣,近年颇为风靡.几乎所有的互联网广告供应商都不约而同的强调,他们有足够精确的人群画像数据,确保能够找到广告主真正的受众.但是事情果真如此吗?人群画 ...

  6. 一些你所不知道的VS Code插件

    摘要: 你所不知道的系列. 原文:提高 JavaScript 开发效率的高级 VSCode 扩展之二! 作者:前端小智 Fundebug经授权转载,版权归原作者所有. 作为一名业余爱好者.专业人员,甚 ...

  7. 高手才知道!七个你所不知道的 D3.js 秘技

    D3.js ,当前最火红的视觉化套件,你用过了吗?越来越多人使用 D3.js 来开发视觉化专题,但- 你对 D3.js 的了解又到哪里呢?这次我们就带大家一起来看看一些 D3.js 很重要.大家却又普 ...

  8. 那些你所不知道的arXiv使用技巧

    作者:Tom Hardy Date:2020-12-23 来源:那些你所不知道的arXiv使用技巧

  9. 系统调优,你所不知道的TIME_WAIT和CLOSE_WAIT

    https://my.oschina.net/fdhay/blog/638631 高性能网络 | 你所不知道的TIME_WAIT和CLOSE_WAIT 2016-02-18 大房 大房说 本文是我将最 ...

最新文章

  1. C语言博客作业--字符数组
  2. 学历是铜牌,能力是银牌,人脉是金牌,思维是王牌——有感
  3. 【数据结构与算法】之深入解析“最好买卖股票的时机含冷冻期”的求解思路与算法示例
  4. 通向架构师的道路(第一天)之Apache整合Tomcat
  5. Linux学习-11月12日(Apache安装)
  6. jQuery应用实例2:简单动画
  7. 前端学习(1910)vue之电商管理系统电商系统之完成用户的修改
  8. MinIO Client完全指南 ​​​​​​​
  9. linux shell脚本举例,Shell脚本去重的几种方法实例
  10. 小学计算机课评价方案,小学信息技术课堂评价方法的研究
  11. 适配器模式之状态模式
  12. 转 Git-fatal: unable to access 'xxx' : Could not resolve host: xxx
  13. 计算机无法选择字体,电脑系统字体缺失怎么办
  14. angular框架的SmartAdmin模板 如何请求后台数据
  15. 水晶报表CrystalReports很强大也很简单!
  16. JS搜索省份匹配出省份的所有城市
  17. ppt模板如何制作动态时间轴?
  18. css深入理解flex布局中的剩余空间分配规则——flex-grow,flex-shrink和flex-basis
  19. zotero与Obsidian联动笔记(一):ob中直接调用zotero的文献,并生成笔记
  20. 单片机c语言给变量赋值,单片机c语言变量的定义和赋值

热门文章

  1. Qualcomm 音频学习一
  2. Jenkins基础:API:10:使用API更新进行节点的连接和断开
  3. 【Python爬虫教学】百度篇·手把手教你抓取百度搜索关键词后的页面源代码
  4. 2019元旦消费大数据
  5. 360需要的,不是打工者——周鸿祎在360新员工入职培训上的讲话
  6. 系统迁移到固态硬盘--高贵的ASUS版
  7. 模式识别——0.绪论
  8. android动态贴纸实现原理,人脸动态贴纸sdk算法详解,人脸动态贴纸功能如何实现...
  9. 亚马逊站外引流如何做?解析厨电大卖的高曝光秘诀
  10. 抓取网易云音乐歌曲热门评论生成词云(转)