作者:某人_Valar
如需转载请保留原文链接
zxing的GitHub地址:https://github.com/zxing/zxing
之前的项目用到了zxing生成二维码,发现周围的白框大的感人,使用EncodeHintType.MARGIN设置也没什么效果。今天看了一下zxing的源码,才发现其中的缘由。
先来看我们是怎么使用zxing生成二维码的
/*
* url :要生成二维码的网址(String类型)
* 250:生成二维码的宽高(int类型)
这里的generateBitmap()方法直接返回一个bitmap对象,便于之后直接放入ImageView
*/
Bitmap mBitmap = MyQRcode.generateBitmap(url,250,250);
MyQRcode文件中的generateBitmap()方法
public static Bitmap generateBitmap(String content, int width, int height) {
QRCodeWriter qrCodeWriter = new QRCodeWriter();
Map<EncodeHintType, Object> hints = new HashMap<>();
hints.put(EncodeHintType.CHARACTER_SET, "utf-8");
hints.put(EncodeHintType.MARGIN,1); //先设置margin为1
try {
BitMatrix encode = qrCodeWriter.encode(content, BarcodeFormat.QR_CODE, width, height, hints);
int[] pixels = new int[width * height];
for (int i = 0; i < height; i++) {
for (int j = 0; j < width; j++) {
if (encode.get(j, i)) {
pixels[i * width + j] = 0x00000000;
} else {
pixels[i * width + j] = 0xffffffff;
}
}
}
return Bitmap.createBitmap(pixels, 0, width, width, height, Bitmap.Config.RGB_565);
} catch (WriterException e) {
e.printStackTrace();
}
return null;
}
顺着源码点下去,最后发现是在renderResult()方法里对宽高做了一系列的处理
image.png image.png image.pngrenderResult里的width,height是我们一开始设定的宽高,
quietZone是我们之前设定的margin。
zxing并没有将这个值直接使用,而是经过一系列的计算得出了 leftPadding 与 topPadding。
在 renderResult 中注意下面这些参数
image.pnginputWidth ,inputHeight 是 zxing 根据我们所传的url与指定的BarcodeFormat.QR_CODE规则生成的。
因为参数都是在源码中,没法直接打印出来,那就debug一下,设几个断点看一下里面的值都是多少
image.png可以看到生成的inputWidth 和 inputHeight 为33,
quietZone就是我们之前传入的1,
width,height都是之前设置的250。
multiple=7,
内容区为33x33,而我们需要的大小为250x250,33放大7倍后为231,距离250还差19。
19/2 =9
最后经过计算得出了 leftPadding 与 topPadding 都为 9。
如果此时将margin设为 4, 宽和高还是设为 250
hints.put(EncodeHintType.MARGIN,4);
故multiple=6,leftPading =(250 - 33*6)/2 = 26,两边就有很大的空白。
image.pngzxing会采用这种策略,是为了提高二维码的容错率,以免出现失真导致二维码无法扫描。
了解了zxing的这种计算机制后,我们就可以根据想要的边框大小来传入相应的值了。
网友评论