手写Service后台下载app——跳出DownloadManager系统7.0之坑

本文主要是介绍手写Service后台下载app——跳出DownloadManager系统7.0之坑,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

之前项目中有关app的现在和更新相关工具类一直用的是Android系统下载管理DownloadManager功能。如果随着Android系统的不断提升再加上Android开源性 手机厂家
对此作恶部分改动。导致一些系统自带的工具类出现异常情况。

华为P9

华为P9/P9 Plus上线,你的APP准备好了吗?

新机入手后,MTC率先从应用市场随机下载部分APP做基于P9/P9 Plus的兼容性测试,在Monkey脚本跑完之后,我们发现部分App会出现Crash、ANR的问题。在此跟开发小伙伴们分享下产生此类问题的原因,开发小伙伴们在App开发过程中多加注意,避免此类问题的产生。

华为P9使用该 华为P9在7.1的时候,利用系统DownloadManager工具类下载时就出现异常——在下载过程中竟突然自动消失!

怀疑这个应该是Android系统和华为P9不兼容导致的一个的bug。

大家都知道,老板不懂代码,有时候出现的问题还不属于你的问题。在其他手机上好使,部分手机就出现问题。 只要有一款手机出现一个问题就认为你的程序有问题。

这是最蛋疼的。so,问题来了,谁让你是安卓程序员呢

哈哈,抱怨是没有用的。有问题还是要解决的。

1.首先是要判断是6.0权限

 /** * 请求运行时权限 * eg: */  public void requestRuntimePermission(QuestPermissionListener questPermissionListener, String... permissions){  BasePermisitionActivity.requestRuntimePermission(permissions,questPermissionListener);  }  .......

6.0运行权限请参考: android6.0运行时权限完美封装

下载service工具类:


/*** 类功能描述:</br>* 自定义Service后台下载app——跳出DownloadManager系统下载之坑 </br>* 修改人:   yuyahao* @version 1.0 </p> 修改时间:</br> 修改备注:</br>*/
public class UpdateService extends Service {public static final String TAG =  "ServiceDownLoadApp";public static final String ACTION = "me.shenfan.UPDATE_APP";public static final String STATUS = "status";public static final String PROGRESS = "progress";public static boolean DEBUG = true;//下载大小通知频率public static final int UPDATE_NUMBER_SIZE = 1;public static final int DEFAULT_RES_ID = -1;public static final int UPDATE_PROGRESS_STATUS = 0;public static final int UPDATE_ERROR_STATUS = -1;public static final int UPDATE_SUCCESS_STATUS = 1;//paramsprivate static final String URL = "downloadUrl";private static final String ICO_RES_ID = "icoResId";private static final String ICO_SMALL_RES_ID = "icoSmallResId";private static final String UPDATE_PROGRESS = "updateProgress";private static final String STORE_DIR = "storeDir";private static final String DOWNLOAD_NOTIFICATION_FLAG = "downloadNotificationFlag";private static final String DOWNLOAD_SUCCESS_NOTIFICATION_FLAG = "downloadSuccessNotificationFlag";private static final String DOWNLOAD_ERROR_NOTIFICATION_FLAG = "downloadErrorNotificationFlag";private static final String IS_SEND_BROADCAST = "isSendBroadcast";private String downloadUrl;private int icoResId;             //default app icoprivate int icoSmallResId;private int updateProgress;   //update notification progress when it add numberprivate static  String storeDir;          //default sdcard/Android/package/updateprivate int downloadNotificationFlag;private int downloadSuccessNotificationFlag;private int downloadErrorNotificationFlag;private boolean isSendBroadcast;private UpdateProgressListener updateProgressListener;private LocalBinder localBinder = new LocalBinder();/*** Class used for the client Binder.*/public class LocalBinder extends Binder{/*** set update progress call back* @param listener*/public void setUpdateProgressListener(UpdateProgressListener listener){UpdateService.this.setUpdateProgressListener(listener);}}private boolean startDownload;//开始下载private int lastProgressNumber;private NotificationCompat.Builder builder;private NotificationManager manager;private int notifyId;private String appName;private LocalBroadcastManager localBroadcastManager;private Intent localIntent;private DownloadApk downloadApkTask;/*** whether debug*/public static void debug(){DEBUG = true;}/*** 点击通知栏去进行安装* @param path* @return*/private static Intent installIntent(String path){Uri uri = Uri.fromFile(new File(path));Intent installIntent = new Intent(Intent.ACTION_VIEW);installIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);installIntent.setDataAndType(uri, "application/vnd.android.package-archive");return installIntent;}/*** 通过浏览器进行下载* @param downloadUrl* @return*/private static Intent webLauncher(String downloadUrl){Uri download = Uri.parse(downloadUrl);Intent intent = new Intent(Intent.ACTION_VIEW, download);intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);return intent;}/*** 通过URL获取要下载的apk的前缀没弄过* @param downloadUrl* @return*/private static String getSaveFileName(String downloadUrl) {if (downloadUrl == null || TextUtils.isEmpty(downloadUrl)) {return "noName.apk";}return downloadUrl.substring(downloadUrl.lastIndexOf("/"));}/*** 来设置要下载apk储存的文件夹* @param service* @return*/private static File getDownloadDir(UpdateService service){File downloadDir = null;if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {if (service.storeDir != null){downloadDir = new File(Environment.getExternalStorageDirectory(), service.storeDir);}else {downloadDir = new File(service.getExternalCacheDir(), "update");}} else {downloadDir = new File(service.getCacheDir(), "update");}if (!downloadDir.exists()) {downloadDir.mkdirs();}return downloadDir;}@Overridepublic void onCreate() {super.onCreate();appName = getApplicationName();}@Overridepublic int onStartCommand(Intent intent, int flags, int startId) {if (!startDownload && intent != null){startDownload = true;downloadUrl = intent.getStringExtra(URL);icoResId = intent.getIntExtra(ICO_RES_ID, DEFAULT_RES_ID);icoSmallResId = intent.getIntExtra(ICO_SMALL_RES_ID, DEFAULT_RES_ID);storeDir = intent.getStringExtra(STORE_DIR);updateProgress = intent.getIntExtra(UPDATE_PROGRESS, UPDATE_NUMBER_SIZE);downloadNotificationFlag = intent.getIntExtra(DOWNLOAD_NOTIFICATION_FLAG, 0);downloadErrorNotificationFlag = intent.getIntExtra(DOWNLOAD_ERROR_NOTIFICATION_FLAG, 0);downloadSuccessNotificationFlag = intent.getIntExtra(DOWNLOAD_SUCCESS_NOTIFICATION_FLAG, 0);isSendBroadcast = intent.getBooleanExtra(IS_SEND_BROADCAST, false);if (DEBUG){LogUtil.e(TAG, "downloadUrl: " + downloadUrl);LogUtil.e(TAG, "icoResId: " + icoResId);LogUtil.e(TAG, "icoSmallResId: " + icoSmallResId);LogUtil.e(TAG, "storeDir: " + storeDir);LogUtil.e(TAG, "updateProgress: " + updateProgress);LogUtil.e(TAG, "downloadNotificationFlag: " + downloadNotificationFlag);LogUtil.e(TAG, "downloadErrorNotificationFlag: " + downloadErrorNotificationFlag);LogUtil.e(TAG, "downloadSuccessNotificationFlag: " + downloadSuccessNotificationFlag);LogUtil.e(TAG, "isSendBroadcast: " + isSendBroadcast);}notifyId = startId;buildNotification();buildBroadcast();downloadApkTask = new DownloadApk(this);downloadApkTask.execute(downloadUrl);}return super.onStartCommand(intent, flags, startId);}@Nullable@Overridepublic IBinder onBind(Intent intent) {return localBinder;}@Overridepublic boolean onUnbind(Intent intent) {return true;}public void setUpdateProgressListener(UpdateProgressListener updateProgressListener) {this.updateProgressListener = updateProgressListener;}@Overridepublic void onDestroy() {if (downloadApkTask != null){downloadApkTask.cancel(true);}if (updateProgressListener != null){updateProgressListener = null;}localIntent = null;builder = null;super.onDestroy();}/*** 获取当前的应用名* @return*/public String getApplicationName() {PackageManager packageManager = null;ApplicationInfo applicationInfo = null;try {packageManager = getApplicationContext().getPackageManager();applicationInfo = packageManager.getApplicationInfo(getPackageName(), 0);} catch (PackageManager.NameNotFoundException e) {applicationInfo = null;}String applicationName =(String) packageManager.getApplicationLabel(applicationInfo);return applicationName;}private void buildBroadcast(){if (!isSendBroadcast){return;}localBroadcastManager = LocalBroadcastManager.getInstance(this);localIntent = new Intent(ACTION);}/*** 发送广播* @param status* @param progress*/private void sendLocalBroadcast(int status, int progress){if (!isSendBroadcast || localIntent == null){return;}localIntent.putExtra(STATUS, status);localIntent.putExtra(PROGRESS, progress);localBroadcastManager.sendBroadcast(localIntent);}/*** 环形通知安*/private void buildNotification(){manager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);builder = new NotificationCompat.Builder(this);builder.setContentTitle(getString(R.string.update_app_model_prepare, appName)).setWhen(System.currentTimeMillis()).setProgress(100, 1, false).setSmallIcon(icoSmallResId).setLargeIcon(BitmapFactory.decodeResource(getResources(), icoResId)).setDefaults(downloadNotificationFlag);manager.notify(notifyId, builder.build());}/*** 开始 下载*/private void start(){builder.setContentTitle(appName);builder.setContentText(getString(R.string.update_app_model_prepare, 1));manager.notify(notifyId, builder.build());sendLocalBroadcast(UPDATE_PROGRESS_STATUS, 1);if (updateProgressListener != null){updateProgressListener.start();}}/**** 通知进度条,进度条* 大小范围(1~100)*/private void update(int progress){if (progress - lastProgressNumber > updateProgress){lastProgressNumber = progress;builder.setProgress(100, progress, false);builder.setContentText(getString(R.string.update_app_model_progress, progress, "%"));manager.notify(notifyId, builder.build());sendLocalBroadcast(UPDATE_PROGRESS_STATUS, progress);if (updateProgressListener != null){updateProgressListener.update(progress);}}}/*** 下载成功的回调* @param path*/private void success(String path) {builder.setProgress(0, 0, false);builder.setContentText(getString(R.string.update_app_model_success));GetToast.useString(this,"asdfasdf");manager.cancel(0);if(FileHelper.checkFileIsExists(path)){Intent i = installIntent(path);PendingIntent intent = PendingIntent.getActivity(this, 0, i, PendingIntent.FLAG_UPDATE_CURRENT);builder.setContentIntent(intent).setAutoCancel(true)//用户点击就自动消失.setDefaults(downloadSuccessNotificationFlag);Notification n = builder.build();n.contentIntent = intent;manager.notify(notifyId, n);if (updateProgressListener != null){updateProgressListener.success();}startActivity(i);IntentFilter filter = new IntentFilter();}else{DataCleanManager.deleteFilesByDirectory2(storeDir);}stopSelf();}/*** 清除本地文件*/public static void deleteFilesByDirectory(){DataCleanManager.deleteFilesByDirectory2(storeDir);}/*** 下载失败通知浏览器下载回调*/private void error(){Intent i = webLauncher(downloadUrl);PendingIntent intent = PendingIntent.getActivity(this, 0, i,PendingIntent.FLAG_UPDATE_CURRENT);builder.setContentText(getString(R.string.update_app_model_error));builder.setContentIntent(intent);builder.setProgress(0, 0, false);builder.setDefaults(downloadErrorNotificationFlag);Notification n = builder.build();n.contentIntent = intent;manager.notify(notifyId, n);sendLocalBroadcast(UPDATE_ERROR_STATUS, -1);if (updateProgressListener != null){updateProgressListener.error();}stopSelf();}/*** 下载异步任务*/private static class DownloadApk extends AsyncTask<String, Integer, String>{private WeakReference<UpdateService> updateServiceWeakReference;public DownloadApk(UpdateService service){updateServiceWeakReference = new WeakReference<>(service);}@Overrideprotected void onPreExecute() {super.onPreExecute();UpdateService service = updateServiceWeakReference.get();if (service != null){service.start();}}@Overrideprotected String doInBackground(String... params) {//注意,这里现在之前先进行清空文件,防止因检查到已经有存在的文件而无法 进行下载DataCleanManager.deleteFilesByDirectory2(""+  UpdateService.getDownloadDir(updateServiceWeakReference.get()) .getAbsolutePath());final String downloadUrl = params[0];final File file = new File(UpdateService.getDownloadDir(updateServiceWeakReference.get()),UpdateService.getSaveFileName(downloadUrl));if (DEBUG){LogUtil.e(TAG, "download url is " + downloadUrl);LogUtil.e(TAG, "download apk cache at " + file.getAbsolutePath());}File dir = file.getParentFile();if (!dir.exists()){dir.mkdirs();}HttpURLConnection httpConnection = null;InputStream is = null;FileOutputStream fos = null;int updateTotalSize = 0;URL url;try {url = new URL(downloadUrl);httpConnection = (HttpURLConnection) url.openConnection();httpConnection.setConnectTimeout(5000);httpConnection.setReadTimeout(5000);if (DEBUG){LogUtil.e(TAG, "download status code: " + httpConnection.getResponseCode());}if (httpConnection.getResponseCode() != 200) {return null;}updateTotalSize = httpConnection.getContentLength();if (file.exists()) {if (updateTotalSize == file.length()) {// 下载完成return file.getAbsolutePath();} else {file.delete();}}file.createNewFile();is = httpConnection.getInputStream();fos = new FileOutputStream(file, false);byte buffer[] = new byte[4096];int readSize = 0;int currentSize = 0;while ((readSize = is.read(buffer)) > 0) {fos.write(buffer, 0, readSize);currentSize += readSize;publishProgress((currentSize * 100 / updateTotalSize));}// download success} catch (Exception e) {e.printStackTrace();return null;} finally {if (httpConnection != null) {httpConnection.disconnect();}if (is != null) {try {is.close();} catch (IOException e) {e.printStackTrace();}}if (fos != null) {try {fos.close();} catch (IOException e) {e.printStackTrace();}}}return file.getAbsolutePath();}@Overrideprotected void onProgressUpdate(Integer... values) {super.onProgressUpdate(values);if (DEBUG){LogUtil.e(TAG, "current progress is " + values[0]);}UpdateService service = updateServiceWeakReference.get();if (service != null){service.update(values[0]);}}@Overrideprotected void onPostExecute(String s) {super.onPostExecute(s);UpdateService service = updateServiceWeakReference.get();if (service != null){if (s != null){service.success(s);}else {service.error();}}}}/*** a builder class helper use UpdateService* 仿AlertDialogUpdateService的构造器*/public static class Builder{private String downloadUrl;private int icoResId = DEFAULT_RES_ID;             //default app icoprivate int icoSmallResId = DEFAULT_RES_ID;private int updateProgress = UPDATE_NUMBER_SIZE;   //update notification progress when it add numberprivate String storeDir;          //default sdcard/Android/package/updateprivate int downloadNotificationFlag;private int downloadSuccessNotificationFlag;private int downloadErrorNotificationFlag;private boolean isSendBroadcast;protected Builder(String downloadUrl){this.downloadUrl = downloadUrl;}public static Builder create(String downloadUrl){if (downloadUrl == null) {throw new NullPointerException("downloadUrl == null");}return new Builder(downloadUrl);}public String getDownloadUrl() {return downloadUrl;}public int getIcoResId() {return icoResId;}public Builder setIcoResId(int icoResId) {this.icoResId = icoResId;return this;}public int getIcoSmallResId() {return icoSmallResId;}public Builder setIcoSmallResId(int icoSmallResId) {this.icoSmallResId = icoSmallResId;return this;}public int getUpdateProgress() {return updateProgress;}public Builder setUpdateProgress(int updateProgress) {if (updateProgress < 1){throw new IllegalArgumentException("updateProgress < 1");}this.updateProgress = updateProgress;return this;}public String getStoreDir() {return storeDir;}public Builder setStoreDir(String storeDir) {this.storeDir = storeDir;return this;}public int getDownloadNotificationFlag() {return downloadNotificationFlag;}public Builder setDownloadNotificationFlag(int downloadNotificationFlag) {this.downloadNotificationFlag = downloadNotificationFlag;return this;}public int getDownloadSuccessNotificationFlag() {return downloadSuccessNotificationFlag;}public Builder setDownloadSuccessNotificationFlag(int downloadSuccessNotificationFlag) {this.downloadSuccessNotificationFlag = downloadSuccessNotificationFlag;return this;}public int getDownloadErrorNotificationFlag() {return downloadErrorNotificationFlag;}public Builder setDownloadErrorNotificationFlag(int downloadErrorNotificationFlag) {this.downloadErrorNotificationFlag = downloadErrorNotificationFlag;return this;}public boolean isSendBroadcast() {return isSendBroadcast;}public Builder setIsSendBroadcast(boolean isSendBroadcast) {this.isSendBroadcast = isSendBroadcast;return this;}public Builder build(Context context){if (context == null){throw new NullPointerException("context == null");}Intent intent = new Intent();intent.setClass(context, UpdateService.class);intent.putExtra(URL, downloadUrl);if (icoResId == DEFAULT_RES_ID){icoResId = getIcon(context);}if (icoSmallResId == DEFAULT_RES_ID){icoSmallResId = icoResId;}intent.putExtra(ICO_RES_ID, icoResId);intent.putExtra(STORE_DIR, storeDir);intent.putExtra(ICO_SMALL_RES_ID, icoSmallResId);intent.putExtra(UPDATE_PROGRESS, updateProgress);intent.putExtra(DOWNLOAD_NOTIFICATION_FLAG, downloadNotificationFlag);intent.putExtra(DOWNLOAD_SUCCESS_NOTIFICATION_FLAG, downloadSuccessNotificationFlag);intent.putExtra(DOWNLOAD_ERROR_NOTIFICATION_FLAG, downloadErrorNotificationFlag);intent.putExtra(IS_SEND_BROADCAST, isSendBroadcast);context.startService(intent);return this;}/*** 得到系当前应用的相对应的图标* @param context* @return*/private int getIcon(Context context){final PackageManager packageManager = context.getPackageManager();ApplicationInfo appInfo = null;try {appInfo = packageManager.getApplicationInfo(context.getPackageName(), PackageManager.GET_META_DATA);} catch (PackageManager.NameNotFoundException e) {e.printStackTrace();}if (appInfo != null){return appInfo.icon;}return 0;}}}

注意这里的LocalBroadcastManager

LocalBroadcastManager基本介绍 这个类是在v4包中的,谷歌官方的介绍是:

Helper to register for and send broadcasts of Intents to local objects within your process. This is has a number of advantages over sending global broadcasts with sendBroadcast(Intent):
You know that the data you are broadcasting won’t leave your app, so don’t need to worry about leaking private data. It is not possible for other applications to send these broadcasts to your app, so you don’t need to worry about having security holes they can exploit. It is more efficient than sending a global broadcast through the system.

大致意思是:
帮助程序注册和发送Intents的广播到您的进程中的本地对象。 这是一个有趣的发送全局广播与sendBroadcast(Intent):
你知道你收音机的数据不会离开你的应用程序,所以不需要担心泄露的私人数据。 其他应用程序不可能将这些广播发送到您的应用程序,因此您不需要担心它们可以利用的安全漏洞。 它比通过系统发送全局广播更有效。

优点:

  • 能够完成在应用内的广播发送,而且比全局广播更具优势:
  • 广播只会在你的应用内发送,所以无需担心数据泄露,更加安全。
  • 其他应用无法发广播给你的应用,所以也不用担心你的应用有别人可以利用的安全漏洞
  • 相比较全局广播,它不需要发送给整个系统,所以更加高效。

使用方式

  • 广播注册:
LocalBroadcastManager localBroadcastManager = LocalBroadcastManager.getInstance(getActivity());IntentFilter filter = new IntentFilter();filter.addAction(ACTION);myBroadcastReciver = new MyBroadcastReciver();localBroadcastManager.registerReceiver(myBroadcastReciver, filter);
  • 广播发送
 Intent intent = new Intent();intent.setAction(SaleLeftFragment.ACTION);intent.putExtra(TAG, data);LocalBroadcastManager.getInstance(getActivity()).sendBroadcast(intent);
  • 使用注意

在使用的时候,请关注以下几点:

  • 1).LocalBroadcastManager注册广播只能通过代码注册的方式。
  • 2).LocalBroadcastManager注册广播后,一定要记得取消监听。
  • 3).重点的重点,使用LocalBroadcastManager注册的广播,您在发送广播的时候务必使用LocalBroadcastManager.sendBroadcast(intent);否则您接收不到广播

