草庐IT

《Android Studio开发实战 从零基础到App上线(第3版)》资源下载和内容勘误

aqi00 2023-09-01 原文

资源下载

下面是《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下载课后练习题答案,包括课后练习题(除动手练习)参考答案动手练习参考答案

参考资料

1、学习本书需要具备Java基础,如果您没学过Java的话,可学习以下系列的Java教程《Java开发笔记》,或阅读笔者的Java专著《好好学Java:从零基础到项目实战》。
2、由于篇幅所限,本书只覆盖了较为常见的Android开发技术,其余的Android开发技术可参考以下的Android笔记《Android开发笔记》
3、本书的技术实现采用的是Java编码,若您想进一步了解App开发中的Kotlin编程技术,可阅读以下系列的Kotlin教程《Kotlin入门教程》
4、更多的App开发进阶内容可参考这本书《Android App开发进阶与项目实战》。

文字勘误

下面对书中的笔误之处进行更正说明:

一、第一批勘误记录(以下的勘误记录在2023年1月的第二次印刷时均已修正)

1、第292页的“11.2.3  跟踪滑动轨迹实现手写签名”
代码注释中的“// 设置画笔的类型,STROK表示空心”,在STROK后面加个E,也就是改为“// 设置画笔的类型,STROKE表示空心”。

2、第302页的“11.4.1  上下滚动与左右滑动的冲突处理”
代码里面的注释文字“// 水平方向的滚动”描述不够完整,改为“// 横轴方向的偏移大等于纵轴方向的偏移,则判定为水平方向的滚动”。

3、第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

4、附录C
该小节第一段第五行的“开发入门代”指 改为“开发入门”代指

5、附录D
GNSS词条后面的中文说明末尾补充“(俄罗斯)”,也就是改为“全球卫星导航系统(俄罗斯)”。

二、第二批勘误记录(以下的勘误记录在2023年3月的第三次印刷时均已修正)

1、第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);
}

2、第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、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

2、chapter18模块的chapter18/src/main/AndroidManifest.xml,在manifest节点下面增加补充下面几行:
    <queries>
        <intent>
            <action android:name="android.intent.action.TTS_SERVICE" />
        </intent>
    </queries>

这是因为从Android11开始,文本转语音功能需要添加额外的服务声明。
完整代码见下
https://gitee.com/aqi00/android3/blob/main/chapter18/src/main/AndroidManifest.xml

3、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

4、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

5、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

6、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

7、服务端源码包的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
 

有关《Android Studio开发实战 从零基础到App上线(第3版)》资源下载和内容勘误的更多相关文章

  1. ruby - 将数组的内容转换为 int - 2

    我需要读入一个包含数字列表的文件。此代码读取文件并将其放入二维数组中。现在我需要获取数组中所有数字的平均值,但我需要将数组的内容更改为int。有什么想法可以将to_i方法放在哪里吗?ClassTerraindefinitializefile_name@input=IO.readlines(file_name)#readinfile@size=@input[0].to_i@land=[@size]x=1whilex 最佳答案 只需将数组映射为整数:@land边注如果你想得到一条线的平均值,你可以这样做:values=@input[x]

  2. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

    我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

  3. ruby - 如何使用 Ruby aws/s3 Gem 生成安全 URL 以从 s3 下载文件 - 2

    我正在编写一个小脚本来定位aws存储桶中的特定文件,并创建一个临时验证的url以发送给同事。(理想情况下,这将创建类似于在控制台上右键单击存储桶中的文件并复制链接地址的结果)。我研究过回形针,它似乎不符合这个标准,但我可能只是不知道它的全部功能。我尝试了以下方法:defauthenticated_url(file_name,bucket)AWS::S3::S3Object.url_for(file_name,bucket,:secure=>true,:expires=>20*60)end产生这种类型的结果:...-1.amazonaws.com/file_path/file.zip.A

  4. ruby-on-rails - 如何在我的 Rails 应用程序 View 中打印 ruby​​ 变量的内容? - 2

    我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby​​中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R

  5. Ruby Sinatra 配置用于生产和开发 - 2

    我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm

  6. ruby-on-rails - 每次我尝试部署时,我都会得到 - (gcloud.preview.app.deploy) 错误响应 : [4] DEADLINE_EXCEEDED - 2

    我是Google云的新手,我正在尝试对其进行首次部署。我的第一个部署是RubyonRails项目。我基本上是在关注thisguideinthegoogleclouddocumentation.唯一的区别是我使用的是我自己的项目,而不是他们提供的“helloworld”项目。这是我的app.yaml文件runtime:customvm:trueentrypoint:bundleexecrackup-p8080-Eproductionconfig.ruresources:cpu:0.5memory_gb:1.3disk_size_gb:10当我转到我的项目目录并运行gcloudprevie

  7. ruby - 安装 Ruby 时遇到问题(无法下载资源 "readline--patch") - 2

    当我尝试安装Ruby时遇到此错误。我试过查看this和this但无济于事➜~brewinstallrubyWarning:YouareusingOSX10.12.Wedonotprovidesupportforthispre-releaseversion.Youmayencounterbuildfailuresorotherbreakages.Pleasecreatepull-requestsinsteadoffilingissues.==>Installingdependenciesforruby:readline,libyaml,makedepend==>Installingrub

  8. ruby - 是否可以覆盖 gemfile 进行本地开发? - 2

    我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI

  9. ruby - 查找字符串中的内容类型(数字、日期、时间、字符串等) - 2

    我正在尝试解析一个CSV文件并使用SQL命令自动为其创建一个表。CSV中的第一行给出了列标题。但我需要推断每个列的类型。Ruby中是否有任何函数可以找到每个字段中内容的类型。例如,CSV行:"12012","Test","1233.22","12:21:22","10/10/2009"应该产生像这样的类型['integer','string','float','time','date']谢谢! 最佳答案 require'time'defto_something(str)if(num=Integer(str)rescueFloat(s

  10. ruby-on-rails - 如何重命名或移动 Rails 的 README_FOR_APP - 2

    当我在我的Rails应用程序根目录中运行rakedoc:app时,API文档是使用/doc/README_FOR_APP作为主页生成的。我想向该文件添加.rdoc扩展名,以便它在GitHub上正确呈现。更好的是,我想将它移动到应用程序根目录(/README.rdoc)。有没有办法通过修改包含的rake/rdoctask任务在我的Rakefile中执行此操作?是否有某个地方可以查找可以修改的主页文件的名称?还是我必须编写一个新的Rake任务?额外的问题:Rails应用程序的两个单独文件/README和/doc/README_FOR_APP背后的逻辑是什么?为什么不只有一个?

随机推荐