最后
我见过很多技术leader在面试的时候,遇到处于迷茫期的大龄程序员,比面试官年龄都大。这些人有一些共同特征:可能工作了7、8年,还是每天重复给业务部门写代码,工作内容的重复性比较高,没有什么技术含量的工作。问到这些人的职业规划时,他们也没有太多想法。
其实30岁到40岁是一个人职业发展的黄金阶段,一定要在业务范围内的扩张,技术广度和深度提升上有自己的计划,才有助于在职业发展上有持续的发展路径,而不至于停滞不前。
不断奔跑,你就知道学习的意义所在!
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
// 字符串内容判空
if (TextUtils.isEmpty(content)) {
Toast.makeText(getApplicationContext(),“输入为空!”,Toast.LENGTH_LONG).show();
return null;
}
Map<EncodeHintType, String> hints = new HashMap<>();
//格式utf-8
hints.put(EncodeHintType.CHARACTER_SET, “utf-8”);
//空白边距设置
hints.put(EncodeHintType.MARGIN, “1”);
try {
//生成BitMatrix对象encode
BitMatrix encode = 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 (encode.get(x, y)) {
pixels[y * width + x] = color_point;
} else {
pixels[y * width + x] = color_back;
}
}
}
Bitmap bitmap = Bitmap.createBitmap(pixels, 0, width, width, height, Bitmap.Config.RGB_565);
if(logoBitmap != null){
return addLogo(bitmap, logoBitmap, logoPercent);
}
return bitmap;
} catch (WriterException e) {
e.printStackTrace();
return null;
}
}
- 从网上下载一张图片,存放在drawable文件中,然后在GenerateQRcode文件中的
onCreate
引用,将结果注入布局的ImageView组件中
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.generate_qrcode);
btn = (Button)findViewById(R.id.btn);
editText = (EditText)findViewById(R.id.editText);
imageView = (ImageView)findViewById(R.id.imageView);
btn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
String content = editText.getText().toString();
Resources res = getResources();
Bitmap logoBitmap= BitmapFactory.decodeResource(res,R.drawable.cat);
//Bitmap qrBitmap = generateSimpleBitmap(content,256, 256);
Bitmap qrBitmap = generateQRCodeBitmap(content, 256, 256, Color.BLUE, Color.GREEN, logoBitmap, 0.2F);
imageView.setImageBitmap(qrBitmap);
}
});
}
- 实现结果
package com.example.qrcodetest;
import com.google.zxing.BarcodeFormat;
import com.google.zxing.EncodeHintType;
import com.google.zxing.WriterException;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.qrcode.QRCodeWriter;
import android.content.res.Resources;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.os.Bundle;
import android.graphics.Bitmap;
import android.text.TextUtils;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.Toast;
import androidx.appcompat.app.AppCompatActivity;
import java.util.HashMap;
import java.util.Map;
public class GenerateQRcode extends AppCompatActivity {
private Button btn;
private ImageView imageView;
private EditText editText;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.generate_qrcode);
btn = (Button)findViewById(R.id.btn);
editText = (EditText)findViewById(R.id.editText);
imageView = (ImageView)findViewById(R.id.imageView);
btn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
String content = editText.getText().toString();
Resources res = getResources();
Bitmap logoBitmap= BitmapFactory.decodeResource(res,R.drawable.cat);
//Bitmap qrBitmap = generateSimpleBitmap(content,256, 256);
Bitmap qrBitmap = generateQRCodeBitmap(content, 256, 256, Color.BLUE, Color.GREEN, logoBitmap, 0.2F);
imageView.setImageBitmap(qrBitmap);
}
});
}
private Bitmap generateSimpleBitmap(String content, int width, int height) {
// 字符串内容判空
if (TextUtils.isEmpty(content)) {
Toast.makeText(getApplicationContext(),“输入为空!”,Toast.LENGTH_LONG).show();
return null;
}
QRCodeWriter qrCodeWriter = new QRCodeWriter();
Map<EncodeHintType, String> hints = new HashMap<>();
hints.put(EncodeHintType.CHARACTER_SET, “utf-8”);
try {
BitMatrix encode = 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 (encode.get(x, y)) {
pixels[y * width + x] = 0x00000000;//黑色
} else {
pixels[y * width + x] = 0xFFFFFFFF;//白色
}
}
}
return Bitmap.createBitmap(pixels, 0, width, width, height, Bitmap.Config.RGB_565);
} catch (WriterException e) {
e.printStackTrace();
}
return null;
}
public Bitmap generateQRCodeBitmap(String content, int width, int height,
int color_point, int color_back,
Bitmap logoBitmap, float logoPercent) {
// 字符串内容判空
if (TextUtils.isEmpty(content)) {
Toast.makeText(getApplicationContext(),“输入为空!”,Toast.LENGTH_LONG).show();
return null;
}
Map<EncodeHintType, String> hints = new HashMap<>();
//格式utf-8
hints.put(EncodeHintType.CHARACTER_SET, “utf-8”);
//空白边距设置
hints.put(EncodeHintType.MARGIN, “1”);
try {
//生成BitMatrix对象encode
BitMatrix encode = 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 (encode.get(x, y)) {
pixels[y * width + x] = color_point;
} else {
pixels[y * width + x] = color_back;
}
}
}
Bitmap bitmap = Bitmap.createBitmap(pixels, 0, width, width, height, Bitmap.Config.RGB_565);
if(logoBitmap != null){
return addLogo(bitmap, logoBitmap, logoPercent);
}
return bitmap;
} catch (WriterException e) {
e.printStackTrace();
return null;
}
}
private static Bitmap addLogo(Bitmap srcBitmap, Bitmap logoBitmap, float logoPercent){
if(srcBitmap == null){
return null;
}
if(logoBitmap == null){
return srcBitmap;
}
if(logoPercent < 0F || logoPercent > 1F){
logoPercent = 0.2F;
}
int srcWidth = srcBitmap.getWidth();
int srcHeight = srcBitmap.getHeight();
int logoWidth = logoBitmap.getWidth();
int logoHeight = logoBitmap.getHeight();
Bitmap bitmap = Bitmap.createBitmap(srcWidth, srcHeight, Bitmap.Config.ARGB_8888);
float scaleWidth = srcWidth * logoPercent / logoWidth;
float scaleHeight = srcHeight * logoPercent / logoHeight;
Canvas canvas = new Canvas(bitmap);
canvas.drawBitmap(srcBitmap, 0, 0, null);
canvas.scale(scaleWidth, scaleHeight, srcWidth/2, srcHeight/2);
canvas.drawBitmap(logoBitmap, srcWidth/2 - logoWidth/2, srcHeight/2 - logoHeight/2, null);
return bitmap;
}
}
================================================================
打开真机调试
-
因为在虚拟机中的摄像头显示是一片黑暗,所以这里使用真机调试
-
首先在将手机和电脑用USB线连接,USB连接方式选择传输文件
- 然后在手机中打开开发者选项,启动USB调试。开发者选项一般是隐藏的,可在关于手机处多次点击版本号,即可出现。
- 即可在Android Studio调用
集成工作
- 在build.gradle中添加依赖com.google.zxing:core:3.4.0,此步在前面已添加,因此可不用再次操作
implementation ‘com.google.zxing:core:3.4.0’
- 打开ZXing的开源地址项目https://github.com/zxing/zxing,在该项目下方的readme中看到,ZXing已将扫描仪的一些功能封装整合到一个包中
-
我们可以下载其中的Android调用,但是,ZXing在封装了多个功能,而我们只需要做到一个简单扫码的功能,因此此处选择网上大神们从其中挑出并修改整合好的扫码的一些基本功能
-
来到下载文件处,获取zxing内容,复制进自己的项目
- 获得音频文件,直接将raw放入自己项目中的res文件下(也可自己去别处下载)
- 拷贝attrs.xml、colors.xml以及ids.xml三个文件
- 获得图片
- 拷贝导航栏布局toolbar_scanner.xml以及扫码界面的布局activity_scanner.xml
- 拷贝util文件中的BitmapUtil.java、Constant.java、UriUtil.java
- 此时运行程序,此处因为引用的是原来作者的包,必定出错,等下再进行修改
实现扫码
- 在AndroidManifest.xml文件添加权限
…
- 新建一个名为ScanQRcode的activity,同时生成布局文件scan_qrcode.xml,修改布局文件,主要有一个按钮用于监听、一个文本框显示扫码返回结果
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android=“http://schemas.android.com/apk/res/android”
xmlns:app=“http://schemas.android.com/apk/res-auto”
xmlns:tools=“http://schemas.android.com/tools”
android:layout_width=“match_parent”
android:layout_height=“match_parent”
tools:context=“.ScanQRcode”>
<Button
android:id=“@+id/button”
android:layout_width=“wrap_content”
android:layout_height=“wrap_content”
android:text=“@string/scan_button”
app:layout_constraintEnd_toEndOf=“parent”
app:layout_constraintStart_toStartOf=“parent”
app:layout_constraintTop_toBottomOf=“@+id/include” />
<TextView
android:id=“@+id/textView”
android:layout_width=“match_parent”
android:layout_height=“wrap_content”
android:text=“@string/scan_result”
android:gravity=“center”
android:layout_gravity=“center_horizontal”
app:layout_constraintEnd_toEndOf=“parent”
app:layout_constraintStart_toStartOf=“parent”
app:layout_constraintTop_toBottomOf=“@+id/button” />
</androidx.constraintlayout.widget.ConstraintLayout>
对应string
扫描二维码
扫码结果
- 回到AndroidManifest.xml对新建内容注册(有的会自动注册,这里主要是手动注册CaptureActivity),因为在该项目中自定义了导航栏,因此此处还有修改主题为
NoActionBar
<application
…
<activity
android:name=“com.google.zxing.activity.CaptureActivity”
android:theme=“@style/Theme.AppCompat.NoActionBar”>
…
- 再回到主界面MainActivity文件,给第二个按钮添加监听操作
button2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Toast.makeText(getApplicationContext(),“扫描二维码”,Toast.LENGTH_LONG).show();
Intent intent = new Intent(MainActivity.this, ScanQRcode.class);
startActivity(intent);
}
});
- 启动扫码程序,在ScanQRcode文件中新建一个
startScanQRcode
来获得权限并开始扫码,在onCreate
中监听调用
private void startScanQRcode() {
// 申请相机权限
if (ActivityCompat.checkSelfPermission(this, Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {
// 申请权限
ActivityCompat.requestPermissions(ScanQRcode.this, new String[]{Manifest.permission.CAMERA}, Constant.REQ_PERM_CAMERA);
return;
}
// 申请文件读写权限(部分朋友遇到相册选图需要读写权限的情况,这里一并写一下)
if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
// 申请权限
ActivityCompat.requestPermissions(ScanQRcode.this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, Constant.REQ_PERM_EXTERNAL_STORAGE);
return;
}
// 二维码扫码
Intent intent = new Intent(ScanQRcode.this, CaptureActivity.class);
startActivityForResult(intent, Constant.REQ_QR_CODE);
}
- 重写
onRequestPermissionsResult
方法,修改获得权限
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
switch (requestCode) {
case Constant.REQ_PERM_CAMERA:
// 摄像头权限申请
if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
// 获得授权
startScanQRcode();
} else {
// 被禁止授权
Toast.makeText(ScanQRcode.this, “请至权限中心打开本应用的相机访问权限”, Toast.LENGTH_LONG).show();
}
break;
case Constant.REQ_PERM_EXTERNAL_STORAGE:
// 文件读写权限申请
if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
// 获得授权
startScanQRcode();
} else {
// 被禁止授权
Toast.makeText(ScanQRcode.this, “请至权限中心打开本应用的文件读写权限”, Toast.LENGTH_LONG).show();
}
break;
}
}
- 修改错误导入包,下图为原先错误的导入,此时将错误的导入删除即可(Android Studio会自动添加正确的包,若是未添加,也可手动修改),对出现类似的错误导入进行修改
修改前
修改后
- 进入扫码界面(虚拟机)
返回二维码内容
在ScanQRcode文件新建一个onActivityResult
,重写该方法,回调扫描结果
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == Constant.REQ_QR_CODE && resultCode == RESULT_OK) {
Bundle bundle = data.getExtras();
String scanResult = bundle.getString(Constant.INTENT_EXTRA_KEY_QR_SCAN);
scan_result.setText(scanResult);
}
}
打开闪光灯
在CaptureActivity文件中新建一个flashListener
监听对象,并将其注入闪光灯按钮中,监听点击
private View.OnClickListener flashListener = new View.OnClickListener() {
@Override
public void onClick(View view) {
try {
boolean isSuccess = CameraManager.get().setFlashLight(!isFlashOn);
if(!isSuccess){
Toast.makeText(CaptureActivity.this, “暂时无法开启闪光灯”, Toast.LENGTH_SHORT).show();
return;
}
if (isFlashOn) {
// 关闭闪光灯
btnFlash.setImageResource(R.drawable.flash_off);
isFlashOn = false;
} else {
// 开启闪光灯
btnFlash.setImageResource(R.drawable.flash_on);
isFlashOn = true;
}
}catch (Exception e){
e.printStackTrace();
}
}
};
注入ImageButton
btnFlash = (ImageButton) findViewById(R.id.btn_flash);
btnFlash.setOnClickListener(flashListener);
ScanQRcode完整代码
package com.example.qrcodetest;
import android.Manifest;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
import android.widget.Toast;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import com.example.qrcodetest.util.Constant;
import com.google.zxing.activity.CaptureActivity;
public class ScanQRcode extends AppCompatActivity{
Button scan_button; // 扫码
TextView scan_result; // 结果
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.scan_qrcode);
scan_button = (Button) findViewById(R.id.scan_button);
scan_result = (TextView) findViewById(R.id.scan_result);
scan_button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Toast.makeText(getApplicationContext(),“进入扫码界面”,Toast.LENGTH_LONG).show();
startScanQRcode();
}
});
}
//开始扫码
private void startScanQRcode() {
// 申请相机权限
if (ActivityCompat.checkSelfPermission(this, Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {
ActivityCompat.requestPermissions(ScanQRcode.this, new String[]{Manifest.permission.CAMERA}, Constant.REQ_PERM_CAMERA);
return;
}
// 申请文件读写权限
if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
ActivityCompat.requestPermissions(ScanQRcode.this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, Constant.REQ_PERM_EXTERNAL_STORAGE);
return;
}
// 二维码扫码
Intent intent = new Intent(ScanQRcode.this, CaptureActivity.class);
startActivityForResult(intent, Constant.REQ_QR_CODE);
}
尾声
如果你想成为一个优秀的 Android 开发人员,请集中精力,对基础和重要的事情做深度研究。
对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。 整理的这些架构技术希望对Android开发的朋友们有所参考以及少走弯路,本文的重点是你有没有收获与成长,其余的都不重要,希望读者们能谨记这一点。
这里,笔者分享一份从架构哲学的层面来剖析的视频及资料给大家梳理了多年的架构经验,筹备近6个月最新录制的,相信这份视频能给你带来不一样的启发、收获。
Android进阶学习资料库
一共十个专题,包括了Android进阶所有学习资料,Android进阶视频,Flutter,java基础,kotlin,NDK模块,计算机网络,数据结构与算法,微信小程序,面试题解析,framework源码!
大厂面试真题
PS:之前因为秋招收集的二十套一二线互联网公司Android面试真题 (含BAT、小米、华为、美团、滴滴)和我自己整理Android复习笔记(包含Android基础知识点、Android扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)
《2019-2021字节跳动Android面试历年真题解析》
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
Intent intent = new Intent(ScanQRcode.this, CaptureActivity.class);
startActivityForResult(intent, Constant.REQ_QR_CODE);
}
尾声
如果你想成为一个优秀的 Android 开发人员,请集中精力,对基础和重要的事情做深度研究。
对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。 整理的这些架构技术希望对Android开发的朋友们有所参考以及少走弯路,本文的重点是你有没有收获与成长,其余的都不重要,希望读者们能谨记这一点。
这里,笔者分享一份从架构哲学的层面来剖析的视频及资料给大家梳理了多年的架构经验,筹备近6个月最新录制的,相信这份视频能给你带来不一样的启发、收获。
[外链图片转存中…(img-GIwl6eXS-1715822567760)]
Android进阶学习资料库
一共十个专题,包括了Android进阶所有学习资料,Android进阶视频,Flutter,java基础,kotlin,NDK模块,计算机网络,数据结构与算法,微信小程序,面试题解析,framework源码!
[外链图片转存中…(img-Bp9IjaqN-1715822567762)]
大厂面试真题
PS:之前因为秋招收集的二十套一二线互联网公司Android面试真题 (含BAT、小米、华为、美团、滴滴)和我自己整理Android复习笔记(包含Android基础知识点、Android扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)
[外链图片转存中…(img-PYFY2zQC-1715822567763)]
《2019-2021字节跳动Android面试历年真题解析》
[外链图片转存中…(img-8RhbpowS-1715822567763)]
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!