Android包管理机制(一) PackageInstaller的初始化

2023-05-16

Android包管理机制(一) PackageInstaller的初始化

包管理机制是Android中的重要机制,是应用开发和系统开发需要掌握的知识点之一。
包指的是Apk、jar和so文件等等,它们被加载到Android内存中,由一个包转变成可执行的代码,这就需要一个机制来进行包的加载、解析、管理等操作,这就是包管理机制。包管理机制由许多类一起组成,其中核心为PackageManagerService(PMS),它负责对包进行管理,如果直接讲PMS会比较难以理解,因此我们需要一个切入点,这个切入点就是常见的APK的安装。
讲到APK的安装之前,先了解下PackageManager、APK文件结构和安装方式。

1.PackageManager简介

与ActivityManager和AMS的关系类似,PMS也有一个对应的管理类PackageManager,用于向应用程序进程提供一些功能。PackageManager是一个抽象类,它的具体实现类为ApplicationPackageManager,ApplicationPackageManager中的方法会通过IPackageManager与AMS进行进程间通信,因此PackageManager所提供的功能最终是由PMS来实现的,这么设计的主要用意是为了避免系统服务PMS直接被访问。PackageManager提供了一些功能,主要有以下几点:

  1. 获取一个应用程序的所有信息(ApplicationInfo)。
  2. 获取四大组件的信息。
  3. 查询permission相关信息。
  4. 获取包的信息。
  5. 安装、卸载APK.

2.APK文件结构和安装方式

APK是AndroidPackage的缩写,即Android安装包,它实际上是zip格式的压缩文件,一般情况下,解压后的文件结构如下表所示。

目录/文件描述
assert存放的原生资源文件,通过AssetManager类访问。
lib存放库文件。
META-INF保存应用的签名信息,签名信息可以验证APK文件的完整性。
res存放资源文件。res中除了raw子目录,其他的子目录都参与编译,这些子目录下的资源是通过编译出的R类在代码中访问。
AndroidManifest.xml用来声明应用程序的包名称、版本、组件和权限等数据。 apk中的AndroidManifest.xml经过压缩,可以通过AXMLPrinter2工具解开。
classes.dexJava源码编译后生成的Java字节码文件。
resources.arsc编译后的二进制资源文件。

APK的安装场景主要有以下几种:

  • 通过adb命令安装:adb 命令包括adb push/install
  • 通过系统安装器packageinstaller进行安装:packageinstaller是系统内置的应用程序,用于安装和卸载应用程序。
  • 系统应用安装
  • 应用商店自动安装

这4种方式最终都会调用PMS的scanPackageDirtyLI方法用来解析包,在此之前的调用链是不同的,本篇文章会介绍第二种方式,对于用户来说,这是最常用的安装方式;对于开发者来说,这是调用链比较长的安装方式,能学到的更多。其他的安装场景会在本系列的后续文章进行讲解。

3.寻找PackageInstaller入口

在Android7.0之前我们可以通过如下代码安装指定路径中的APK。


Intent intent = new Intent(Intent.ACTION_VIEW);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(Uri.parse("file://" + path),"application/vnd.android.package-archive");
context.startActivity(intent);  

  但是Android7.0或更高版本再这么做,就会报FileUriExposedException异常。这是因为StrictMode API 政策禁止应用程序将file:// Uri暴露给另一个应用程序,如果包含file:// Uri的 intent 离开你的应用,就会报FileUriExposedException 异常。为了解决这个问题,谷歌提供了FileProvider,FileProvider继承自ContentProvider ,使用它可以将file://Uri替换为content://Uri,具体怎么使用FileProvider并不是本文的重点,只要知道无论是Android7.0之前还是Android7.0以及更高版本,都会调用如下代码:


Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setDataAndType(xxxxx, "application/vnd.android.package-archive");  

Intent的Action属性为ACTION_VIEW,Type属性指定Intent的数据类型为application/vnd.android.package-archive。
能隐式匹配的Activity为InstallStart,需要注意的是,这里分析的源码基于Android8.0,7.0能隐式匹配的Activity为PackageInstallerActivity。
packages/apps/PackageInstaller/AndroidManifest.xml

复制代码


