2024年Android最新安卓APP实现在线更新功能,适配Androidx和9(2),java面试

尾声

面试成功其实都是必然发生的事情,因为在此之前我做足了充分的准备工作,不单单是纯粹的刷题,更多的还会去刷一些Android核心架构进阶知识点,比如:JVM、高并发、多线程、缓存、热修复设计、插件化框架解读、组件化框架设计、图片加载框架、网络、设计模式、设计思想与代码质量优化、程序性能优化、开发效率优化、设计模式、负载均衡、算法、数据结构、高级UI晋升、Framework内核解析、Android组件内核等。

不仅有学习文档,视频+笔记提高学习效率,还能稳固你的知识,形成良好的系统的知识体系。这里,笔者分享一份从架构哲学的层面来剖析的视频及资料分享给大家梳理了多年的架构经验,筹备近6个月最新录制的,相信这份视频能给你带来不一样的启发、收获。

Android进阶学习资料库

一共十个专题,包括了Android进阶所有学习资料,Android进阶视频,Flutter,java基础,kotlin,NDK模块,计算机网络,数据结构与算法,微信小程序,面试题解析,framework源码!

image

大厂面试真题

PS:之前因为秋招收集的二十套一二线互联网公司Android面试真题 (含BAT、小米、华为、美团、滴滴)和我自己整理Android复习笔记(包含Android基础知识点、Android扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)

《2017-2021字节跳动Android面试历年真题解析》

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

工具类三个:

LoadingService.java

import android.app.IntentService;

import android.app.Notification;

import android.app.NotificationChannel;

import android.app.NotificationManager;

import android.content.Context;

import android.content.Intent;

import android.content.SharedPreferences;

import android.net.Uri;

import android.os.Build;

import android.widget.RemoteViews;

import android.widget.Toast;

import androidx.annotation.Nullable;

import androidx.core.app.NotificationCompat;

import androidx.core.content.FileProvider;

import com.example.administrator.loginclient.R;

import com.lidroid.xutils.HttpUtils;

import com.lidroid.xutils.exception.HttpException;

import com.lidroid.xutils.http.ResponseInfo;

import com.lidroid.xutils.http.callback.RequestCallBack;

import java.io.File;

public class LoadingService extends IntentService {

private HttpUtils httpUtils;

NotificationManager nm;

private String url,path;

private SharedPreferences sharedPreferences;

public LoadingService(String name) {

super(name);

}

public LoadingService() {

super(“MyService”);

}

public static void startUploadImg(Context context)

{

Intent intent = new Intent(context, LoadingService.class);

context.startService(intent);

}

public void onCreate() {

super.onCreate();

httpUtils = new HttpUtils();

httpUtils.configCurrentHttpCacheExpiry(0);

nm = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);

sharedPreferences = getSharedPreferences(“data”,MODE_PRIVATE);

}

@Override

protected void onHandleIntent(@Nullable Intent intent) {

updateApk();

}

private void updateApk(){

url = sharedPreferences.getString(“url”,“”);

path = sharedPreferences.getString(“path”,“”);

httpUtils.download(url,

path , new RequestCallBack() {

@Override

public void onLoading(final long total, final long current,

boolean isUploading) {

createNotification(total,current);

sendBroadcast(new Intent().setAction(“android.intent.action.loading”));//发送正在下载的广播

super.onLoading(total, current, isUploading);

}

@Override

public void onSuccess(ResponseInfo arg0) {

nm.cancel(R.layout.notification_item);

Toast.makeText(LoadingService.this,“下载成功…”,Toast.LENGTH_SHORT).show();

installApk();//下载成功 打开安装界面

stopSelf();//结束服务

sendBroadcast(new Intent().setAction(“android.intent.action.loading_over”));//发送下载结束的广播

}

@Override

public void onFailure(HttpException arg0, String arg1) {

Toast.makeText(LoadingService.this,“下载失败…”,Toast.LENGTH_SHORT).show();

sendBroadcast(new Intent().setAction(“android.intent.action.loading_over”));//发送下载结束的广播

nm.cancel(R.layout.notification_item);

stopSelf();

}

});

}

/**

  • 安装下载的新版本

*/

protected void installApk() {

Intent intent = new Intent(Intent.ACTION_VIEW);

intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

File file = new File(path);

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {

// 7.0 以上

try {

Uri apkUri = FileProvider.getUriForFile(this, “com.jw.fileprovider”, file);

intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

intent.setDataAndType(apkUri, “application/vnd.android.package-archive”);

}catch (Exception e){

}

} else {

// 7.0以下

Uri uri = Uri.fromFile(file);

intent.setDataAndType(uri, “application/vnd.android.package-archive”);

}

this.startActivity(intent);

}

