【干货】Android系统定制基础篇:第三部分(Android静默安装、Android支持usb打印机)

2023-11-09

1、Android静默安装

一些产品要求APP在升级时能够实现静默安装,而无需弹出安装界面让用户确认。这里提出两种实现方案:

方案一 APP调用『pm』命令实现静默安装,此方案无须修改Android源码,但需要root权限。
方案二 修改Android PackageInstaller 源码,增加Intent参数来指定是否要静默安装,同时支持通过属性配置需要静默安装的APP包名,只要是属性配置的包名就走静默安装,其它APP进默认安装。

方案一 通过pm命令安装

APP调用『pm』命令实现静默安装,此方案无须修改Android源码,但需要root权限。实现如下:

/**
 * Silent install
 *
 * @param path Package
 * @return true: success false: failed
 */
public static boolean installSilent(String path) {
    boolean result = false;
    BufferedReader es = null;
    DataOutputStream os = null;

    try {
        Process process = Runtime.getRuntime().exec("su");
        os = new DataOutputStream(process.getOutputStream());

        String command = "pm install -r " + path + "\n";
        os.write(command.getBytes(Charset.forName("utf-8")));
        os.flush();
        os.writeBytes("exit\n");
        os.flush();

        process.waitFor();
        es = new BufferedReader(new InputStreamReader(process.getErrorStream()));

        String line;
        StringBuilder builder = new StringBuilder();
        while ((line = es.readLine()) != null) {
            builder.append(line);
        }
        Log.d(TAG, "install msg is " + builder.toString());

        /* Installation is considered a Failure if the result contains
            the Failure character, or a success if it is not.
             */
        if (!builder.toString().contains("Failure")) {
            result = true;
        }
    } catch (Exception e) {
        Log.e(TAG, e.getMessage(), e);
    } finally {
        try {
            if (os != null) {
                os.close();
            }
            if (es != null) {
                es.close();
            }
        } catch (IOException e) {
            Log.e(TAG, e.getMessage(), e);
        }
    }

    return result;
}

方案二 修改PackageInstaller源码

如果没有root权限,方案一将无法实现,因此我们通过定制 PackageInstaller 来实现指定包名可以静默安装,并增加Intent参数来指定静默安装还是默认安装。具体修改如下:

diff --git a/packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStart.java b/packages/apps/Pac
old mode 100644
new mode 100755
index 12441b5..cbf8c41
--- a/packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStart.java
+++ b/packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStart.java
@@ -22,17 +22,30 @@ import android.app.ActivityManager;
 import android.app.AppGlobals;
 import android.app.IActivityManager;
 import android.content.ContentResolver;
+import android.content.Context;
 import android.content.Intent;
 import android.content.pm.ApplicationInfo;
 import android.content.pm.IPackageManager;
 import android.content.pm.PackageInstaller;
 import android.content.pm.PackageManager;
+import android.content.pm.PackageInfo;
+import android.content.pm.PackageParser;
+import android.content.pm.PackageUserState;
+import android.content.pm.ProviderInfo;
 import android.net.Uri;
 import android.os.Build;
 import android.os.Bundle;
 import android.os.RemoteException;
+import android.os.SystemProperties;
 import android.support.annotation.Nullable;
+import android.text.TextUtils;
 import android.util.Log;
+import android.content.pm.IPackageInstallObserver;
+import android.support.v4.content.FileProvider;
+
+import java.io.File;
+import java.lang.reflect.Method;
+import java.util.List;
 
 import com.android.internal.annotations.VisibleForTesting;
 