<activity android:name=".InstallStart"
               android:exported="true"
               android:excludeFromRecents="true">
           <intent-filter android:priority="1">
               <action android:name="android.intent.action.VIEW" />
               <action android:name="android.intent.action.INSTALL_PACKAGE" />
               <category android:name="android.intent.category.DEFAULT" />
               <data android:scheme="file" />
               <data android:scheme="content" />
               <data android:mimeType="application/vnd.android.package-archive" />
           </intent-filter>
        ...
</activity>  

复制代码

InstallStart是PackageInstaller中的入口Activity,其中PackageInstaller是系统内置的应用程序,用于安装和卸载应用。当我们调用PackageInstaller来安装应用时会跳转到InstallStart,并调用它的onCreate方法:
packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStart.java

复制代码


@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
         if (PackageInstaller.ACTION_CONFIRM_PERMISSIONS.equals(intent.getAction())) {//1
          nextActivity.setClass(this, PackageInstallerActivity.class);
      } else {
          Uri packageUri = intent.getData();
          if (packageUri == null) {//2
              Intent result = new Intent();
              result.putExtra(Intent.EXTRA_INSTALL_RESULT,
                      PackageManager.INSTALL_FAILED_INVALID_URI);
              setResult(RESULT_FIRST_USER, result);
              nextActivity = null;
          } else {
              if (packageUri.getScheme().equals(SCHEME_CONTENT)) {//3
                  nextActivity.setClass(this, InstallStaging.class);
              } else {
                  nextActivity.setClass(this, PackageInstallerActivity.class);
              }
          }
      }
      if (nextActivity != null) {
          startActivity(nextActivity);
      }
      finish();
  }  

复制代码

注释1处判断Intent的Action是否为CONFIRM_PERMISSIONS,根据本文的应用情景显然不是,接着往下看,注释2处判断packageUri 是否为空也不成立,注释3处,判断Uri的Scheme协议是否是content,如果是就跳转到InstallStaging,如果不是就跳转到PackageInstallerActivity。本文的应用情景中,Android7.0以及更高版本我们会使用FileProvider来处理URI ,FileProvider会隐藏共享文件的真实路径,将路径转换成content://Uri路径,这样就会跳转到InstallStaging。InstallStaging的onResume方法如下所示。

packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStaging.java

复制代码


@Override
protected void onResume() {
      super.onResume();
      if (mStagingTask == null) {
          if (mStagedFile == null) {
              try {
                  mStagedFile = TemporaryFileManager.getStagedFile(this);//1
              } catch (IOException e) {
                  showError();
                  return;
              }
          }
          mStagingTask = new StagingAsyncTask();
          mStagingTask.execute(getIntent().getData());//2
      }
  }  

复制代码

注释1处如果File类型的mStagedFile 为null,则创建mStagedFile ,mStagedFile用于存储临时数据。 注释2处启动StagingAsyncTask,并传入了content协议的Uri,如下所示。
packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStaging.java

复制代码


private final class StagingAsyncTask extends AsyncTask<Uri, Void, Boolean> {
        @Override
        protected Boolean doInBackground(Uri... params) {
            if (params == null || params.length <= 0) {
                return false;
            }
            Uri packageUri = params[0];
            try (InputStream in = getContentResolver().openInputStream(packageUri)) {
                if (in == null) {
                    return false;
                }
                try (OutputStream out = new FileOutputStream(mStagedFile)) {
                    byte[] buffer = new byte[4096];
                    int bytesRead;
                    while ((bytesRead = in.read(buffer)) >= 0) {
                        if (isCancelled()) {
                            return false;
                        }
                        out.write(buffer, 0, bytesRead);
                    }
                }
            } catch (IOException | SecurityException e) {
                Log.w(LOG_TAG, "Error staging apk from content URI", e);
                return false;
            }
            return true;
        }
        @Override
        protected void onPostExecute(Boolean success) {
            if (success) {
                Intent installIntent = new Intent(getIntent());
                installIntent.setClass(InstallStaging.this, PackageInstallerActivity.class);
                installIntent.setData(Uri.fromFile(mStagedFile));
                installIntent
                        .setFlags(installIntent.getFlags() & ~Intent.FLAG_ACTIVITY_FORWARD_RESULT);
                installIntent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION);
                startActivityForResult(installIntent, 0);
            } else {
                showError();
            }
        }
    }
}  

复制代码

doInBackground方法中将packageUri(content协议的Uri)的内容写入到mStagedFile中,如果写入成功,onPostExecute方法中会跳转到PackageInstallerActivity中,并将mStagedFile传进去。绕了一圈又回到了PackageInstallerActivity,这里可以看出InstallStaging主要起了转换的作用,将content协议的Uri转换为File协议,然后跳转到PackageInstallerActivity,这样就可以像此前版本(Android7.0之前)一样启动安装流程了。