private void createNotification(final long total, final long current){

NotificationCompat.Builder builder = new NotificationCompat.Builder(this,“default”);

builder.setSmallIcon(R.mipmap.ic_launcher)//必须要设置这个属性,否则不显示

;

RemoteViews contentView = new RemoteViews(this.getPackageName(),R.layout.notification_item);

contentView.setProgressBar(R.id.progress, (int)total, (int)current, false);

builder.setOngoing(true);//设置左右滑动不能删除

// Notification notification = builder.build();

// notification.contentView = contentView;

String id = “channel_001”;

String name = “name”;

Notification notification = null;

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {

NotificationChannel mChannel = new NotificationChannel(id, name, NotificationManager.IMPORTANCE_LOW);

nm.createNotificationChannel(mChannel);

notification = new Notification.Builder(LoadingService.this,“default”)

.setChannelId(id)

.setContentTitle(“客户端更新”)

.setContentText(“更新完毕”)

.setSmallIcon(R.mipmap.ic_launcher).build();

notification.contentView = contentView;

} else {

NotificationCompat.Builder notificationBuilder = new NotificationCompat.Builder(LoadingService.this,“default”)

.setContentTitle(“更新了”)

.setContentText(“更新了xxxxxxxxx内容”)

.setSmallIcon(R.mipmap.ic_launcher)

.setOngoing(true)

;//无效

notification = notificationBuilder.build();

notification.contentView = contentView;

}

nm.notify(R.layout.notification_item,notification);//发送通知

}

}

Utils.java

import android.content.Context;

import android.content.pm.PackageInfo;

import android.content.pm.PackageManager;

public class Utils {

/**

  • 获取版本号

  • @throws PackageManager.NameNotFoundException

*/

public static String getVersionName(Context context) throws PackageManager.NameNotFoundException {

// 获取packagemanager的实例

PackageManager packageManager = context.getPackageManager();

// getPackageName()是你当前类的包名,0代表是获取版本信息

PackageInfo packInfo = packageManager.getPackageInfo(context.getPackageName(), 0);

String version = packInfo.versionName;

return version;

}

/**

  • 版本号比较

*0代表相等,1代表version1大于version2,-1代表version1小于version2

  • @param version1

  • @param version2

  • @return

*/

public static int compareVersion(String version1, String version2) {

if (version1.equals(version2)) {

return 0;

}

String[] version1Array = version1.split(“\.”);

String[] version2Array = version2.split(“\.”);

int index = 0;

// 获取最小长度值

int minLen = Math.min(version1Array.length, version2Array.length);

int diff = 0;

// 循环判断每位的大小

while (index < minLen

&& (diff = Integer.parseInt(version1Array[index])

  • Integer.parseInt(version2Array[index])) == 0) {

index++;

}

if (diff == 0) {

// 如果位数不一致,比较多余位数

for (int i = index; i < version1Array.length; i++) {

if (Integer.parseInt(version1Array[i]) > 0) {

return 1;

}

}

for (int i = index; i < version2Array.length; i++) {

if (Integer.parseInt(version2Array[i]) > 0) {

return -1;

}

}

return 0;

} else {

return diff > 0 ? 1 : -1;

}

}

}

VersionInfoBean.java

public class VersionInfoBean {

private String versionName;

private String downloadUrl;

private String desc;

private String path;

public VersionInfoBean(String versionName, String downloadUrl, String desc, String path) {

this.versionName = versionName;

this.downloadUrl = downloadUrl;

this.desc = desc;

this.path = path;

}

public String getPath() {

return path;

}

public void setPath(String path) {

this.path = path;

}

public String getVersionName() {

return versionName;

}

public void setVersionName(String versionName) {

this.versionName = versionName;

}

public String getDownloadUrl() {

return downloadUrl;

}

public void setDownloadUrl(String downloadUrl) {

this.downloadUrl = downloadUrl;

}

public String getDesc() {

return desc;

}

public void setDesc(String desc) {

this.desc = desc;

}

public VersionInfoBean(String versionName, String downloadUrl, String desc) {

this.versionName = versionName;

this.downloadUrl = downloadUrl;

this.desc = desc;

}

}

layout下的xml

activity_update.xml

<?xml version="1.0" encoding="utf-8"?>

<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=“.Activities.UpdateActivity”>

<Button

android:id=“@+id/btn_refresh”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:text=“版本更新”

app:layout_constraintBottom_toBottomOf=“parent”

app:layout_constraintLeft_toLeftOf=“parent”

app:layout_constraintRight_toRightOf=“parent”

app:layout_constraintTop_toTopOf=“parent” />

</androidx.constraintlayout.widget.ConstraintLayout>

version_update.xml

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:orientation=“vertical”>

<TextView

android:id=“@+id/title”

android:layout_width=“match_parent”

android:layout_height=“50dp”

android:gravity=“center_vertical|center_horizontal”

android:paddingLeft=“10dp”

android:text=“更新提示”

android:textColor=“#000000”

android:textSize=“18sp” />

<TextView

android:layout_width=“match_parent”

android:layout_height=“1dp”

android:background=“#000000” />

<TextView