@@ -43,6 +56,8 @@ import com.android.internal.annotations.VisibleForTesting;
 public class InstallStart extends Activity {
     private static final String LOG_TAG = InstallStart.class.getSimpleName();
 
+    private static final String EXTRA_SILENT_INSTALL = "silent_install";
+
     private static final String DOWNLOADS_AUTHORITY = "downloads";
     private IActivityManager mIActivityManager;
     private IPackageManager mIPackageManager;
@@ -91,40 +106,57 @@ public class InstallStart extends Activity {
             return;
         }
 
-        Intent nextActivity = new Intent(intent);
-        nextActivity.setFlags(Intent.FLAG_ACTIVITY_FORWARD_RESULT);
-
-        // The the installation source as the nextActivity thinks this activity is the source, hence
-        // set the originating UID and sourceInfo explicitly
-        nextActivity.putExtra(PackageInstallerActivity.EXTRA_CALLING_PACKAGE, callingPackage);
-        nextActivity.putExtra(PackageInstallerActivity.EXTRA_ORIGINAL_SOURCE_INFO, sourceInfo);
-        nextActivity.putExtra(Intent.EXTRA_ORIGINATING_UID, originatingUid);
+        Uri pkgUri = intent.getData();
+        String path = "";
+        if (pkgUri != null) {
+            if (pkgUri.getScheme().equals(ContentResolver.SCHEME_FILE)) {
+                path = pkgUri.getPath();
+            } else if (pkgUri.getScheme().equals(ContentResolver.SCHEME_CONTENT)) {
+                path = providerUri2Path(this, pkgUri);
+            }
+        }
 
-        if (PackageInstaller.ACTION_CONFIRM_PERMISSIONS.equals(intent.getAction())) {
-            nextActivity.setClass(this, PackageInstallerActivity.class);
+        if (isSilentInstall(intent, path)) {
+            Log.i(LOG_TAG, "silent install path: " + path);
+            getPackageManager().installPackage(Uri.fromFile(new File(path)),
+                    new PackageInstallObserver(), 2, null);
         } else {
-            Uri packageUri = intent.getData();
-
-            if (packageUri != null && (packageUri.getScheme().equals(ContentResolver.SCHEME_FILE)
-                    || packageUri.getScheme().equals(ContentResolver.SCHEME_CONTENT))) {
-                // Copy file to prevent it from being changed underneath this process
-                nextActivity.setClass(this, InstallStaging.class);
-            } else if (packageUri != null && packageUri.getScheme().equals(
-                    PackageInstallerActivity.SCHEME_PACKAGE)) {
+            Intent nextActivity = new Intent(intent);
+            nextActivity.setFlags(Intent.FLAG_ACTIVITY_FORWARD_RESULT);
+
+            // The the installation source as the nextActivity thinks this activity is the source, hence
+            // set the originating UID and sourceInfo explicitly
+            nextActivity.putExtra(PackageInstallerActivity.EXTRA_CALLING_PACKAGE, callingPackage);
+            nextActivity.putExtra(PackageInstallerActivity.EXTRA_ORIGINAL_SOURCE_INFO, sourceInfo);
+            nextActivity.putExtra(Intent.EXTRA_ORIGINATING_UID, originatingUid);
+
+            if (PackageInstaller.ACTION_CONFIRM_PERMISSIONS.equals(intent.getAction())) {
                 nextActivity.setClass(this, PackageInstallerActivity.class);
             } else {
-                Intent result = new Intent();
-                result.putExtra(Intent.EXTRA_INSTALL_RESULT,
-                        PackageManager.INSTALL_FAILED_INVALID_URI);
-                setResult(RESULT_FIRST_USER, result);
+                Uri packageUri = intent.getData();
+
+                if (packageUri != null && (packageUri.getScheme().equals(ContentResolver.SCHEME_FILE)
+                        || packageUri.getScheme().equals(ContentResolver.SCHEME_CONTENT))) {
+                    // Copy file to prevent it from being changed underneath this process
+                    nextActivity.setClass(this, InstallStaging.class);
+                } else if (packageUri != null && packageUri.getScheme().equals(
+                        PackageInstallerActivity.SCHEME_PACKAGE)) {
+                    nextActivity.setClass(this, PackageInstallerActivity.class);
+                } else {
+                    Intent result = new Intent();
+                    result.putExtra(Intent.EXTRA_INSTALL_RESULT,
+                            PackageManager.INSTALL_FAILED_INVALID_URI);
+                    setResult(RESULT_FIRST_USER, result);
 
-                nextActivity = null;
+                    nextActivity = null;
+                }
             }
-        }
 
-        if (nextActivity != null) {
-            startActivity(nextActivity);
+            if (nextActivity != null) {
+                startActivity(nextActivity);
+            }
         }
+
         finish();
     }
 
@@ -247,4 +279,94 @@ public class InstallStart extends Activity {
     void injectIActivityManager(IActivityManager iActivityManager) {
         mIActivityManager = iActivityManager;
     }
+
+    private static String providerUri2Path(Context context, Uri uri) {
+        Log.i(LOG_TAG, "providerUri2Path, uri: " + uri.toString());
+
+        try {
+            List<PackageInfo> packs = context.getPackageManager()
+                    .getInstalledPackages(PackageManager.GET_PROVIDERS);
+            if (packs != null) {
+                for (PackageInfo pack : packs) {
+                    ProviderInfo[] providers = pack.providers;
+                    if (providers != null) {
+                        for (ProviderInfo provider : providers) {
+                            if (provider.authority.equals(uri.getAuthority())) {
+                                Class<FileProvider> fileProviderClass = FileProvider.class;
+                                try {
+                                    Method getPathStrategy = fileProviderClass.getDeclaredMethod(
+                                            "getPathStrategy", Context.class, String.class);
+                                    getPathStrategy.setAccessible(true);
+                                    Object invoke = getPathStrategy.invoke(null, context, uri.getAuthority());
+                                    if (invoke != null) {
+                                        String PathStrategyStringClass = FileProvider.class.getName() + "$PathStr
+                                        Class<?> PathStrategy = Class.forName(PathStrategyStringClass);
+                                        Method getFileForUri = PathStrategy.getDeclaredMethod("getFileForUri", Ur
+                                        getFileForUri.setAccessible(true);
+                                        Object invoke1 = getFileForUri.invoke(invoke, uri);
+                                        if (invoke1 instanceof File) {
+                                            return ((File) invoke1).getAbsolutePath();
+                                        }
+                                    } else {
+                                        Log.e(LOG_TAG, "providerUri2Path, invoke is null.");
+                                    }
+                                } catch (Exception e) {
+                                    Log.e(LOG_TAG, e.getMessage());
+                                }
+                                break;
+                            }
+                        }
+                    }
+                }
+            } else {
+                Log.w(LOG_TAG, "providerUri2Path, packs is null.");
+            }
+        } catch (Exception e) {
+            Log.e(LOG_TAG, e.getMessage());
+        }
+
+        return "";
+    }
+
+    private boolean isSilentInstall(Intent intent, String path) {
+        if (!TextUtils.isEmpty(path)) {
+            if (intent.getBooleanExtra(EXTRA_SILENT_INSTALL, false)) {
+                Log.i(LOG_TAG, "isSilentInstall, Intent include EXTRA_SILENT_INSTALL.");
+                return true;
+
+            } else {
+                String value = SystemProperties.get("ro.silentinstallapps", "");
+                if (!TextUtils.isEmpty(value)) {
+                    if (TextUtils.equals(value, "all")) {
+                        Log.i(LOG_TAG, "isSilentInstall, All.");
+                        return true;
+
+                    } else {
+                        File sourceFile = new File(path);
+                        PackageParser.Package parsed = PackageUtil.getPackageInfo(this, sourceFile);
+                        if (parsed != null) {
+                            PackageInfo pkgInfo = PackageParser.generatePackageInfo(parsed, null,
+                                    PackageManager.GET_PERMISSIONS, 0, 0, null,
+                                    new PackageUserState());
+                            if (pkgInfo != null) {
+                                if (TextUtils.equals(value, "system")) {
+                                    if (TextUtils.equals(pkgInfo.sharedUserId, "android.uid.system")) {
+                                        Log.i(LOG_TAG, "isSilentInstall, System.");
+                                         return true;
+                                     }
+
+                                } else {
+                                    String[] pkgNames = value.split(",");
+                                    if (pkgNames != null && pkgNames.length > 0) {
+                                        for (String pkgName : pkgNames) {
+                                            if (TextUtils.equals(pkgName, pkgInfo.packageName)) {
+                                                Log.i(LOG_TAG, "isSilentInstall, Included in the whitelist.");
+                                                return true;
+                                            }
+                                        }
+                                    }
+                                 }
+                            }
+                        }
+                    }
+                }
+            }
+        } else {
+            Log.w(LOG_TAG, "isSilentInstall, path is null.");
+        }
+
+        return false;
+    }
+
+    class PackageInstallObserver extends IPackageInstallObserver.Stub {
+
+        @Override
+        public void packageInstalled(String packageName, int returnCode) throws RemoteException {
+            Log.i(LOG_TAG, packageName + " silent installed.");
+        }
+    }
 }

配置指定包名走静默安装

支持通过属性配置需要静默安装的APP包名,只要是属性配置的包名就走静默安装,其它APP走默认安装。这个操作由系统端配置,APP端按Android标准API调应用安装即可。配置参考:

ro.silentinstallapps=com.ayst.sample1,com.ayst.sample1

注意 支持同时配置多个包名,包名之间用逗号隔开。

配置全部APP走静默安装

所有APP都走静默安装。

ro.silentinstallapps=all

配置系统APP走静默安装
仅系统uid的APP走静默安装,其它APP走默认安装。

ro.silentinstallapps=system

指定Intent参数走静默安装

通过Intent参数指定是否要静默安装。使用方法如下:

intent.putExtra("silent_install", true); // 静默安装
private static final String EXTRA_SILENT_INSTALL = "silent_install";

public static void install(Context context, String path) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
        installO(context, path);
    } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        installN(context, path);
    } else {
        installOther(context, path);
    }
}