主Activity中的代码:


public class MainActivity extends AppCompatActivity {/*** 下载地址的URL*/private static final String URL = "http://192.168.1.11/mydoctor.apk";@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);ButterKnife.inject(this);}public void update(View view){UpdateService.Builder.create(URL).build(this);}@OnClick(R.id.btn_downLoad)public void onClick(View view){UpdateService.Builder.create(URL).setStoreDir("update/flag").setDownloadSuccessNotificationFlag(Notification.DEFAULT_ALL).setDownloadErrorNotificationFlag(Notification.DEFAULT_ALL).build(this);}
}

项目下载地址:
https://github.com/androidstarjack/ServiceDownLoadApp-master

如果你觉得此文对您有所帮助,欢迎入群 QQ交流群 :232203809   
微信公众号:终端研发部

Markdown

(欢迎关注学习和交流)

这篇关于手写Service后台下载app——跳出DownloadManager系统7.0之坑的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/212279

相关文章

CentOS系统Maven安装教程分享

《CentOS系统Maven安装教程分享》本文介绍了如何在CentOS系统中安装Maven,并提供了一个简单的实际应用案例,安装Maven需要先安装Java和设置环境变量,Maven可以自动管理项目的... 目录准备工作下载并安装Maven常见问题及解决方法实际应用案例总结Maven是一个流行的项目管理工具

