FrameWork层源码的分析(12)-热修复框架AndFix完全解析

主目录见:Android高级进阶知识(这是总目录索引)
框架地址:AndFix
在线源码查看:AndroidXRef

 我们前面讲了插件化的框架,今天我们来讲讲我们的热修复框架,首先在选热修复框架的时候有犹豫过,到底是要讲美团的robust,微信的Tinker,还是阿里的AndFix等等那么这里用一张图来做一个开始:

热修复框架对比

这里是缺少了美团的robust,还有阿里的最新热修复框架Sophix(这个框架也有借鉴了AndFix的思想)的对比,但是我们来想想,热修复最可能出现的时候应该是方法逻辑的修复,而且修复包要尽量小且能即时生效,同时AndFix用C++中的指针思想来做方法的替换,这个代价也是非常小的。在这几个方面来看看,AndFix还是很理想的。

一.AndFix的使用

怎么使用我们其实可以从GitHub的[AndFix]开源地址获取,但是这里还是列举一下:
1.增加dependency依赖

dependencies {
    compile 'com.alipay.euler:andfix:0.5.0@aar'
}

2.初始化PatchManager

patchManager = new PatchManager(context);
patchManager.init(appversion);//current version

3.加载补丁包

patchManager.loadPatch();

框架推荐你要尽可能早地加载补丁包,最好是在应用的初始化阶段例如

Application.onCreate())

4.添加补丁包

patchManager.addPatch(path);//path of the patch file that was downloaded

当一个新的补丁包下载下来,通过addPath方法可以即时生效

5.补丁包生成
AndFix框架提供了一个补丁包生成工具apkpatch,可以到这里下载apkpatch
使用的时候要准备两个apk文件,其中一个是修复过的,然后生成.patch补丁文件:

usage: apkpatch -f <new> -t <old> -o <output> -k <keystore> -p <***> -a <alias> -e <***>
 -a,--alias <alias>     keystore entry alias.
 -e,--epassword <***>   keystore entry password.
 -f,--from <loc>        new Apk file path.
 -k,--keystore <loc>    keystore path.
 -n,--name <name>       patch name.
 -o,--out <dir>         output dir.
 -p,--kpassword <***>   keystore password.
 -t,--to <loc>          old Apk file path.

然后就可以把这个补丁包分发给客户端了。
有时候团队的成员会修复各自的bug,会生成不止一个补丁包,这样我们可以合并这多个补丁包:

usage: apkpatch -m <apatch_path...> -o <output> -k <keystore> -p <***> -a <alias> -e <***>
 -a,--alias <alias>     keystore entry alias.
 -e,--epassword <***>   keystore entry password.
 -k,--keystore <loc>    keystore path.
 -m,--merge <loc...>    path of .apatch files.
 -n,--name <name>       patch name.
 -o,--out <dir>         output dir.
 -p,--kpassword <***>   keystore password.

好啦,使用起来并不是很难,我们接下来就从使用来分析源码了。

二.AndFix框架源码解析

从使用我们知道,我们的程序会首先初始化PatchManager,所以我们看看初始化做了啥:

    public PatchManager(Context context) {
        mContext = context;
        //初始化AndFixManager
        mAndFixManager = new AndFixManager(mContext);
        //初始化存放patch补丁文件的目录
        mPatchDir = new File(mContext.getFilesDir(), DIR);
       //初始化存在Patch类的集合
        mPatchs = new ConcurrentSkipListSet<Patch>();
       //初始化存放类对应的类加载器集合
        mLoaders = new ConcurrentHashMap<String, ClassLoader>();
    }

这里面有个比较重要的类AndFixManager,我们来看看它的构造函数:

public AndFixManager(Context context) {
        mContext = context;
        //兼容性判断,判断AndFix是否适用
        mSupport = Compat.isSupport();
        if (mSupport) {
         //主要是对补丁包的安全检测
            mSecurityChecker = new SecurityChecker(mContext);
         //初始化补丁包的路径
            mOptDir = new File(mContext.getFilesDir(), DIR);
            if (!mOptDir.exists() && !mOptDir.mkdirs()) {// make directory fail
           //如果不定目录不存在则把支持标志设为false
                mSupport = false;
                Log.e(TAG, "opt dir create error.");
            } else if (!mOptDir.isDirectory()) {// not directory
           //如果不是目录则标志设置为false
                mOptDir.delete();
                mSupport = false;
            }
        }
    }