android:layout_marginTop=“10dp”

android:id=“@+id/version”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:paddingLeft=“12dp”

android:paddingRight=“10dp”

android:text=“content”

android:textSize=“16sp” />

<TextView

android:layout_marginTop=“14dp”

android:id=“@+id/content”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:paddingLeft=“12dp”

android:paddingRight=“10dp”

android:text=“content”

android:textSize=“14sp” />

<TextView

android:layout_marginTop=“10dp”

android:layout_width=“match_parent”

android:layout_height=“1dp”

android:background=“#f0f0f0” />

<LinearLayout

android:layout_marginTop=“4dp”

android:layout_width=“match_parent”

android:layout_height=“45dp”

android:orientation=“horizontal”

android:paddingLeft=“10dp”

android:paddingRight=“10dp”>

<Button

android:textColor=“#000000”

android:id=“@+id/left”

android:layout_width=“0dp”

android:layout_height=“match_parent”

android:layout_weight=“1”

android:background=“@null”

android:gravity=“center_vertical|center_horizontal”

android:text=“取消”

android:textSize=“16sp”

/>

<TextView

android:textColor=“#f0f0f0”

android:id=“@+id/line”

android:layout_width=“1dp”

android:layout_height=“match_parent”

android:layout_centerHorizontal=“true”

/>

<Button

android:textColor=“#d90c0c”

android:id=“@+id/right”

android:layout_width=“0dp”

android:layout_height=“match_parent”

android:layout_alignParentRight=“true”

android:layout_weight=“1”

android:background=“@null”

android:gravity=“center_vertical|center_horizontal”

android:text=“更新”

android:textSize=“16sp”

/>

notification_item.xml

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”

android:orientation=“horizontal” android:layout_width=“match_parent”

android:paddingTop=“8dp”

android:layout_height=“wrap_content”>

<ImageView

android:src=“@mipmap/ic_launcher”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content” />

<LinearLayout

android:layout_marginRight=“6dp”

android:layout_marginLeft=“8dp”

android:layout_gravity=“center_vertical”

android:orientation=“vertical”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”>

<TextView

android:textSize=“14sp”

android:text=“正在下载”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content” />

<ProgressBar

android:layout_marginTop=“4dp”

style=“?android:attr/progressBarStyleHorizontal”

android:id=“@+id/progress”

android:layout_width=“match_parent”

android:layout_height=“wrap_content” />

在res新建一个xml文件夹

在xml文件夹里新建资源文件update_apk_paths.xml

<?xml version="1.0" encoding="utf-8"?>

在xml文件夹里新建资源文件network_security_config.xml

<?xml version="1.0" encoding="utf-8"?>

添加的依赖:xUtils-2.6.14.jar 提取码:p5q3

implementation fileTree(include: [‘*.jar’], dir: ‘libs’)

implementation ‘androidx.core:core:1.3.0’

implementation ‘androidx.constraintlayout:constraintlayout:1.1.3’

testImplementation ‘junit:junit:4.12’

implementation ‘androidx.appcompat:appcompat:1.1.0’

implementation files(‘libs/xUtils-2.6.14.jar’)

androidTestImplementation(‘androidx.test.espresso:espresso-core:3.1.0’, {

exclude group: ‘com.android.support’, module: ‘support-annotations’

})

manifest清单

1.在<manifest > ...</manifest> 里面加上权限:

2.在 <application> ... </application>里面加上:

android:networkSecurityConfig=“@xml/network_security_config”

<uses-library

android:name=“org.apache.http.legacy”

android:required=“false”

/>

<activity android:name=“.Activities.UpdateActivity”

tools:ignore=“InnerclassSeparator”>

<service

android:name=“.Updates.LoadingService”

android:process=“system”

tools:ignore=“InnerclassSeparator” />

<provider

android:name=“androidx.core.content.FileProvider”

android:authorities=“com.jw.fileprovider”

android:exported=“false”

android:grantUriPermissions=“true”>

<meta-data

android:name=“android.support.FILE_PROVIDER_PATHS”

android:resource=“@xml/update_apk_paths” />

(2)Tomcat的配置:

【附】相关架构及资料

源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,和技术大牛一起讨论交流解决问题。

image

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

<activity android:name=“.Activities.UpdateActivity”

tools:ignore=“InnerclassSeparator”>

<service

android:name=“.Updates.LoadingService”

android:process=“system”

tools:ignore=“InnerclassSeparator” />

<provider

android:name=“androidx.core.content.FileProvider”

android:authorities=“com.jw.fileprovider”

android:exported=“false”

android:grantUriPermissions=“true”>

<meta-data

android:name=“android.support.FILE_PROVIDER_PATHS”

android:resource=“@xml/update_apk_paths” />

(2)Tomcat的配置:

【附】相关架构及资料

源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,和技术大牛一起讨论交流解决问题。

[外链图片转存中…(img-DoluZl96-1715675765450)]

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 5
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值