当朋友圈的你们才开始分享支付宝AR红包的消息的时候,我已经对它动了一二三四次歪脑筋了,虽然事实证明并不是那么顺利,至今我也只在电脑前识别出5个不知道在哪里的红包,其中一个还因为定位信息不符开不了。

昨天上午听公司的小伙伴说起支付宝新推出的AR红包,LBS加图像识别的另一个创新,可说是支付宝在社交互动这一块终于打出来的一张好牌。然而,在许多人还在奔走相告这个消息的时候,我已经和小伙伴们对它动起了歪心思。

首先当然是抓包,但并没有结果。想必是汲取了之前微信朋友圈的红包照片的教训,通过其他流的方式来传输,或许加了密?都是猜测,但是图片是不能直接抓到的了。

通过对AR红包的体验,我们能想到找红包的大概原理就是打开摄像头时,不断获取当前的图像,然后通过图片识别算法计算与藏红包时拍的图片的相似程度。这种效果的处理方案可以是:通过感知哈希算法对藏红包的图片生成一组摘要信息(指纹),然后在找红包时不断获取当前图像生成指纹,再通过汉明距离算法进行比较, 结果越接近就表示越相似,用户很难拍出一张与原图完全一样的图片,所以大概会设定一个阙值,当小于这个阙值时就认为图片是一样的。

类似于这样的处理方式,它有一个明显的特点就是忽略细节。比如上面所提的感知哈希算法,就是先将图片缩小成8*8的大小,再转为64级的灰度,这种情况下的一些细小差别已经变成几乎不存在了。

很直接的一个证明就是,我在找红包时按住找线索,在弹出图片时对手机截屏,然后把线索图在电脑屏幕上放大,再通过手机上直接扫图。就是这样简单的方式,对于少数虽然被加上了斑马线条的图片,还是可以勉强识别出来。就是比较考验手机对准的角度,而且对图片要求较高,得是缩小之后看起来那些黑条对图片的识别影响不大的图片。

我通过上面的方式已经在电脑前收了几个不知道是在哪里的红包。但是这方式还是略显笨拙,有没有什么方式能够提高成功率呢?

注意线索图,这些黑条是等距离的遮在图片上的,并且没被挡住的部分要比黑条大一点。并且线索图是一个正方形且所在位置固定。那么,对于一些比较简单的图片,可以通过把每条黑条上面的图片内容复制,并遮到黑条上面,就可以把黑条去掉,而得到一张与原图较相似的图片了。我打开了ubuntu上的gimp手动试了一下,又识别出一张图片拿到一个红包。既然这种方式可行,那就可以用代码实现一下了:

首先在找红包时按住查看线索,截屏,然后用gimp打开,定位到在我1080p带虚拟导航栏的手机中,线索图在图片中的左上角及右下角的xy坐标分别是330, 963, 750, 1383。

拿到这些数据之后,把图片加到项目中,就可以开始写代码了。先实现后优化,就不考虑代码性能等各方面的问题了,直接简单粗暴地自定义一个View并在主线程中计算处理(先实现嘛)并绘制,代码如下:

public class ARView extends View {
    private Paint mPaint;
    private int dividerHeight = 7;// 这个值是慢慢试出来的
    private int pictureHeight = 8;// 这个不是整图的高,而是每一条没有被挡住的高
    private Bitmap mBitmap;

    public ARView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        mPaint = new Paint();
    }

    @android.support.annotation.RequiresApi(api = Build.VERSION_CODES.KITKAT)
    @Override
    protected void onDraw(Canvas canvas) {
        if (mBitmap == null) {
            final int width = getWidth();//  获取View宽,为了最后把图片放大到填满
            InputStream is = null;
            try {
                is = getContext().getAssets().open("a.png");//我截的图,直接丢assets里了
                final BitmapRegionDecoder decoder = BitmapRegionDecoder.newInstance(is, true);//只需要加载线索图那一部分内容即可
                final Rect rect = new Rect(330, 963, 750, 1383);// 前面所定位到的线索图的坐标
                final Bitmap origin = decoder.decodeRegion(rect, null);//拿到线索图
                final Bitmap bitmap = origin.copy(Bitmap.Config.ARGB_8888, true);//cp一个可以通过setPixel修改的新的bitmap
                if (!origin.isRecycled()) {
                    origin.recycle();
                }
                if (!decoder.isRecycled()) {
                    decoder.recycle();
                }
                restoreBitmap(bitmap);//在这个方法里处理图片

                final Matrix matrix = new Matrix();
                final float scale = (float) width / bitmap.getWidth();//计算控件宽度与图片宽度的比,用于放大图片显示
                matrix.setScale(scale, scale);
                mBitmap = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, false);
            } catch (IOException e) {
                e.printStackTrace();
            } finally {
                if (is != null) {
                    try {
                        is.close();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }
        }
        canvas.drawBitmap(mBitmap, 0, 0, mPaint);
    }

    private void restoreBitmap(Bitmap bitmap) {
    // ...
    }
}