这个类构造函数做的工作很简单,就是Android系统兼容性的检查,然后就是对补丁包安全性和路径的检验。接着我们看看兼容性判断源码:

public static synchronized boolean isSupport() {
        if (isChecked)
            return isSupport;

        isChecked = true;
        // not support alibaba's YunOs
               //写的很清楚,这里不支持阿里的YunOs,且会判断Davilk还是art虚拟机,来注册native方法
        if (!isYunOS() && AndFix.setup() && isSupportSDKVersion()) {
            isSupport = true;
        }
              //是否在黑名单中,默认是返回false
        if (inBlackList()) {
            isSupport = false;
        }

        return isSupport;
    }

判断是不是YunOs系统和支持的Sdk版本这两个方法比较简单,这里不深究,重点来看AndFix.setup()方法:

public static boolean setup() {
        try {
            final String vmVersion = System.getProperty("java.vm.version");
            boolean isArt = vmVersion != null && vmVersion.startsWith("2");
            int apilevel = Build.VERSION.SDK_INT;
            return setup(isArt, apilevel);
        } catch (Exception e) {
            Log.e(TAG, "setup", e);
            return false;
        }
    }

这个方法也是很简单,判断vm的版本是不是以2开头,如果是就是art虚拟机,然后将isArt和api的版本传给了native的setup方法,这个方法在jni目录的AndFix.cpp下面:

static jboolean setup(JNIEnv* env, jclass clazz, jboolean isart,
        jint apilevel) {
    isArt = isart;
    LOGD("vm is: %s , apilevel is: %i", (isArt ? "art" : "dalvik"),
            (int )apilevel);
    if (isArt) {
        return art_setup(env, (int) apilevel);
    } else {
        return dalvik_setup(env, (int) apilevel);
    }
}

可以看到方法里面根据isArt这个标志来调用不同的方法,首先我们看下art_setup方法:

extern jboolean __attribute__ ((visibility ("hidden"))) art_setup(JNIEnv* env,
        int level) {
    apilevel = level;
    return JNI_TRUE;
}

可以看出这个方法就记录了下apilevel然后直接返回了true。接着我们来看看dalvik_setup方法:

extern jboolean __attribute__ ((visibility ("hidden"))) dalvik_setup(
        JNIEnv* env, int apilevel) {
//打开libdvm.so文件
    void* dvm_hand = dlopen("libdvm.so", RTLD_NOW);
    if (dvm_hand) {
 //获取dvmDecodeIndirectRef_fnPtr和dvmThreadSelf_fnPtr两个函数,
 //这两个函数可以通过类对象获取ClassObject结构体
        dvmDecodeIndirectRef_fnPtr = dvm_dlsym(dvm_hand,
                apilevel > 10 ?
                        "_Z20dvmDecodeIndirectRefP6ThreadP8_jobject" :
                        "dvmDecodeIndirectRef");
        if (!dvmDecodeIndirectRef_fnPtr) {
            return JNI_FALSE;
        }
        dvmThreadSelf_fnPtr = dvm_dlsym(dvm_hand,
                apilevel > 10 ? "_Z13dvmThreadSelfv" : "dvmThreadSelf");
        if (!dvmThreadSelf_fnPtr) {
            return JNI_FALSE;
        }
      //这里用jni里面的方法来获取Java层Method对象的getDeclaringClass方法
      //后续会调用该方法获取某个方法所属的类对象
      //因为Java层只传递了Method对象到native层
        jclass clazz = env->FindClass("java/lang/reflect/Method");
        jClassMethod = env->GetMethodID(clazz, "getDeclaringClass",
                        "()Ljava/lang/Class;");

        return JNI_TRUE;
    } else {
        return JNI_FALSE;
    }
}

这里面我们先来看看Android虚拟机初始化方面的知识图:


Android虚拟机初始化

1.跟java虚拟机不同,Android虚拟机执行的是dex文件而不是class文件,同时在android版本4.4开始就推出了art虚拟机,所以现在android的虚拟机分为dalvik虚拟机art虚拟机
2.虚拟机启动的时候会加载两个很重要的动态库文件libdvm.solibart.so,这就是dalvik虚拟机art虚拟机的动态库文件
3.Java在虚拟机环境中执行,每个Java方法都会对应一个底层的函数指针,当Java方法被调用的时候,实质虚拟机会找到这个函数指针然后去执行底层的方法,从而Java方法被执行。