4.PackageInstallerActivity解析

从功能上来说,PackageInstallerActivity才是应用安装器PackageInstaller真正的入口Activity,PackageInstallerActivity的onCreate方法如下所示。
packages/apps/PackageInstaller/src/com/android/packageinstaller/PackageInstallerActivity.java

复制代码


@Override
protected void onCreate(Bundle icicle) {
    super.onCreate(icicle);
    if (icicle != null) {
        mAllowUnknownSources = icicle.getBoolean(ALLOW_UNKNOWN_SOURCES_KEY);
    }
    mPm = getPackageManager();
    mIpm = AppGlobals.getPackageManager();
    mAppOpsManager = (AppOpsManager) getSystemService(Context.APP_OPS_SERVICE);
    mInstaller = mPm.getPackageInstaller();
    mUserManager = (UserManager) getSystemService(Context.USER_SERVICE);
    ...
    //根据Uri的Scheme进行预处理
    boolean wasSetUp = processPackageUri(packageUri);//1
    if (!wasSetUp) {
        return;
    }
    bindUi(R.layout.install_confirm, false);
    //判断是否是未知来源的应用,如果开启允许安装未知来源选项则直接初始化安装
    checkIfAllowedAndInitiateInstall();//2
}  

复制代码

首先初始话安装所需要的各种对象,比如PackageManager、IPackageManager、AppOpsManager和UserManager等等,它们的描述如下表所示。

类名描述
PackageManager用于向应用程序进程提供一些功能,最终的功能是由PMS来实现的
IPackageManager一个AIDL的接口,用于和PMS进行进程间通信
AppOpsManager用于权限动态检测,在Android4.3中被引入
PackageInstaller提供安装、升级和删除应用程序功能
UserManager用于多用户管理

注释1处的processPackageUri方法如下所示。
packages/apps/PackageInstaller/src/com/android/packageinstaller/PackageInstallerActivity.java

复制代码