Python实现文件下载、Cookie以及重定向的方法代码

《Python实现文件下载、Cookie以及重定向的方法代码》本文主要介绍了如何使用Python的requests模块进行网络请求操作,涵盖了从文件下载、Cookie处理到重定向与历史请求等多个方面,... 目录前言一、下载网络文件(一)基本步骤(二)分段下载大文件(三)常见问题二、requests模块处理

使用TomCat,service输出台出现乱码的解决

《使用TomCat,service输出台出现乱码的解决》本文介绍了解决Tomcat服务输出台中文乱码问题的两种方法,第一种方法是修改`logging.properties`文件中的`prefix`和`... 目录使用TomCat,service输出台出现乱码问题1解决方案问题2解决方案总结使用TomCat,

C#实现系统信息监控与获取功能

《C#实现系统信息监控与获取功能》在C#开发的众多应用场景中,获取系统信息以及监控用户操作有着广泛的用途,比如在系统性能优化工具中,需要实时读取CPU、GPU资源信息,本文将详细介绍如何使用C#来实现... 目录前言一、C# 监控键盘1. 原理与实现思路2. 代码实现二、读取 CPU、GPU 资源信息1.

在C#中获取端口号与系统信息的高效实践

《在C#中获取端口号与系统信息的高效实践》在现代软件开发中,尤其是系统管理、运维、监控和性能优化等场景中,了解计算机硬件和网络的状态至关重要,C#作为一种广泛应用的编程语言,提供了丰富的API来帮助开... 目录引言1. 获取端口号信息1.1 获取活动的 TCP 和 UDP 连接说明:应用场景:2. 获取硬

JAVA系统中Spring Boot应用程序的配置文件application.yml使用详解

《JAVA系统中SpringBoot应用程序的配置文件application.yml使用详解》:本文主要介绍JAVA系统中SpringBoot应用程序的配置文件application.yml的... 目录文件路径文件内容解释1. Server 配置2. Spring 配置3. Logging 配置4. Ma

2.1/5.1和7.1声道系统有什么区别? 音频声道的专业知识科普

《2.1/5.1和7.1声道系统有什么区别?音频声道的专业知识科普》当设置环绕声系统时,会遇到2.1、5.1、7.1、7.1.2、9.1等数字,当一遍又一遍地看到它们时,可能想知道它们是什... 想要把智能电视自带的音响升级成专业级的家庭影院系统吗?那么你将面临一个重要的选择——使用 2.1、5.1 还是

解决systemctl reload nginx重启Nginx服务报错:Job for nginx.service invalid问题

《解决systemctlreloadnginx重启Nginx服务报错:Jobfornginx.serviceinvalid问题》文章描述了通过`systemctlstatusnginx.se... 目录systemctl reload nginx重启Nginx服务报错:Job for nginx.javas

高效管理你的Linux系统: Debian操作系统常用命令指南

《高效管理你的Linux系统:Debian操作系统常用命令指南》在Debian操作系统中,了解和掌握常用命令对于提高工作效率和系统管理至关重要,本文将详细介绍Debian的常用命令,帮助读者更好地使... Debian是一个流行的linux发行版,它以其稳定性、强大的软件包管理和丰富的社区资源而闻名。在使用

Ubuntu系统怎么安装Warp? 新一代AI 终端神器安装使用方法

《Ubuntu系统怎么安装Warp?新一代AI终端神器安装使用方法》Warp是一款使用Rust开发的现代化AI终端工具,该怎么再Ubuntu系统中安装使用呢?下面我们就来看看详细教程... Warp Terminal 是一款使用 Rust 开发的现代化「AI 终端」工具。最初它只支持 MACOS,但在 20