接着我们继续来分析dalvik_setup方法,首先这个方法会判断apilevel是不是大于10来Hook不同的系统函数,在Hook成功了之后,我们就可以调用我们自己的代码进行替换,这就是框架github上面说的:

方法替换

替换方法我们留着后面具体说,我们接着来看PatchManager类中的init方法:

public void init(String appVersion) {
//首先判断补丁路径是否存在或者是不是目录
        if (!mPatchDir.exists() && !mPatchDir.mkdirs()) {// make directory fail
            Log.e(TAG, "patch dir create error.");
            return;
        } else if (!mPatchDir.isDirectory()) {// not directory
            mPatchDir.delete();
            return;
        }
//这里用sp来存储补丁包的配置信息
        SharedPreferences sp = mContext.getSharedPreferences(SP_NAME,
                Context.MODE_PRIVATE);
        String ver = sp.getString(SP_VERSION, null);
//这里首先获取本地补丁包的版本和传入的版本号做对比
        if (ver == null || !ver.equalsIgnoreCase(appVersion)) {
//如果版本不同,说明本地的版本过久则要删除存入新的补丁包版本
            cleanPatch();
            sp.edit().putString(SP_VERSION, appVersion).commit();
        } else {
//相同说明本地有本地补丁包了,则初始化
            initPatchs();
        }
    }

我们看到这里如果本地已经有补丁包且版本未更新,则初始化,我们来看这个initPatchs方法:

private void initPatchs() {
        File[] files = mPatchDir.listFiles();
        for (File file : files) {
            addPatch(file);
        }
    }

这个方法很简单,就是将补丁路径下的所有补丁文件调用addPatch方法添加进去,我们来看看addPatch方法:

private Patch addPatch(File file) {
        Patch patch = null;
        if (file.getName().endsWith(SUFFIX)) {
            try {
                patch = new Patch(file);
                mPatchs.add(patch);
            } catch (IOException e) {
                Log.e(TAG, "addPatch", e);
            }
        }
        return patch;
    }

我们看到这个方法主要就是判断文件是不是以.patch结尾,如果是的话就初始化一个Patch实例,这个是添加补丁包的关键,我们跟进构造函数看下:

public Patch(File file) throws IOException {
        mFile = file;
        init();
    }

@SuppressWarnings("deprecation")
    private void init() throws IOException {
        JarFile jarFile = null;
        InputStream inputStream = null;
        try {
//使用JarFile读取Patch文件
            jarFile = new JarFile(mFile);
//获取META-INF/PATCH.MF文件
            JarEntry entry = jarFile.getJarEntry(ENTRY_NAME);
            inputStream = jarFile.getInputStream(entry);
            Manifest manifest = new Manifest(inputStream);
            Attributes main = manifest.getMainAttributes();
//因为PATCH.MF里面是以键值对的形式存储的,这里是获取键Patch-Name对应的值
            mName = main.getValue(PATCH_NAME);
//这里是获取创建的时间
            mTime = new Date(main.getValue(CREATED_TIME));

            mClassesMap = new HashMap<String, List<String>>();
            Attributes.Name attrName;
            String name;
            List<String> strings;
            for (Iterator<?> it = main.keySet().iterator(); it.hasNext();) {
                attrName = (Attributes.Name) it.next();
                name = attrName.toString();
//判断name的后缀是否是-Classes,并把name对应的值加入到集合中,对应的值就是class类名的列表
                if (name.endsWith(CLASSES)) {
                    strings = Arrays.asList(main.getValue(attrName).split(","));
                    if (name.equalsIgnoreCase(PATCH_CLASSES)) {
                        mClassesMap.put(mName, strings);
                    } else {
//  //为了移除掉"-Classes"的后缀
                        mClassesMap.put(
                                name.trim().substring(0, name.length() - 8),// remove
                                                                            // "-Classes"
                                strings);
                    }
                }
            }
        } finally {
            if (jarFile != null) {
                jarFile.close();
            }
            if (inputStream != null) {
                inputStream.close();
            }
        }

    }