private boolean processPackageUri(final Uri packageUri) {
     mPackageURI = packageUri;
     final String scheme = packageUri.getScheme();//1
     switch (scheme) {
         case SCHEME_PACKAGE: {
             try {
              ...
         } break;
         case SCHEME_FILE: {
             File sourceFile = new File(packageUri.getPath());//1
             //得到sourceFile的包信息
             PackageParser.Package parsed = PackageUtil.getPackageInfo(this, sourceFile);//2
             if (parsed == null) {
                 Log.w(TAG, "Parse error when parsing manifest. Discontinuing installation");
                 showDialogInner(DLG_PACKAGE_ERROR);
                 setPmResult(PackageManager.INSTALL_FAILED_INVALID_APK);
                 return false;
             }
             //对parsed进行进一步处理得到包信息PackageInfo
             mPkgInfo = PackageParser.generatePackageInfo(parsed, null,
                     PackageManager.GET_PERMISSIONS, 0, 0, null,
                     new PackageUserState());//3
             mAppSnippet = PackageUtil.getAppSnippet(this, mPkgInfo.applicationInfo, sourceFile);
         } break;
         default: {
             Log.w(TAG, "Unsupported scheme " + scheme);
             setPmResult(PackageManager.INSTALL_FAILED_INVALID_URI);
             finish();
             return false;
         }
     }
     return true;
 }  

复制代码

首先在注释1处得到packageUri的Scheme协议,接着根据这个Scheme协议分别对package协议和file协议进行处理,如果不是这两个协议就会关闭PackageInstallerActivity并return false。我们主要来看file协议的处理,注释1处根据packageUri创建一个新的File。注释2处的内部会用PackageParser的parsePackage方法解析这个File(这个File其实是APK文件),得到APK的包信息Package ,Package包含了该APK的所有信息。注释3处会将Package根据uid、用户状态信息和PackageManager的配置等变量对包信息Package做进一步处理得到PackageInfo。
回到PackageInstallerActivity的onCreate方法的注释2处,checkIfAllowedAndInitiateInstall方法如下所示。
packages/apps/PackageInstaller/src/com/android/packageinstaller/PackageInstallerActivity.java

复制代码


private void checkIfAllowedAndInitiateInstall() {
       //判断如果允许安装未知来源或者根据Intent判断得出该APK不是未知来源
       if (mAllowUnknownSources || !isInstallRequestFromUnknownSource(getIntent())) {//1
           //初始化安装
           initiateInstall();//2
           return;
       }
       // 如果管理员限制来自未知源的安装, 就弹出提示Dialog或者跳转到设置界面
       if (isUnknownSourcesDisallowed()) {
           if ((mUserManager.getUserRestrictionSource(UserManager.DISALLOW_INSTALL_UNKNOWN_SOURCES,
                   Process.myUserHandle()) & UserManager.RESTRICTION_SOURCE_SYSTEM) != 0) {    
               showDialogInner(DLG_UNKNOWN_SOURCES_RESTRICTED_FOR_USER);
               return;
           } else {
               startActivity(new Intent(Settings.ACTION_SHOW_ADMIN_SUPPORT_DETAILS));
               finish();
           }
       } else {
           handleUnknownSources();//3
       }
   }  

复制代码

注释1处判断允许安装未知来源或者根据Intent判断得出该APK不是未知来源,就调用注释2处的initiateInstall方法来初始化安装。如果管理员限制来自未知源的安装, 就弹出提示Dialog或者跳转到设置界面,否则就调用注释3处的handleUnknownSources方法来处理未知来源的APK。注释2处的initiateInstall方法如下所示。
packages/apps/PackageInstaller/src/com/android/packageinstaller/PackageInstallerActivity.java

复制代码


private void initiateInstall() {
      String pkgName = mPkgInfo.packageName;//1
      String[] oldName = mPm.canonicalToCurrentPackageNames(new String[] { pkgName });
      if (oldName != null && oldName.length > 0 && oldName[0] != null) {
          pkgName = oldName[0];
          mPkgInfo.packageName = pkgName;
          mPkgInfo.applicationInfo.packageName = pkgName;
      }
      try {
          //根据包名获取应用程序信息
          mAppInfo = mPm.getApplicationInfo(pkgName,
                  PackageManager.MATCH_UNINSTALLED_PACKAGES);//2
          if ((mAppInfo.flags&ApplicationInfo.FLAG_INSTALLED) == 0) {
              mAppInfo = null;
          }
      } catch (NameNotFoundException e) {
          mAppInfo = null;
      }
      //初始化安装确认界面
      startInstallConfirm();//3
  }  

复制代码

注释1处得到包名,注释2处根据包名获取获取应用程序信息ApplicationInfo。注释3处的startInstallConfirm方法如下所示。
packages/apps/PackageInstaller/src/com/android/packageinstaller/PackageInstallerActivity.java

复制代码


private void startInstallConfirm() {
    //省略初始化界面代码
     ...
     AppSecurityPermissions perms = new AppSecurityPermissions(this, mPkgInfo);//1
     final int N = perms.getPermissionCount(AppSecurityPermissions.WHICH_ALL);
     if (mAppInfo != null) {
         msg = (mAppInfo.flags & ApplicationInfo.FLAG_SYSTEM) != 0
                 ? R.string.install_confirm_question_update_system
                 : R.string.install_confirm_question_update;
         mScrollView = new CaffeinatedScrollView(this);
         mScrollView.setFillViewport(true);
         boolean newPermissionsFound = false;
         if (!supportsRuntimePermissions) {
             newPermissionsFound =
                     (perms.getPermissionCount(AppSecurityPermissions.WHICH_NEW) > 0);
             if (newPermissionsFound) {
                 permVisible = true;
                 mScrollView.addView(perms.getPermissionsView(
                         AppSecurityPermissions.WHICH_NEW));//2
             }
         }
     ...
 }  

复制代码

startInstallConfirm方法中首先初始化安装确认界面,就是我们平常安装APK时出现的界面,界面上有确认和取消按钮并会列出安装该APK需要访问的系统权限。需要注意的是,不同厂商定制的Android系统会有不同的安装确认界面。
注释1处会创建AppSecurityPermissions,它会提取出APK中权限信息并展示出来,这个负责展示的View是AppSecurityPermissions的内部类PermissionItemView。注释2处调用AppSecurityPermissions的getPermissionsView方法来获取PermissionItemView,并将PermissionItemView添加到CaffeinatedScrollView中,这样安装该APK需要访问的系统权限就可以全部的展示出来了,PackageInstaller的初始化工作就完成了。

5.总结

现在来总结下PackageInstaller初始化的过程:

  1. 根据Uri的Scheme协议不同,跳转到不同的界面,content协议跳转到InstallStart,其他的跳转到PackageInstallerActivity。本文应用场景中,如果是Android7.0以及更高版本会跳转到InstallStart。
  2. InstallStart将content协议的Uri转换为File协议,然后跳转到PackageInstallerActivity。
  3. PackageInstallerActivity会分别对package协议和file协议的Uri进行处理,如果是file协议会解析APK文件得到包信息PackageInfo。
  4. PackageInstallerActivity中会对未知来源进行处理,如果允许安装未知来源或者根据Intent判断得出该APK不是未知来源,就会初始化安装确认界面,如果管理员限制来自未知源的安装, 就弹出提示Dialog或者跳转到设置界面。

PackageInstaller的初始化就讲到这,关于PackageInstaller的安装APK的过程会在本系列的下一篇文章进行讲解。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Android包管理机制(一) PackageInstaller的初始化 的相关文章

  • QT QCompleter 用法小结

    1 有何用处 QCompleter类提供基于项模型的补全功能 您可以使用QCompleter在任何Qt小部件 xff08 如QLineEdit和QComboBox xff09 中提供自动补全 当用户开始键入单词时 xff0c QComple
  • QT 关于全屏 与 退出全屏

    在某些场景 xff08 比如说 xff1a 车机系统 医疗仪器等 xff09 可能需要确保软件的全屏显示 xff0c 本次探讨的是如何实现全屏与退出全屏 实现全屏的几种方式 span class token operator lt span
  • QT 如何设置 QCompleter 样式

    在使用QCompleter xff0c 一般都会面临两个问题 xff1a 1 如何设置Completer的行高 xff1b 2 如何设置Completer的样式 本文将一次为你解答 xff0c 顺便讲讲ComboBox设置Completer
  • QT 如何知道用户点击 QTextBrowser 中的具体链接地址

    在通过鼠标点击 QTextBrowser 中的链接地址时 xff0c 如果是希望使用外部浏览器打开链接地址 xff0c 只需要添加一行代码 ui gt textBrowser gt setOpenExternalLinks true 但如果
  • QT 常用控件类型命名参考

    规范的定义 规范 xff0c 有名词 动词 代词等词性 意指符合逻辑 xff0c 客观 真实 全面 完整 准确 及时 xff0c 达标 明文规定或约定俗成的标准 如 xff1a 道德规范 技术规范等 或是指按照既定标准 规范的要求进行操作
  • C++ 如何让代码在main()函数之前或者之后执行

    程序最早执行的函数其实并不是main xff0c 在main函数之前会有一系列初始化的操作 xff08 通常是由链接器等完成的 xff09 比如在windows中 xff0c 这个函数是mainCRTStartup xff0c 这个函数是链
  • QT 如何提高 Qt Creator 的编译速度

    如何提高编译速度 xff0c 貌似是一个老生常谈的话题 对于Qter而言 xff0c 如何提高QT Creator 的编辑速度是一直都是大家所期盼的 本文也是查阅了各路大神的方法后整理出来的 xff0c 希望对各位有所帮助 1 在 pro文
  • 卧槽!出了一个Python实时目标跟踪系统神器!

    在当下自动驾驶 智慧城市 安防等领域对车辆 行人 飞行器等快速移动的物体进行实时跟踪及分析的需求可谓比比皆是 xff0c 但单纯的目标检测算法只能输出目标的定位 43 分类 xff0c 无法对移动的目标具体的运动行为及特征进行分析 xff0
  • C++ 什么是BCC校验

    BCC Block Check Character xff0c 俗称异或校验 BCC的实现方法 xff1a 将所有数据都和一个指定的初始值 通常是0 异或一次 xff0c 所得结果为校验值 BCC一般只是用来排错的 xff0c 并不是加密算
  • QT 如何让QLineEdit的文字从最左边开始显示

    当QLineEdit的文字长度超过了控件宽度 xff0c 一般而言 xff08 控件文字设为靠左对齐 xff09 xff0c 用户只能看到这一长串文字的后半段部分 但是如果想让QLineEdit从最左边开始显示字符串内容呢 xff1f 举个
  • QT setBuddy 用法小结

    将此标签的好友 xff08 窗口其他小部件 xff09 设置为好友 当用户按下此标签指示的快捷键时 xff0c 键盘焦点将转移到标签的好友小部件 好友机制仅适用于包含一个字符前缀为 amp 的文本的QLabels 此字符被设置为快捷键 设置
  • QT stackUnder 用法小结

    stackUnder是什么意思 xff1f 根据官方的解析 xff0c 简而言之一句话 xff1a 把窗口的小部件放置于父窗口的堆栈窗口中 本文将通过一则简单的示例 xff0c 来具体解析stackUnder该如何使用 官方解析 void
  • QT 线程安全的单例模式(使用Q_GLOBAL_STATIC实现)

    Qt提供了宏 Q GLOBAL STATIC xff08 官方说明文档 xff1a https doc qt io qt 5 qglobalstatic html xff09 xff0c 用于创建全局静态对象 鉴于此 xff0c 我们可以通
  • git push提示“fatal: the remote end hung up unexpectedly”的解决方法

    问题描述 在执行push命令时 xff0c 命令行提示 xff1a span class token function git span push origin master Enumerating objects 4968 done Co
  • 四旋翼无人机动力学模型及控制

    四旋翼无人机动力学模型及控制 I 欧拉角与旋转矩阵 Overview欧拉角与旋转矩阵Body Frame Angular Velocity and
  • Urbannav数据集/novatel_data/inspvax订阅

    使用Urbannav数据集做多原融合定位评估精度 xff0c 需要使用 novatel data inspvax作为真值 xff0c 但无法直接订阅该话题 rotopic echo novatel data inspvax 显示无法订阅 x
  • 软路由硬件, 研究了一圈还是J1900 , i211网卡又如何

    软路由的心路历程还真是纠结 xff0c 现写出来供大家参考 需求上也是几经翻车 xff1a 路由 xff0c NAS 之类的一体机是最先迸发出来也是最先被否的 本着大道至简 xff0c 设备专用的原则 软路由即路由 xff0c 不做他用 功
  • putty远程连接ubuntu18失败处理方法

    putty客户端远程连接报错 xff1a Network error Connection refused 解决方法 xff1a 1 先在Windows系统cmd命令下ping下此ip地址连接是否正常 xff0c 如果连接正常说明主机没有问
  • KITTI榜单新SOTA!相机-LiDAR双向融合新范式 | CVPR 2022 Oral & arXiv 2023

    点击下方卡片 xff0c 关注 CVer 公众号 AI CV重磅干货 xff0c 第一时间送达 点击进入 gt 计算机视觉 微信技术交流群 作者 xff1a 王利民 xff08 已授权转载 xff09 编辑 xff1a CVer https
  • C++ Primer 学习笔记 第七章 类

    233 定义类 span class token macro property span class token directive hash span span class token directive keyword include

随机推荐

  • 如何编写CMakeLists

    Preface 构建一个工程的时候 xff0c CMake需要知道的几个点 xff08 如果下面几件事你知道怎么做了 xff0c 多大的工程就都不是问题了 A 源代码在哪里 xff1f B 头文件在哪里 xff1f C 怎么生成静态或者动态
  • 【算法】最快最简单的排序——桶排序

    在我们生活的这个世界中到处都是被排序过的 站队的时候会按照身高排序 xff0c 考试的名次需要按照分数排序 xff0c 网上购物的时候会按照价格排序 xff0c 电子邮箱中的邮件按照时间排序 总之很多东西都需要排序 xff0c 可以说排序是
  • TCP、UDP数据包大小的限制

    1 概述 首先要看TCP IP协议 xff0c 涉及到四层 xff1a 链路层 xff0c 网络层 xff0c 传输层 xff0c 应用层 其中以太网 xff08 Ethernet xff09 的数据帧在链路层 IP包在网络层 TCP或UD
  • Jetson TX2刷机教程

    介绍 xff1a 本文介绍如何对Jetson TX2进行刷机 xff0c 系统版本为Jetpack4 6 0 准备 xff1a 主机 xff08 虚拟机 xff09 xff1a Ubuntu18 04Jetson TX2USB 连接线 刷机
  • 研扬Jetson NX镜像备份和恢复

    0 环境依赖 研扬RC S ARES 200AI NX CSC00型号Jetpack4 6 0 1 如何进入Recovery模式 1 先按住Recovery键再插电源上电 xff0c 此间一直按住Recovery键2 3秒之后松开 2 将U
  • SpringCloud-Eureka快速入门,集群搭建

    Eureka 个人主页 xff1a https blog csdn net hello list type 61 blog 前言 第一章 xff1a SpringCloud环境搭建 Rest使用 这里博主从说更新springcloud xf
  • SpringCloud-Ribbon和Feign快速上手

    Ribbon 个人中心 xff1a https blog csdn net hello list 前情提要 xff1a SpringCloud环境搭建 Rest使用SpringCloud Eureka快速入门 xff0c 集群搭建 首先我们
  • Java IO流详解

    Java IO流详解 个人主页 xff1a https blog csdn net hello list 今天我们来学习下java中的io部分 首先我们要知道io指的是什么 xff0c 输入输出 xff0c 就是输入输出流 xff0c 我们
  • SpringCloud-Hystrix服务熔断,快速入门

    Hystrix服务熔断 个人中心 xff1a 学习日记的博客 前情提要 xff1a SpringCloud环境搭建 Rest使用SpringCloud Eureka快速入门 xff0c 集群搭建SpringCloud Ribbon和Feig
  • 最新!!Intel首发UP SQUARED* GROVE物联网开发套件

    xfeff Intel官方网站正式推出UP SQUARED GROVE 物联网开发套件 https software intel com zh cn iot hardware up squared grove dev kit UP Squa
  • 再探指针:指针有什么用?(课堂笔记,来自翁恺老师的C语言进阶课,有着个人理解)

    指针的用处 xff1a 三个比较常用的场景 我们希望函数可以返回多个值的时候 我们都知道函数只可以return一个值 xff0c 使用指针便可以在一个函数里面返回多个值 举例 xff1a swap函数和数组中寻找最大值最小值的函数 xff0
  • java反射与注解详解,共同实现动态代理模式

    java反射与注解详解 xff0c 共同实现动态代理模式 个人主页 xff1a https blog csdn net hello list id xff1a 学习日记 不知不觉一年过去了 xff0c 整整一年 xff0c 这一年写了60多
  • 微信小程序快速入门

    微信小程序快速入门 在这里首先祝大家国庆节快乐 xff0c 其实原本文章都没有准备好 xff0c 也没有打算更文的 xff0c 那还是将就一下吧 xff0c 发个简单的 前言 相信大家对微信小程序并不陌生 xff0c 以前我们接触网络刚开始
  • RabbitMQ快速入门,这一篇看完教你学会

    RabbitMQ快速入门 今天学习RabbitMQ xff0c 你知道RabbitMQ是什么吗 xff0c RabbitMQ是一种消息中间件 xff0c 我们在写很多业务的时候 xff0c 有时候我们需要考虑到消息的实时性 xff0c 时效
  • 什么是协议栈

    协议栈是什么 1 协议栈是什么 简介 协议栈 xff0c 英语名称为Protocol stack xff0c 又称协议堆叠 xff0c 是计算机网络协议套件的一个具体的软件实现 协议套件中的一个协议通常是只为一个目的而设计的 xff0c 这
  • 任务,任务的切换,(TCB)

    任务也可以称作为进程 xff0c 是一个简单的程序 xff0c 该程序认为 CPU完全属于自己 xff0c 实时的应用的程序的设计的时候分割成了许多的任务 xff0c 每一个任务都对应应用的某一部分 每一个任务都被赋予一定的优先级 xff0
  • 浅谈pthread_setschedparam的使用

    浅谈pthread setschedparam 的使用 int pthread setschedparam pthread t target thread int policy const struct sched param param
  • 互斥量、临界区、信号量、事件标志组和消息邮箱

    http ejs90ejs iteye com blog 1351642 互斥量 临界区 信号量 事件标志组和消息邮箱 2010年07月23日 为了好的理解互斥量 临界区 信号量 事件标志组和消息邮箱 xff0c 下面一些知识对初学者来说很
  • C语言中的多线程简介

    线程 Thread 专业术语称之为程序执行流的最小单元 线程是不会执行程序的 xff0c 可以理解成线程就是一个载体 xff0c 将 要执行的代码 运送到CPU进行处理 多线程就是多个线程同时并发执行 xff08 注意并发与并行的区别 xf
  • Android包管理机制(一) PackageInstaller的初始化

    Android包管理机制 xff08 一 xff09 PackageInstaller的初始化 前言 包管理机制是Android中的重要机制 xff0c 是应用开发和系统开发需要掌握的知识点之一 包指的是Apk jar和so文件等等 xff