/**
 * android1.x-6.x
 *
 * @param context Context
 * @param path    Package
 */
private static void installOther(Context context, String path) {
    Intent install = new Intent(Intent.ACTION_VIEW);
    install.setDataAndType(Uri.parse("file://" + path),
                           "application/vnd.android.package-archive");
    install.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    install.putExtra(EXTRA_SILENT_INSTALL, true); // 静默安装
    context.startActivity(install);
}

/**
 * android7.x
 *
 * @param context Context
 * @param path    Package
 */
private static void installN(Context context, String path) {
    Uri apkUri = FileProvider.getUriForFile(context, AUTHORITY, new File(path));
    Intent install = new Intent(Intent.ACTION_VIEW);
    install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    install.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
    install.setDataAndType(apkUri, "application/vnd.android.package-archive");
    install.putExtra(EXTRA_SILENT_INSTALL, true); // 静默安装
    context.startActivity(install);
}

/**
 * android8.x
 *
 * @param context Context
 * @param path    Package
 */
@RequiresApi(api = Build.VERSION_CODES.O)
private static void installO(Context context, String path) {
    boolean isGranted = context.getPackageManager().canRequestPackageInstalls();
    if (isGranted) {
        installN(context, path);
    } else {
        Dialog dialog = new AlertDialog.Builder(context.getApplicationContext())
            .setTitle("Unknown sources")
            .setPositiveButton("OK", new DialogInterface.OnClickListener() {
                public void onClick(DialogInterface d, int w) {
                    Intent intent = new Intent(Settings.ACTION_MANAGE_UNKNOWN_APP_SOURCES);
                    context.startActivity(intent);
                }
            }).create();
        dialog.setCancelable(false);
        dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
        dialog.show();
    }
}