这个方法的逻辑其实就是读取补丁.patch文件,每个修复包apatch文件其实都是一个jarFile文件,然后获得其中META-INF/PATCH.MF文件,PATCH.MF文件中都是键值对的形式,获取key是-Classes的所有的value,这些value就是所有要修复的类,他们是以“,”进行分割的,将它们放入list列表,将其存储到一个集合中mClassesMap,list列表中存储的就是所有要修复的类名。我们看到我们还有一个addPatch方法:

public void addPatch(String path) throws IOException {
        File src = new File(path);
        File dest = new File(mPatchDir, src.getName());
        if(!src.exists()){
            throw new FileNotFoundException(path);
        }
        if (dest.exists()) {
            Log.d(TAG, "patch [" + path + "] has be loaded.");
            return;
        }
        FileUtil.copyFile(src, dest);// copy to patch's directory
        Patch patch = addPatch(dest);
        if (patch != null) {
            loadPatch(patch);
        }
    }

这个方法里面就多了一步将指定的路径下的文件拷贝到补丁包路径下面,这个方法的作用是什么呢?我们知道,我们第一次app下载下来的时候,我们本地是没有补丁包的,只有当需要补丁修复的时候,我们才会从服务器下载补丁包下来,这时候我们可以指定我们下载下来的补丁包的存放路径,然后将它拷贝到补丁包路径,然后进行加载。
接下来,我们来看看方法调用loadPatch()方法:

public void loadPatch() {
        mLoaders.put("*", mContext.getClassLoader());// wildcard
        Set<String> patchNames;
        List<String> classes;
//遍历mPatchs中的所有Patch
        for (Patch patch : mPatchs) {
//遍历补丁包中待修复的所有patch名
            patchNames = patch.getPatchNames();
            for (String patchName : patchNames) {
//获取patch中的所有class
                classes = patch.getClasses(patchName);
//修复bug
                mAndFixManager.fix(patch.getFile(), mContext.getClassLoader(),
                        classes);
            }
        }
    }

我们看到这个方法主要就是最后的fix方法:

public synchronized void fix(String patchPath) {
        fix(new File(patchPath), mContext.getClassLoader(), null);
    }

public synchronized void fix(File file, ClassLoader classLoader,
            List<String> classes) {
        if (!mSupport) {
            return;
        }
//检测patch包的签名,检查是不是安全的
        if (!mSecurityChecker.verifyApk(file)) {// security check fail
            return;
        }

        try {
//判断优化后的文件是否存在,如果存在则检测optfile的安全性
            File optfile = new File(mOptDir, file.getName());
            boolean saveFingerprint = true;
            if (optfile.exists()) {
                // need to verify fingerprint when the optimize file exist,
                // prevent someone attack on jailbreak device with
                // Vulnerability-Parasyte.
                // btw:exaggerated android Vulnerability-Parasyte
                // http://secauo.com/Exaggerated-Android-Vulnerability-Parasyte.html
                if (mSecurityChecker.verifyOpt(optfile)) {
                    saveFingerprint = false;
                } else if (!optfile.delete()) {
                    return;
                }
            }
//加载补丁文件包中的dex文件
            final DexFile dexFile = DexFile.loadDex(file.getAbsolutePath(),
                    optfile.getAbsolutePath(), Context.MODE_PRIVATE);

            if (saveFingerprint) {
                mSecurityChecker.saveOptSig(optfile);
            }
//这里重写了ClassLoader中的findClass方法,并且实例化它
            ClassLoader patchClassLoader = new ClassLoader(classLoader) {
                @Override
                protected Class<?> findClass(String className)
                        throws ClassNotFoundException {
                    Class<?> clazz = dexFile.loadClass(className, this);
                    if (clazz == null
                            && className.startsWith("com.alipay.euler.andfix")) {
                        return Class.forName(className);// annotation’s class
                                                        // not found
                    }
                    if (clazz == null) {
                        throw new ClassNotFoundException(className);
                    }
                    return clazz;
                }
            };
//遍历dex然后加载有bug的类文件
            Enumeration<String> entrys = dexFile.entries();
            Class<?> clazz = null;
            while (entrys.hasMoreElements()) {
                String entry = entrys.nextElement();
                if (classes != null && !classes.contains(entry)) {
                    continue;// skip, not need fix
                }
                clazz = dexFile.loadClass(entry, patchClassLoader);
                if (clazz != null) {
//对有bug的文件进行替换
                    fixClass(clazz, classLoader);
                }
            }
        } catch (IOException e) {
            Log.e(TAG, "pacth", e);
        }
    }