大致代码就是上面这样了。

restoreBitmap的方法是通过遍历每一个黑色的像素点,然后把其上面没挡住的像素点设置上去。过程大概如下:

假设原图是这样的,数字是显示出来的部分,星号是黑条:

1234567887654321
****************
3456789009876543
****************

那么我们只需要把它变成如下所示就可以了:

1234567887654321
1234567887654321
3456789009876543
3456789009876543

这种处理方式对图片本身有一定高求,比如图片的像素内容在竖直方面应该是变化不大的,这样还原起来才比较逼真。

在一开始要找出黑条高度及没有被挡住的图片条的高度的时候,我们可以先对黑条设置为另一个颜色,看能不能把黑条完全覆盖,就可以知道我们最终所设置的这两个值是否正确了。

具体实现就是循环一下图片的高度(每次加上图片图及黑色条的高度),再循环黑条的高度,再循环一下宽度,逐个设置像素点(你优化成两个循环我也没话说)。代码如下:

    private void restoreBitmap(Bitmap bitmap) {
        final int width = bitmap.getWidth();
        final int height = bitmap.getHeight();
        final float itemHeight = dividerHeight + pictureHeight;
        out:
        for (int y = 3 + pictureHeight; y < height; y += itemHeight) {//第一条要高一点
            for (int offset = 0; offset < dividerHeight; offset++) {
                if ((y + itemHeight) >= height) {
                    break out;
                }
                for (int x = 0; x < width; x++) {
                    bitmap.setPixel(x, y + offset, bitmap.getPixel(x, y - pictureHeight + offset + 1));
//                    bitmap.setPixel(x, y + offset, bitmap.getPixel(x, y - 2));//另一种处理方式,拉伸黑条上面的那一行像素,没试过识别效果
//                    bitmap.setPixel(x, y + offset, Color.BLUE); //用于测试是否能完全覆盖黑条
                }
            }
            // 处理黑线
//            for (int x = 0; x < width; x++) {
//                bitmap.setPixel(x, y + dividerHeight, bitmap.getPixel(x, y + dividerHeight - 1));
//            }
        }
    }

下图是我处理之后的最终结果:

另外,用没被遮挡的图片条覆盖黑色条是我这里的处理方法。昨晚一个朋友在朋友圈里还提出另一种猜想,即把没被遮挡的图片条拉伸到覆盖到黑色条上面。这也是一种不错的处理方案。我这里没有试验。

还有一个技巧,由于没被遮挡住的图片条与黑色条高度相差不大,所以可以处理之后还是会有比较细的黑线。这时候可以先放大再处理,或者是再用图片的内容覆盖到黑线上。效率会稍低点,但效果应该会更好。

PS:刚才打开支付宝又去看了一下AR红包,发现新的图片所加的黑条粗细已经是不一样的了。不得不说,支付宝的响应真快。