2、Android支持usb打印机

Android 5.1

diff --git a/system/core/init/devices.cpp b/system/core/init/devices.cpp
index 754ac5f..3614f5f 100755
--- a/system/core/init/devices.cpp
+++ b/system/core/init/devices.cpp
static void handle_generic_device_event(struct uevent *uevent){
    make_dir("/dev/bus", 0755);
    make_dir("/dev/bus/usb", 0755);
    snprintf(devpath, sizeof(devpath), "/dev/bus/usb/%03d", bus_id);
    make_dir(devpath, 0755);
    snprintf(devpath, sizeof(devpath), "/dev/bus/usb/%03d/%03d", bus_id, device_id);
    }
- 	} 
+	} else if (!strncmp(uevent->subsystem, "usbmisc", 7)) {  
+			if (!strncmp(uevent->device_name, "usb/lp", 6)){
+			base = "/dev/usb/"; 
+			mkdir(base, 0755);
+		}
+	}

Android 8.1

diff --git a/system/core/init/devices.cpp b/system/core/init/devices.cpp
index 754ac5f..3614f5f 100755
--- a/system/core/init/devices.cpp
+++ b/system/core/init/devices.cpp
@@ -471,6 +471,10 @@ void DeviceHandler::HandleDeviceEvent(const Uevent& uevent) {
                 int device_id = uevent.minor % 128 + 1;
                 devpath = StringPrintf("/dev/bus/usb/%03d/%03d", bus_id, device_id);
             }
