需要全部代码请点赞关注收藏后评论区留言~~~
一、二维码基本内容介绍
条形码只能表达十几位数字编码,无法表示更复杂的数据。 二维码在二维方格上描出一个个黑点,从而表达更丰富的信息。 二维码早已在手机App中广泛使用,不管是添加好友还是支付收款,只要出示二维码让别人扫一扫,滴的一下便轻松操作完成了
二维码在二维方格上描出一个个黑点,从而表达更丰富的信息,二维码中每个点有是(显示黑点)和否(不显示黑点)两种状态,二维码的左上角、右上角、左下角各有一个带黑框的黑色方块,这三个黑方块用于确定二维码的上下左右方位。因为用户可能斜着扫描,也可能倒过来扫描,通过三个角落的黑色方块便能将二维码旋转至正常方向,另外二维码中间放着公众号的logo图标,并不影响扫描识别,这是因为二维码留出了一定的容错率,即使部分区域被遮盖或者被污损,扫码软件依然能够根据剩余的大部分区域自动纠错,从而解析得到原始的编码信息。
当然,功能如此强悍的二维码可不是随随便便生成的,而要根据特定的算法将一串文本经过编码处理而形成的,若要使用App生成二维码,需要借助谷歌开源的zxing库,在代码中引入二维码撰写器QRCodeWriter,并调用撰写器的encode方法即可生成二维码位图
二、生成自己的二维码
运行测试App效果如下 可在下拉框中选择二维码的不同容错率,对应的二维码形状也会发生变化
代码如下
package com.example.face; import androidx.appcompat.app.AppCompatActivity; import android.graphics.Bitmap; import android.graphics.Color; import android.os.Bundle; import android.os.Environment; import android.text.TextUtils; import android.util.Log; import android.view.View; import android.widget.AdapterView; import android.widget.ArrayAdapter; import android.widget.EditText; import android.widget.ImageView; import android.widget.Spinner; import android.widget.Toast; import com.example.face.util.BitmapUtil; import com.example.face.util.DateUtil; import com.google.zxing.BarcodeFormat; import com.google.zxing.EncodeHintType; import com.google.zxing.common.BitMatrix; import com.google.zxing.qrcode.QRCodeWriter; import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel; import java.util.HashMap; import java.util.Map; public class GenerateQrcodeActivity extends AppCompatActivity { private final static String TAG = "GenerateQrcodeActivity"; private EditText et_content; // 声明一个编辑框对象 private ImageView iv_qrcode; // 声明一个图像视图对象 private Bitmap mBitmap; // 声明一个位图对象 private ErrorCorrectionLevel mErrorRate; // 容错率 @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_generate_qrcode); et_content = findViewById(R.id.et_content); iv_qrcode = findViewById(R.id.iv_qrcode); findViewById(R.id.btn_generate).setOnClickListener(v -> { String content = et_content.getText().toString(); if (TextUtils.isEmpty(content)) { Toast.makeText(this, "请先输入原始文本", Toast.LENGTH_SHORT).show(); return; } // 生成原始文本对应的二维码位图 mBitmap = createQrcodeBitmap(content, mErrorRate); iv_qrcode.setImageBitmap(mBitmap); // 设置图像视图的位图对象 }); findViewById(R.id.btn_save).setOnClickListener(v -> { if (mBitmap == null) { Toast.makeText(this, "请先生成二维码图片", Toast.LENGTH_SHORT).show(); return; } // 生成图片文件的保存路径 String path = String.format("%s/%s.jpg", getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS).toString(), DateUtil.getNowDateTime()); BitmapUtil.saveImage(path, mBitmap); // 把位图保存为图片文件 BitmapUtil.notifyPhotoAlbum(this, path); // 通知相册来了张新图片 Toast.makeText(this, "成功保存二维码图片:" + path, Toast.LENGTH_SHORT).show(); }); initErrorSpinner(); // 初始化容错率下拉框 } // 初始化容错率下拉框 private void initErrorSpinner() { ArrayAdapter<String> errorAdapter = new ArrayAdapter<>(this, R.layout.item_select, errorNameArray); Spinner sp_error = findViewById(R.id.sp_error); sp_error.setPrompt("请选择容错率"); sp_error.setAdapter(errorAdapter); sp_error.setOnItemSelectedListener(new ErrorSelectedListener()); sp_error.setSelection(0); } private String[] errorNameArray = {"30%", "25%", "15%", "7%"}; private ErrorCorrectionLevel[] erroLevelArray = {ErrorCorrectionLevel.H, ErrorCorrectionLevel.Q, ErrorCorrectionLevel.M, ErrorCorrectionLevel.L}; class ErrorSelectedListener implements AdapterView.OnItemSelectedListener { public void onItemSelected(AdapterView<?> arg0, View arg1, int arg2, long arg3) { mErrorRate = erroLevelArray[arg2]; } public void onNothingSelected(AdapterView<?> arg0) {} } // 生成原始文本对应的二维码位图 private Bitmap createQrcodeBitmap(String content, ErrorCorrectionLevel errorRate) { if (TextUtils.isEmpty(content)) { return null; } int width = content.length()*6; // 二维码图片的宽度 int height = width; // 二维码图片的高度 int margin = width / 20; // 二维码图片的空白边距 Log.d(TAG, "content="+content+",width="+width+",height="+height+",margin="+margin+",errorRate="+errorRate.name()); Map<EncodeHintType, Object> hints = new HashMap<>(); hints.put(EncodeHintType.MARGIN, margin); // 设置空白边距 hints.put(EncodeHintType.CHARACTER_SET, "UTF-8"); // 设置字符编码格式 hints.put(EncodeHintType.ERROR_CORRECTION, errorRate); // 设置容错率 try { // 根据配置参数生成位矩阵对象 BitMatrix bitMatrix = new QRCodeWriter().encode(content, BarcodeFormat.QR_CODE, width, height, hints); // 创建像素数组,并根据位矩阵对象为数组元素赋色值 int[] pixels = new int[width * height]; for (int y = 0; y < height; y++) { for (int x = 0; x < width; x++) { if (bitMatrix.get(x, y)) { // 返回true表示黑色色块 pixels[y * width + x] = Color.BLACK; } else { // 返回false表示白色色块 pixels[y * width + x] = Color.WHITE; } } } // 创建位图对象,并根据像素数组设置每个像素的色值 Bitmap bitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888); bitmap.setPixels(pixels, 0, width, 0, 0, width, height); if (bitmap.getWidth() < 300) { // 图片太小的话,要放大图片 bitmap = BitmapUtil.getScaleBitmap(bitmap, 300.0/bitmap.getWidth()); } return bitmap; } catch (Exception e) { e.printStackTrace(); return null; } } }
创作不易 觉得有帮助请点赞关注收藏~~~