说一说关于破解支付宝AR红包的事的更多相关文章

  1. 支付宝AR红包引出Python中的PIL小试

    这两天支付宝AR红包火了,周围的同学全在玩.可是我一直在想这个原理是什么?通过请教大神和思考,知道了它有两个限定条件:GPS地理位置和图片的识别.所以,只要我们有了这两个限定条件,就不难进行该红包的破 ...

  2. Python + PIL 处理支付宝AR红包

    思路比较简单:1.对图片进行锐化处理:2.设(r_h, g_h, b_h)为支付宝遮罩黑条的RGB值,以此为中心,查找半径为Diff_radius的范围内所有的色值: 3.对每一行符合步骤2的像素点个 ...

  3. 支付宝AR抢红包?前端轻松就破解~

    近期阿里搞了各LBS+AR实景的红包玩法,小伙伴们在公司里都玩疯了~ 有时候为了抢一个红包,会跑到另一个地方去拍照,虽然略麻烦,但整体的互动还是很有意思的. 不过对于机智的前端童鞋来说,只需要简单的一 ...

  4. 支付宝AR实景红包上线不久即遭破解,官方已提高技术门槛

    临近春节,阿里巴巴和腾讯的红包大战可谓下足功夫,上周支付宝推出了AR实景红包,该玩法基于"LBS+AR+红包"的方式,类似与今年火爆全球的AR手游Pekomon Go ,只不过这次 ...

  5. 手机浏览网页或打开App时莫名弹出支付宝领红包界面的原因及应对措施

    自从支付宝推出扫码领红包活动后,这种模式独特的赏金机制,短时间内吸引了大量的关注,但是随之也产生了很多的问题,比由于如在赏金的驱动下,微信群里铺天盖地的红包口令,朋友圈里各式各样的领红包二维码图片, ...

  6. Unity -- 用EasyAR制作出AR红包

    第一部分:前言 实现功能:打开APP或运行该demo后,会从天而降红包,有些红包是空的(大一点的),抖动的红包里面“有钱”,点击之后会产生相应的交互.   第二部分:预览 所用到的资源:   红包模型 ...

  7. android蜂巢效果、环形菜单、Kotlin影视应用、简约时钟、查看导出App、支付宝AR扫码效果等源码

    Android精选源码 一个蜂巢布局管理器,外观帅气外,动画效果也是很赞 一个基础 UI 框架项目,实现不同布局格式的混排 仿建行app效果,一个环形菜单的布局管理器源码 基于组件化实现的一款用Kot ...

  8. Android破解——支付宝内购破解方法总结

    支付宝破解三种方式: 想学一下支付宝内购的相关知识,但是搜索了论坛,发现但是没有相关的帖子,于是便是打算自己来写一篇总结 一.9000的十六进制代码修改 搜索9000的十六进制,也就是搜索0x2328 ...

  9. 支付宝推AR实景红包,抢红包得拼脑力和体力

    近年春节,各大互联网平台都会借机掀起"红包大战",其中支付宝和微信的红包玩法备受用户关注.今年,微信尚未公布春节红包相关的方案信息,不过,今天支付宝率先推出"AR实景红包 ...

随机推荐

  1. MySQL实现全关联 full outer join

    SQL LEFT JOIN 关键字 LEFT JOIN 关键字会从左表 (table_name1) 那里返回所有的行,即使在右表 (table_name2) 中没有匹配的行. LEFT JOIN 关键 ...

  2. javascript 错误处理和堆栈追踪浅析

    合理地处理堆栈信息能使你清除无用的数据, 而只专注于有用的数据. 同时, 当更好地理解 Errors 对象及其相关属性之后, 能有助于你更充分地利用 Errors. (函数的)调用栈是怎么工作的 在谈 ...

  3. javaIO流--Writer,Reader

    Writer /** *<li> Writer中定义的一个重要的方法: * public void writer(String str)throws IOException; */ pac ...

  4. BZOJ4894 天赋

    Description 小明有许多潜在的天赋,他希望学习这些天赋来变得更强.正如许多游戏中一样,小明也有n种潜在的天赋,但有 一些天赋必须是要有前置天赋才能够学习得到的.也就是说,有一些天赋必须是要在 ...

  5. [HNOI2001]软件开发

    题目描述 某软件公司正在规划一项n天的软件开发计划,根据开发计划第i天需要ni个软件开发人员,为了提高软件开发人员的效率,公司给软件人员提供了很多的 服务,其中一项服务就是要为每个开发人员每天提供一块 ...

  6. [SDOI2009]学校食堂Dining

    题目描述 小F 的学校在城市的一个偏僻角落,所有学生都只好在学校吃饭.学校有一个食堂,虽然简陋,但食堂大厨总能做出让同学们满意的菜肴.当然,不同的人口味也不一定相同,但每个人的口味都可以用一个非负整数 ...

  7. bzoj 1076: [SCOI2008]奖励关

    Description 你正在玩你最喜欢的电子游戏,并且刚刚进入一个奖励关.在这个奖励关里,系统将依次随机抛出k次宝物,每次你都可以选择吃或者不吃(必须在抛出下一个宝物之前做出选择,且现在决定不吃的宝 ...

  8. WISCO信息组NOIP模拟赛-数据结构

    传送门 差分+暴力 #include<cstdio> #include<cstdlib> #include<algorithm> #include<cstri ...

  9. [USACO 5.1.3]乐曲主题

    Description 我们用N(1 <= N <=5000)个音符的序列来表示一首乐曲,每个音符都是1..88范围内的整数,每个数表示钢琴上的一个键.很不幸这种表示旋律的方法忽略了音符的 ...

  10. [UOJ UNR #2]积劳成疾

    来自FallDream的博客,未经允许,请勿转载,谢谢. 传送门 区间最大值的题emmmm 想到构建笛卡尔树,这样自然就想到了一种dp f[i][j]表示大小为i的笛卡尔树,根的权值是j的答案. 转移 ...