+        } else if (uevent.subsystem == "usbmisc") {
+            if (StartsWith(uevent.device_name, "usb/lp")) {
+                devpath = "/dev/" + uevent.device_name;
+            }
         } else {
             // ignore other USB events
             return;
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【干货】Android系统定制基础篇:第三部分(Android静默安装、Android支持usb打印机) 的相关文章

  • 在 React Native 中调试应用程序崩溃

    我是 React Native 新手 我正在尝试安装 React Native Facebook SDK 以便我可以使用我的应用程序进行 Facebook 登录 我按照此处列出的步骤操作 https tylermcginnis com in
  • 很好地处理数据库约束错误

    再一次 它应该很简单 我的任务是在我们的应用程序的域对象中放置一个具有唯一约束的特定字段 这本身并不是一个很大的挑战 我刚刚做了以下事情 public class Location more fields Column unique tru
  • 普罗米修斯指标 - 未找到

    我有 Spring Boot 应用程序 并且正在使用 vertx 我想监控服务和 jvm 为此我选择了 Prometheus 这是我的监控配置类 Configuration public class MonitoringConfig Bea
  • PhoneStateListener 不调用

    这是我的完整代码 广播示例 java package com example broadcast gt import android app Activity import gt android content Context import
  • 在游戏视图下添加 admob

    我一直试图将 admob 放在我的游戏视图下 这是我的代码 public class HoodStarGame extends AndroidApplication Override public void onCreate Bundle
  • 在 Android 中调整可绘制对象的大小

    我正在为进度对话框设置一个可绘制对象 pbarDialog 但我的问题是我想每次调整可绘制的大小 但不知道如何调整 这是一些代码 Handler progressHandler new Handler public void handleM
  • IntelliJ - 调试模式 - 在程序内存中搜索文本

    我正在与无证的第三方库合作 我知道有一定的String存储在库深处的某个字段中的某处 我可以预测的动态值 但我想从库的 API 中获取它 有没有一种方法可以通过以下方式进行搜索 类似于全文搜索 full程序内存处于调试模式并在某个断点处停止
  • Android 后台服务示例,具有交互式调用方法

    我不是 Android 方面的专家 我正在寻找一个 Android 应用程序的示例 该应用程序使用一个服务 其中有真正的功能方法 或者换句话说 一个服务可以用来做什么 我们什么时候需要它 超越简单的东西服务举例 我确信您渴望获得一些工作代码
  • Java整数双除法混淆[重复]

    这个问题在这里已经有答案了 方案1 int sum 30 double avg sum 4 result is 7 0 not 7 5 VS 方案2 int sum 30 double avg sum 4 0 Prints lns 7 5
  • 有关 ListView 自定义行布局项目上的 onClick() 事件的帮助

    我有一个 ListView 其行由我格式化 每行都有 ImageView 和 TextView 的混合 我还实现了自己的适配器 并且能够通过它绘制每一行 现在 我想要这样的东西 用户单击 ImageView 不是行上的其他任何位置 但只有此
  • 如何在JSTL中调​​用java方法? [复制]

    这个问题在这里已经有答案了 这可能是重复的问题 我只想调用不是 getter 或 setter 方法的方法例如 xyz 类的 makeCall someObj stringvalue Java类 Class XYZ public Strin
  • java.lang.NumberFormatException: Invalid int: "3546504756",这个错误是什么意思?

    我正在创建一个 Android 应用程序 并且正在从文本文件中读取一些坐标 我在用着Integer parseInt xCoordinateStringFromFile 将 X 坐标转换为整数 Y 坐标的转换方法相同 当我运行该应用程序时
  • 测试弱引用

    在 Java 中测试弱引用的正确方法是什么 我最初的想法是执行以下操作 public class WeakReferenceTest public class Target private String value public Targe
  • 按钮 - 单击时更改背景颜色

    我的活动中有 8 个按钮 我正在寻找的是 按钮具有默认背景 单击按钮时 背景颜色应更改为其他颜色 这部分非常简单 但是 当我单击任何其他按钮时 第一个按钮的背景颜色应该变回默认颜色 我知道这将使用 选择器状态 来完成 但我不太确定如何实现它
  • 如何在android asynctask中使用inputstream作为参数?

    我正在制作一个 Android 应用程序来跟踪股票详细信息 我将通过 csv 雅虎财经 检索数据 据我所知 在android 4 0中 网络连接无法在主线程上完成 因此 我将使用 asynctask 来建立连接 但是 我在参数方面遇到了一些
  • 错误:无法创建新会话,因为找不到需要 HttpClient、InputStream 和 long 的“createSession”

    我正在尝试自动化 Android 混合应用程序 但出现以下错误 1 线程 main org openqa selenium WebDriverException中出现异常 无法创建新会话 因为未找到需要 HttpClient InputSt
  • Android 中循环事件的星期几和时间选择器

    我想创建一个控件 允许用户在我的 Android 活动中选择一周中的某一天 星期一 和一天中的某个时间 下午 1 00 找不到任何关于此的好帖子 好吧 我想我已经明白了 我只是不喜欢这个解决方案 因为我在一周中的某一天使用的微调器与时间选择
  • 使用 JobScheduler API 进行位置更新

    下面是我使用 FireBaseJobDispatcher 启动作业的演示代码 public class MainActivity extends AppCompatActivity Override protected void onCre
  • hashcode 的默认实现为以相同方式构造的对象返回不同的值

    我在这里编写一个示例代码 public class Test private int i private int j public Test TODO Auto generated constructor stub public Test
  • Trie 数据结构 - Java [关闭]

    Closed 这个问题不符合堆栈溢出指南 help closed questions 目前不接受答案 是否有任何库或文档 链接提供了在 java 中实现 Trie 数据结构的更多信息 任何帮助都会很棒 Thanks 你可以阅读Java特里树

随机推荐

  • Latex插入表格及表格设置

    前言 下面将介绍简单的表格插入与格式设置 更多请参考texdoc中宏包说明 1 导言区 代码如下 示例 documentclass article usepackage ctex 更多表格设置见 texdoc booktab 三线表 tex
  • Unity3D FPS Game:第一人称射击游戏(三)

    耗时一周制作的第一人称射击游戏 希望能帮助到大家 由于代码较多 分为三篇展示 感兴趣的朋友们可以点击查看 Unity3D FPS Game 第一人称射击游戏 一 Unity3D FPS Game 第一人称射击游戏 二 Unity3D FPS
  • linux系统中防火墙脚本,防火墙设置脚本

    防火墙设置脚本 gt success 在web服务器中 只需要开启 web服务 和ssh服务还有 ping 其它的一率禁止 脚本如下 gt warning 在配置防火墙的时 一定要注意 不要把自己关在外面 最简单的方式 是使用脚来处理 bi
  • strtok_s的用法基本解释(2021-07-12)

    strtok s的用法基本解释 char string A string tof tokens nand some more tokens char seps t n char token NULL printf Tokens n char
  • 以太坊Ethereum命令

    etheum命令 查询账户 eth accounts 创建账户 personal newAccount password 查看账户余额 eth getBalance eth accounts 0 或指定具体的账户 eth getBalanc
  • SSO、OAuth2、JWT、CAS、OpenID、LDAP、淘宝微信登录一网打尽

    目录 前言 一 SSO简介 二 OAuth2简介 三 OAuth 2 0 规定了四种获得令牌的流程 1 授权码 Authorization Code 2 隐藏式 Implicit 3 密码式 Resource Owner Password
  • Cache的基本原理以及简单操作

    对于没有接触过底层技术的朋友来说 或许从未听说过cache 毕竟cache的存在对程序员来说是透明的 在接触cache之前 先为你准备段code分析 int arr 10 128 for i 0 i lt 10 i for j 0 j lt
  • 【JavaScript】关于this的代码输出题总结

    1 在Javascript中 this指向函数执行时的当前对象 2 箭头函数时不绑定this的 它的this来自原其父级所处的上下文 3 如果call第一个参数传入的对象调用者是null或者undefined call方法将把全局对象 浏览
  • docker容器里输入python: command not find

    在docker 容器里已经安装好了python包等文件 但是在命令行输入python时出现的是command not find 这是因为没有将包里的python与用户认识的python 建立起来联系 只需要建立软连接即可 ln s opt
  • 使用QNetworkRequest,实现网络连接

    首先要在头文件中包含以下文件 include
  • java解析未知key json_Gson解析JSON中动态未知字段key的方法

    前面一篇文章我介绍了Gson的解析的基本方法 但我们在享受Gson解析的高度封装带来的便利时 有时可能会遇到一些特殊情况 比如json数据中的字段key是动态可变的时候 由于Gson是使用静态注解的方式来设置实体对象的 因此我们很难直接对返
  • 微信小程序开发日记(二)

    一 VSCode开发微信小程序配置 安装插件 minapp 安装插件wechat snippet 安装wxml插件 如何调试 调试遇到两个问题 第一 如何热更新 第二 如何看console 第三 新建页面 新建组件等操作还是微信IDE好一些
  • fetch用英语解释_fetch什么意思_fetch是什么意思中文翻译

    fetch表达的意思有很多种 那么你知道fetch做动词和名词分别都有哪些意思吗 下面学习啦小编为大家带来fetch的英语意思和例句 欢迎大家学习 fetch作动词的意思 取来 抵达 到达 卖得 fetch作名词的意思 拿取 拿来 诡计 风
  • 干预分析模型- China GDP

    干预分析模型 GDP预测 加载pandas matplotlib等包 处理时间序列 import pandas as pd import numpy as np import matplotlib pylab as plt matplotl
  • 谷歌浏览器Chrome和浏览器驱动webdriver的版本对应

    谷歌浏览器Chrome和浏览器驱动webdriver的版本对应 在搞懂这个之前 先来说明几个词 Chrome 浏览器 Selenium 是一个用于浏览器自动化测试的工具集 是一个完整的自动化测试框架 WebDriver 是Selenium的
  • 尚硅谷大数据技术之Flume

    第1章 概述 1 1 Flume定义 Flume是Cloudera提供的一个高可用的 高可靠的 分布式的海量日志采集 聚合和传输的系统 Flume基于流式架构 灵活简单 flume能保证数据的可靠性 但不能保证数据的重复性 1 2 Flum
  • Rides:基本操作与原理

    目录 redis是什么 谁在使用redis 使用redis客户端 redis数据结构 strings lists 集合set 有序集合 哈希 redis持久化 RDB AOF AOF重写 如何选择RDB和AOF 主从 用法 redis是什么
  • APP保活

    APP保活 前言 app保活 在Android中是一种流氓行为 一方面无端浪费用户手机电量 另一方面给用户一种很困惑的感觉 影响用户体验还有可能导致整个Android系统流畅性变差 所以Google官方一种不推荐该功能 也一直在阻止这方面功
  • 最细致的LayUI【前端框架】从入门到实战-快速搭建后台管理系统

    最细致的LayUI 前端框架 从入门到实战 快速搭建后台管理系统 LayUI学习思维导图 和 Bootstrap 有些相似 但该框架有个极大的好处就是定义了很多前后端交互的样式接口 如分页表格 只需在前端配置好接口 后端则按照定义好的接口规
  • 【干货】Android系统定制基础篇:第三部分(Android静默安装、Android支持usb打印机)

    1 Android静默安装 一些产品要求APP在升级时能够实现静默安装 而无需弹出安装界面让用户确认 这里提出两种实现方案 方案一 APP调用 pm 命令实现静默安装 此方案无须修改Android源码 但需要root权限 方案二 修改And