我们看见我们为什么要用DexFile来加载补丁文件呢?这个得从补丁包.patch文件来看:

Patch包信息

我们看到我们的.patch文件实质上里面包含了两个文件,其中一个.dex就是我们要修复的类,META-INF文件如下:
META-INF

这个文件就是我们前面说的PATCH.MF,里面存放补丁包的一些配置信息。上面的方法就是加载dex,然后获取到所有要修复的类,然后进行替换,所以我们接下来看fixClass()方法:

private void fixClass(Class<?> clazz, ClassLoader classLoader) {
        Method[] methods = clazz.getDeclaredMethods();
        MethodReplace methodReplace;
        String clz;
        String meth;
//遍历要修复类中的所有方法
        for (Method method : methods) {
//获取带有MethodReplace注解的方法
            methodReplace = method.getAnnotation(MethodReplace.class);
            if (methodReplace == null)
                continue;
            clz = methodReplace.clazz();
            meth = methodReplace.method();
//获取注解中的class信息和method信息
            if (!isEmpty(clz) && !isEmpty(meth)) {
//然后调用replaceMethod方法进行替换
                replaceMethod(classLoader, clz, meth, method);
            }
        }
    }

我们看到上面方法会获取带有MethodReplace注解的所有方法,这是在补丁包生成的时候添加上去的,然后获取class和method的信息之后,通过调用方法repalceMethod方法进行方法替换:

private void replaceMethod(ClassLoader classLoader, String clz,
            String meth, Method method) {
        try {
            String key = clz + "@" + classLoader.toString();
//判断这个类是否已经被修复了
            Class<?> clazz = mFixedClass.get(key);
//如果class还没有被类加载加载进来,则去加载
            if (clazz == null) {// class not load
                Class<?> clzz = classLoader.loadClass(clz);
                // initialize target class
                clazz = AndFix.initTargetClass(clzz);
            }
            if (clazz != null) {// initialize class OK
                mFixedClass.put(key, clazz);
//根据反射获取到有bug的类的方法
                Method src = clazz.getDeclaredMethod(meth,
                        method.getParameterTypes());
//src是有bug的方法,method是补丁方法
                AndFix.addReplaceMethod(src, method);
            }
        } catch (Exception e) {
            Log.e(TAG, "replaceMethod", e);
        }
    }

我们看到通过注解中的method信息获取到有bug的方法,然后调用AndFix.addReplaceMethod(src, method)进行替换,这里是native来实现的,我们来看看:

static void replaceMethod(JNIEnv* env, jclass clazz, jobject src,
        jobject dest) {
    if (isArt) {
        art_replaceMethod(env, src, dest);
    } else {
        dalvik_replaceMethod(env, src, dest);
    }
}

我们的看到dalvik和art虚拟机的替换方法不同,我们先来看看dalvik虚拟机上面的:

extern void __attribute__ ((visibility ("hidden"))) dalvik_replaceMethod(
        JNIEnv* env, jobject src, jobject dest) {
//这里通过jni获取jobject对象
    jobject clazz = env->CallObjectMethod(dest, jClassMethod);
//我们看到刚才Hook的系统的函数,通过这个系统函数我们可以获取class的ClassObject对象指针
    ClassObject* clz = (ClassObject*) dvmDecodeIndirectRef_fnPtr(
            dvmThreadSelf_fnPtr(), clazz);
//更改状态为类初始化完成的状态
    clz->status = CLASS_INITIALIZED;
//通过java层传递的方法对象,在native层获得他们的结构体
    Method* meth = (Method*) env->FromReflectedMethod(src);
    Method* target = (Method*) env->FromReflectedMethod(dest);
    LOGD("dalvikMethod: %s", meth->name);

//  meth->clazz = target->clazz;
替换method结构体中的accessFlags
    meth->accessFlags |= ACC_PUBLIC;
//替换method结构体中的方法的索引methodIndex
    meth->methodIndex = target->methodIndex;
// 替换method结构体的缓存的JNI参数jniArgInfo
    meth->jniArgInfo = target->jniArgInfo;
    meth->registersSize = target->registersSize;
    meth->outsSize = target->outsSize;
//替换method结构体的insSize
    meth->insSize = target->insSize;
// 替换method结构体的方法原型描述prototype
    meth->prototype = target->prototype;
// 替换method结构体的真实代码insns
    meth->insns = target->insns;
//替换method结构体的真实函数或者JNI桥接函数
    meth->nativeFunc = target->nativeFunc;
}

