下面是《Android Studio开发实战 从零基础到App上线(第3版)》一书用到的工具和代码资源:
1、本书使用的Android Studio版本为Android Studio Dolphin(小海豚版本),最新的安装包可前往Android官网页面下载。
2、本书使用的Android NDK版本为r23b,最新的安装包可前往Android官网页面下载。
3、本书提供所有示例源码的demo工程下载,源码(适配Android5.0到Android12)的下载方式见该书前言末尾的二维码,获取ppt课件同样扫描前言末尾的二维码。最新的源码也可访问我的gitee获取,gitee地址是android3: 《Android Studio开发实战:从零基础到App上线(第3版)》配套源码,服务端的gitee地址是https://gitee.com/aqi00/net_server。
4、本书第10章使用了一些反编译和重签名工具,这些工具的下载页面是百度网盘 请输入提取码 (提取码93i5)
5、书本前言末尾的提供下载源码、导图和PPT,另外可在我的gitee下载课后练习题答案,包括课后练习题(除动手练习)参考答案和动手练习参考答案。
学习本书需要具备Java基础,如果您没学过Java的话,可学习以下系列的Java教程《Java开发笔记》,或阅读笔者的Java专著《好好学Java:从零基础到项目实战》。
2、由于篇幅所限,本书只覆盖了较为常见的Android开发技术,其余的Android开发技术可参考以下的Android笔记《Android开发笔记》。
3、本书的技术实现采用的是Java编码,若您想进一步了解App开发中的Kotlin编程技术,可阅读以下系列的Kotlin教程《Kotlin入门教程》。
4、更多的App开发进阶内容可参考这本书《Android App开发进阶与项目实战》。
下面对书中的笔误之处进行更正说明:
一、第一批勘误记录(以下的勘误记录在2023年1月的第二次印刷时均已修正)
第292页的“11.2.3 跟踪滑动轨迹实现手写签名”
代码注释中的“// 设置画笔的类型,STROK表示空心”,在STROK后面加个E,也就是改为“// 设置画笔的类型,STROKE表示空心”。
第302页的“11.4.1 上下滚动与左右滑动的冲突处理”
代码里面的注释文字“// 水平方向的滚动”描述不够完整,改为“// 横轴方向的偏移大等于纵轴方向的偏移,则判定为水平方向的滚动”。
第20章的“20.1.3 搭建穿透服务器”
该小节中间的“结合cygwin与coturn的安装配置步骤说明”,步骤04的“然后打开turnserver.conf,补充以下几行服务器的参数配置”后面要补充红字部分的listening-ip,完整的参数配置举例如下:
#监听端口
listening-port=3478
#内网IP(可通过ipconfig /all查看)
listening-ip=192.168.1.5
#外网IP
external-ip=120.36.33.151
#用户名和密码
user=admin:123456
#域名
realm=stun.xxx.cn
附录C
该小节第一段第五行的“开发入门代”指 改为“开发入门”代指。
附录D
GNSS词条后面的中文说明末尾补充“(俄罗斯)”,也就是改为“全球卫星导航系统(俄罗斯)”。
二、第二批勘误记录(以下的勘误记录在2023年3月的第三次印刷时均已修正)
第125页下到第126页上的“6.1.3 更安全的数据仓库”
往仓库中保存数据和获取数据的代码例子要调换位置。也就是把这两段代码互相换成下面这样:
前面把数据仓库的初始化以及读写操作封装在DatastoreUtil中,接下来通过该工具类即可方便地访问数据仓库了。往数据仓库保存数据的代码示例如下:
(完整代码见chapter06\src\main\java\com\example\chapter06\DatastoreWriteActivity.java)
// 获取数据仓库工具的实例
DatastoreUtil datastore = DatastoreUtil.getInstance(this);
datastore.setStringValue("name", name); // 添加一个名叫name的字符串
// 添加一个名叫age的整数
datastore.setIntValue("age", Integer.parseInt(age));
// 添加一个名叫height的整数
datastore.setIntValue("height", Integer.parseInt(height));
// 添加一个名叫weight的双精度数
datastore.setDoubleValue("weight", Double.parseDouble(weight));
// 添加一个名叫married的布尔值
datastore.setBooleanValue("married", isMarried);
datastore.setStringValue("update_time",
DateUtil.getNowDateTime("yyyy-MM-dd HH:mm:ss"));
从数据仓库获取数据的代码示例如下:
(完整代码见chapter06\src\main\java\com\example\chapter06\DatastoreReadActivity.java)
// 从数据仓库中读取信息
private void readDatastore() {
// 获取数据仓库工具的实例
DatastoreUtil datastore = DatastoreUtil.getInstance(this);
String desc = "数据仓库中保存的信息如下:";
desc = String.format("%s\n %s为%s", desc, "姓名",
datastore.getStringValue("name"));
desc = String.format("%s\n %s为%d", desc, "年龄",
datastore.getIntValue("age"));
desc = String.format("%s\n %s为%d", desc, "身高",
datastore.getIntValue("height"));
desc = String.format("%s\n %s为%.2f", desc, "体重",
datastore.getDoubleValue("weight"));
desc = String.format("%s\n %s为%b", desc, "婚否",
datastore.getBooleanValue("married"));
desc = String.format("%s\n %s为%s", desc, "更新时间",
datastore.getStringValue("update_time"));
tv_data.setText(desc);
}
第584页的“19.2.1 人脸检测”
把图19-18下面的两行依赖库配置
implementation 'com.huawei.hms:ml-computer-vision-faceverify:2.2.0.300'
implementation
'com.huawei.hms:ml-computer-vision-faceverify-model:2.2.0.300'
改为下面几行
implementation 'com.huawei.hms:ml-computer-vision-face:2.0.5.300'
implementation
'com.huawei.hms:ml-computer-vision-face-emotion-model:2.0.5.300'
implementation
'com.huawei.hms:ml-computer-vision-face-feature-model:2.0.5.300'
implementation
'com.huawei.hms:ml-computer-vision-face-shape-point-model:2.0.5.300'
3、第586页的“19.2.2 人脸比对”
该小节第二段之后补充下面的第三段红字:
引入人脸比对功能需要修改模块的build.gradle,往dependencies节点添加如下配置,表示导入指定版本的人脸比对库(公共的agconnect插件和库工程hmsml也要导入):
implementation 'com.huawei.hms:ml-computer-vision-faceverify:2.2.0.300'
implementation
'com.huawei.hms:ml-computer-vision-faceverify-model:2.2.0.300'
以比对两张人脸图片为例,详细的比对过程说明如下。
下面对随书源码的疏漏之处进行更正说明:
1、代码注释中的“// 设置画笔的类型,STROK表示空心”,在STROK后面加个E,也就是改为“// 设置画笔的类型,STROKE表示空心”。
2、chapter04模块里的chapter04\src\main\java\com\example\chapter04\AlarmActivity.java,把sendAlarm()方法里面的“FLAG_UPDATE_CURRENT”改为“PendingIntent.FLAG_IMMUTABLE | PendingIntent.FLAG_UPDATE_CURRENT”,避免在Android12以上机型运行闪退,因为从Android12开始,必须添加 FLAG_IMMUTABLE 或者 FLAG_MUTABLE。
完整代码见下
https://gitee.com/aqi00/android3/blob/main/chapter04/src/main/java/com/example/chapter04/AlarmActivity.java
chapter14模块里的chapter14\src\main\res\layout\item_video.xml
把com.google.android.exoplayer2.ui.PlayerView改为com.google.android.exoplayer2.ui.StyledPlayerView,避免实战项目的短视频页面闪退。
完整代码见下
https://gitee.com/aqi00/android3/blob/main/chapter14/src/main/res/layout/item_video.xml
4、chapter16模块里chapter16\src\main\java\com\example\chapter16\SatelliteSphereActivity.java的initLocation方法,为了避免Android11以上无法获取卫星信息而注释下面几行代码。
// if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
// // 实测发现部分手机的android11系统使用卫星定位会没返回
// bestProvider = LocationManager.NETWORK_PROVIDER;
// }
完整代码见下
https://gitee.com/aqi00/advanceapp/blob/main/location/src/main/java/com/example/location/SatelliteSphereActivity.java
chapter18模块的chapter18/src/main/AndroidManifest.xml,在manifest节点下面增加补充下面几行:
这是因为从Android11开始,文本转语音功能需要添加额外的服务声明。
完整代码见下
https://gitee.com/aqi00/android3/blob/main/chapter18/src/main/AndroidManifest.xml
chapter19模块的chapter19/build.gradle,在dependencies节点下面补充下面几行依赖库配置:
// 人脸检测和笑脸捕捉需要
implementation 'com.huawei.hms:ml-computer-vision-face:2.0.5.300'
implementation 'com.huawei.hms:ml-computer-vision-face-emotion-model:2.0.5.300'
implementation 'com.huawei.hms:ml-computer-vision-face-feature-model:2.0.5.300'
implementation 'com.huawei.hms:ml-computer-vision-face-shape-point-model:2.0.5.300'
// 人像抠图需要
implementation 'com.huawei.hms:ml-computer-vision-segmentation:2.2.0.300'
implementation 'com.huawei.hms:ml-computer-vision-image-segmentation-body-model:2.2.0.300'
完整代码见下
https://gitee.com/aqi00/android3/blob/main/chapter19/build.gradle
chapter19模块的chapter19/src/main/java/com/example/chapter19/FaceDetectActivity.java,在saveFace方法内部,把下面两行
String path = String.format("%s/%s.jpg",
getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS).toString(), DateUtil.getNowDateTime());
改为下面两行
String path = String.format("%s/%s.jpg",
"/storage/emulated/0/DCIM/Camera", DateUtil.getNowDateTime());
完整代码见下
https://gitee.com/aqi00/android3/blob/main/chapter19/src/main/java/com/example/chapter19/FaceDetectActivity.java
hmsml模块的hmsml/src/main/java/com/example/hmsml/image/util/ImageUtils.java,在saveToAlbum方法内部,把下面两行
File root = new File(Environment.getExternalStorageDirectory().getAbsoluteFile(), this.context.getPackageName());
File dir = new File(root, "image");
改成下面这行
File dir = new File("/storage/emulated/0/DCIM/Camera");
完整代码见下
https://gitee.com/aqi00/android3/blob/main/hmsml/src/main/java/com/example/hmsml/image/util/ImageUtils.java
hmsml模块的hmsml/src/main/java/com/example/hmsml/face/transactor/LocalFaceTransactor.java,在saveImage方法末尾增加下面一行
context.sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri.fromFile(new File(path))));
完整代码见下
https://gitee.com/aqi00/android3/blob/main/hmsml/src/main/java/com/example/hmsml/face/transactor/LocalFaceTransactor.java
服务端源码包的HttpServer/src/com/servlet/verifycode/GenerateCode.java
把下面四行
JPEGImageDecoder decoder = JPEGCodec.createJPEGDecoder(fis);
BufferedImage image = decoder.decodeAsBufferedImage();
JPEGImageEncoder encoder = JPEGCodec.createJPEGEncoder(os);
encoder.encode(image);
改为以下两行
BufferedImage buffer = ImageIO.read(fis);
ImageIO.write(buffer, "jpeg", os);
同时去掉代码开头的这几个import语句“import com.sun.image.codec.jpeg.***”。
这是因为原来的代码在部分jdk版本会提示编译失败,修改之后即可消除编译错误。
完整代码见下
https://gitee.com/aqi00/net_server/blob/master/HttpServer/src/com/servlet/verifycode/GenerateCode.java
chapter17模块的chapter17/src/main/java/com/example/chapter17/MainActivity.java,涉及到蓝牙操作的活动跳转,都要增加判断:如果当前系统版本大于等于Android 12,就要动态申请以下权限:BLUETOOTH_SCAN、BLUETOOTH_ADVERTISE、BLUETOOTH_CONNECT
完整代码见下
https://gitee.com/aqi00/android3/blob/main/chapter17/src/main/java/com/example/chapter17/MainActivity.java
chapter19模块的chapter19/src/main/AndroidManifest.xml,增加READ_PHONE_STATE的权限声明,也就是补充下列配置
同时chapter19/src/main/java/com/example/chapter19/MainActivity.java在跳转到扫码页面HmsScanActivity之前,要增加判断:如果当前系统版本大于等于Android 12,就要动态申请READ_PHONE_STATE权限。
完整代码见下
https://gitee.com/aqi00/android3/blob/main/chapter19/src/main/AndroidManifest.xml
https://gitee.com/aqi00/android3/blob/main/chapter19/src/main/java/com/example/chapter19/MainActivity.java