我们看到这里就是指针指向了修复完的方法的结构体信息,这样就达到了修复的目的,所以说指针还是一个非常有用的东西,这也是c和c++的精髓所在,接着我们来看看art虚拟机的replaceMethod方法:

extern void __attribute__ ((visibility ("hidden"))) art_replaceMethod(
        JNIEnv* env, jobject src, jobject dest) {
    if (apilevel > 23) {
        replace_7_0(env, src, dest);
    } else if (apilevel > 22) {
        replace_6_0(env, src, dest);
    } else if (apilevel > 21) {
        replace_5_1(env, src, dest);
    } else if (apilevel > 19) {
        replace_5_0(env, src, dest);
    }else{
        replace_4_4(env, src, dest);
    }
}

这个地方还是比较蛋疼的,我们看见根据api的不同版本来做了好几个适配,现在8.0出来了,可想而知也可能要做适配,我们先来看下4.4的这个方法:

void replace_4_4(JNIEnv* env, jobject src, jobject dest) {
    art::mirror::ArtMethod* smeth =
            (art::mirror::ArtMethod*) env->FromReflectedMethod(src);

    art::mirror::ArtMethod* dmeth =
            (art::mirror::ArtMethod*) env->FromReflectedMethod(dest);
//替换 ArtMethod的class_loader指针
    dmeth->declaring_class_->class_loader_ =
            smeth->declaring_class_->class_loader_; //for plugin classloader
//替换ArtMethod的clinit_thread_id_指针
    dmeth->declaring_class_->clinit_thread_id_ =
            smeth->declaring_class_->clinit_thread_id_;
//替换ArtMethod的status_指针
    dmeth->declaring_class_->status_ = smeth->declaring_class_->status_-1;
    //for reflection invoke
    reinterpret_cast<art::mirror::Class*>(dmeth->declaring_class_)->super_class_ = 0;
//替换ArtMethod的declaring_class_指针
    smeth->declaring_class_ = dmeth->declaring_class_;
    smeth->dex_cache_initialized_static_storage_ = dmeth->dex_cache_initialized_static_storage_;
//替换ArtMethod的access_flags_指针
    smeth->access_flags_ = dmeth->access_flags_  | 0x0001;
    smeth->dex_cache_resolved_types_ = dmeth->dex_cache_resolved_types_;
    smeth->dex_cache_resolved_methods_ = dmeth->dex_cache_resolved_methods_;
    smeth->dex_cache_strings_ = dmeth->dex_cache_strings_;
    smeth->code_item_offset_ = dmeth->code_item_offset_;
    smeth->core_spill_mask_ = dmeth->core_spill_mask_;
    smeth->fp_spill_mask_ = dmeth->fp_spill_mask_;
    smeth->method_dex_index_ = dmeth->method_dex_index_;
    smeth->mapping_table_ = dmeth->mapping_table_;
//替换ArtMethod的method_index_指针
    smeth->method_index_ = dmeth->method_index_;
    smeth->gc_map_ = dmeth->gc_map_;
    smeth->frame_size_in_bytes_ = dmeth->frame_size_in_bytes_;
    smeth->native_method_ = dmeth->native_method_;
    smeth->vmap_table_ = dmeth->vmap_table_;

    smeth->entry_point_from_compiled_code_ = dmeth->entry_point_from_compiled_code_;
    
    smeth->entry_point_from_interpreter_ = dmeth->entry_point_from_interpreter_;
    
    smeth->method_index_ = dmeth->method_index_;

    LOGD("replace_4_4: %d , %d", smeth->entry_point_from_compiled_code_,
            dmeth->entry_point_from_compiled_code_);

}

我们看到这里要替换的东西还是比较多的,大家可以通过查看虚拟机中的方法结构体来查看,这里也就不详细列出了。

总结:AndFix的源码还是比较好理解,但是想到用这种方法还是非常fashion的有没有,而且需要我们了解你虚拟机相关的知识,这方面我也是有欠缺,可以适当补充了,希望大家一起进步。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容