Android漏洞之战——整体加壳原理和脱壳技巧详解
data:image/s3,"s3://crabby-images/a3bd7/a3bd7545d8d518c7486017ee75b2fa80cbea4a1f" alt=""
本文为看雪论坛优秀文章
看雪论坛作者ID:随风而行aa
一
前言
二
相关介绍
1.Android App启动流程
(1)Android系统启动流程
data:image/s3,"s3://crabby-images/dbc54/dbc54a8411c46d69940febf7c0884c6748234063" alt=""
data:image/s3,"s3://crabby-images/2c9cf/2c9cf599252115c44f27eea86ce1700204a92734" alt=""
加载BootLoader --> 初始化内核 --> 启动init进程 --> init进程fork出Zygote进程 --> Zygote进程fork出SystemServer进程
data:image/s3,"s3://crabby-images/c43fd/c43fd869e6d6ee9ef6ea7c5b4b5a56afe088f828" alt=""
· 系统启动时安装,没有安装界面
· 第三方应用安装,有安装界面,也是我们最熟悉的方式
· ADB命令安装,没有安装界面
· 通过各类应用市场安装,没有安装界面
data:image/s3,"s3://crabby-images/d393b/d393bd4decb2351b759549b205cdc5fd073217d2" alt=""
public static final IPackageManager main(Context context, Installer installer,
boolean factoryTest, boolean onlyCore) {
PackageManagerService m = new PackageManagerService(context, installer, factoryTest, onlyCore);
ServiceManager.addService("package", m);
return m;
}
data:image/s3,"s3://crabby-images/1819c/1819c40788cfb214e585c5e7beb20e12467ebc64" alt=""
(2)App启动流程
data:image/s3,"s3://crabby-images/5fcc1/5fcc16024a40bd8e7ac4871cddd4c22bce8287ad" alt=""
(3)ActivityThread启动流程
data:image/s3,"s3://crabby-images/05b5f/05b5f444d32fd0d32ae42afd5796097556018ade" alt=""
public void handleMessage(Message msg) {
****
case BIND_APPLICATION:
Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "bindApplication");
AppBindData data = (AppBindData)msg.obj;
handleBindApplication(data);
Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
break;
****
}
data:image/s3,"s3://crabby-images/8d2d8/8d2d85a0a0c2c17fbf77bc98d5301c6a628ca83d" alt=""
data:image/s3,"s3://crabby-images/8dd52/8dd520a0651484e7d10413b114566563eb06cecb" alt=""
data:image/s3,"s3://crabby-images/81bf8/81bf84efac59130a228897bc9773c8b5cce8836a" alt=""
① 完成了Application的实例化;
② 并调用Application.attach()函数。
data:image/s3,"s3://crabby-images/7abda/7abdaed74c21793bfba95112b0ba55a57ab9c02f" alt=""
data:image/s3,"s3://crabby-images/2cee2/2cee2117ec2a7bb9760cd28760853e7c7d9ea37a" alt=""
data:image/s3,"s3://crabby-images/dbc54/dbc54a8411c46d69940febf7c0884c6748234063" alt=""
2.整体加壳原理详解
(1)整体加壳原理
data:image/s3,"s3://crabby-images/59b5f/59b5fcd6aa0f1115c1d463bd172763e5b47df87c" alt=""
data:image/s3,"s3://crabby-images/a78ce/a78ce65da7b0335798e76244d2d4388fe72e4e7d" alt=""
data:image/s3,"s3://crabby-images/f6341/f6341526d4c2f78069d77419cb3f4769118abb70" alt=""
data:image/s3,"s3://crabby-images/b0f5f/b0f5f057542a7ea62849ae72fb22b31506581e97" alt=""
① BootClassLoader加载系统核心库。
② PathClassLoader加载APP自身dex。
③ 进入APP自身组件,解析AndroidManifest.xml,然后查找Application代理。
④ 调用声明Application的attachBaseContext()对源程序进行动态加载或解密。
⑤ 调用声明Application的onCreate()对源程序进行动态加载或解密。
⑥ 进入MainActivity中的attachBaseContext(),然后进入onCreate()函数,执行源程序代码。
(2)类加载器的修正
data:image/s3,"s3://crabby-images/660a9/660a9ba9f05707a09a7638a303ff9dba8cfc3835" alt=""
data:image/s3,"s3://crabby-images/efb54/efb54735fe940d6064826c011d4e11ac45b811e8" alt=""
① 获取ActivityThread实例;
② 通过反射获取类加载器;
③ 获取LoadedApk;
④ 获取mClassLoader系统类加载器;
⑤ 替换自定义类加载器为系统类加载器。
public static void replaceClassLoader(Context context,ClassLoader dexClassLoader){
ClassLoader pathClassLoader = MainActivity.class.getClassLoader();
try {
//1.获取ActivityThread实例
Class ActivityThread = pathClassLoader.loadClass("android.app.ActivityThread");
Method currentActivityThread = ActivityThread.getDeclaredMethod("currentActivityThread");
Object activityThreadObj = currentActivityThread.invoke(null);
//2.通过反射获得类加载器
//final ArrayMap<String, WeakReference<LoadedApk>> mPackages = new ArrayMap<>();
Field mPackagesField = ActivityThread.getDeclaredField("mPackages");
mPackagesField.setAccessible(true);
//3.拿到LoadedApk
ArrayMap mPackagesObj = (ArrayMap) mPackagesField.get(activityThreadObj);
String packagename = context.getPackageName();
WeakReference wr = (WeakReference) mPackagesObj.get(packagename);
Object LoadApkObj = wr.get();
//4.拿到mclassLoader
Class LoadedApkClass = pathClassLoader.loadClass("android.app.LoadedApk");
Field mClassLoaderField = LoadedApkClass.getDeclaredField("mClassLoader");
mClassLoaderField.setAccessible(true);
Object mClassLoader =mClassLoaderField.get(LoadApkObj);
Log.e("mClassLoader",mClassLoader.toString());
//5.将系统组件ClassLoader给替换
mClassLoaderField.set(LoadApkObj,dexClassLoader);
}
catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
}
public static void replaceClassLoader(Context context, ClassLoader dexClassLoader){
//将pathClassLoader父节点设置为DexClassLoader
ClassLoader pathClassLoaderobj = context.getClassLoader();
Class<ClassLoader> ClassLoaderClass = ClassLoader.class;
try {
Field parent = ClassLoaderClass.getDeclaredField("parent");
parent.setAccessible(true);
parent.set(pathClassLoaderobj,dexClassLoader);
} catch (NoSuchFieldException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}
三
整体加壳案例实现
源程序
加壳程序
data:image/s3,"s3://crabby-images/92750/92750a0e5b600399fd85bcc633736d26108c886f" alt=""
2.编写壳程序
(1)准备工作
data:image/s3,"s3://crabby-images/70ddd/70ddd4c3f037f0beb44f3448aa29a71dd83fce74" alt=""
data:image/s3,"s3://crabby-images/06cc6/06cc6f89f93db413b90671e085c343a320f20646" alt=""
(2)编写代理类
data:image/s3,"s3://crabby-images/4bf19/4bf197ba29c8c816480cd116776b7afcd913d9dc" alt=""
data:image/s3,"s3://crabby-images/dbe5c/dbe5c0bf677fad3027c607547664065f7aafe47a" alt=""
data:image/s3,"s3://crabby-images/7b182/7b182dc49467df1c5eb7da7eac50ce83b191e5e5" alt=""
(3)动态加载
data:image/s3,"s3://crabby-images/7162c/7162c9a585012868b1edfb0f6046646bd1c17fb4" alt=""
data:image/s3,"s3://crabby-images/5c784/5c7847b8caa3ef3d91d8739ff15984ec7648de26" alt=""
data:image/s3,"s3://crabby-images/1cb02/1cb02b374b29c7fbcfefc9fb95ae125614f6350c" alt=""
四
脱壳点相关概念详解
1.Dex加载流程
data:image/s3,"s3://crabby-images/041c7/041c7399c3e264aa1509d077d813920f26d5329b" alt=""
DexPathList:该类主要用来查找Dex、SO库的路径,并这些路径整体呈一个数组;
Element:根据多路径的分隔符“;”将dexPath转换成File列表,记录所有的dexFile;
DexFile:用来描述Dex文件,Dex的加载以及Class的查找都是由该类调用它的native方法完成的。
/libcore/dalvik/src/main/java/dalvik/system/DexPathList.java
public DexPathList(ClassLoader definingContext, String dexPath,
String librarySearchPath, File optimizedDirectory) {
**********************
this.dexElements = makeDexElements(splitDexPath(dexPath), optimizedDirectory,
suppressedExceptions, definingContext);
**********************
}
private static Element[] makeDexElements(List<File> files, File optimizedDirectory,
List<IOException> suppressedExceptions, ClassLoader loader) {
**********************
DexFile dex = loadDexFile(file, optimizedDirectory, loader, elements);
**********************
}
private static DexFile loadDexFile(File file, File optimizedDirectory, ClassLoader loader,
Element[] elements)
throws IOException {
if (optimizedDirectory == null) {
return new DexFile(file, loader, elements);
} else {
String optimizedPath = optimizedPathFor(file, optimizedDirectory);
return DexFile.loadDex(file.getPath(), optimizedPath, 0, loader, elements);
}
}
static DexFile loadDex(String sourcePathName, String outputPathName,
int flags, ClassLoader loader, DexPathList.Element[] elements) throws IOException {
return new DexFile(sourcePathName, outputPathName, flags, loader, elements);
}
/libcore/dalvik/src/main/java/dalvik/system/DexFile.java
DexFile(String fileName, ClassLoader loader, DexPathList.Element[] elements) throws IOException {
mCookie = openDexFile(fileName, null, 0, loader, elements);
mInternalCookie = mCookie;
mFileName = fileName;
//System.out.println("DEX FILE cookie is " + mCookie + " fileName=" + fileName);
}
private static Object openDexFile(String sourceName, String outputName, int flags,
ClassLoader loader, DexPathList.Element[] elements) throws IOException {
// Use absolute paths to enable the use of relative paths when testing on host.
return openDexFileNative(new File(sourceName).getAbsolutePath(),
(outputName == null)
? null
: new File(outputName).getAbsolutePath(),
flags,
loader,
elements);
}
data:image/s3,"s3://crabby-images/2b78a/2b78aa29fcb6f4b75645b397bc9f5196289170e0" alt=""
OpenDexFilesFromOat()
MakeUpToDate()
GenerateOatFileNoChecks()
Dex2Oat()
data:image/s3,"s3://crabby-images/ac56d/ac56dc4340e2d49b8e7d53e1b836bf97af4ece31" alt=""
data:image/s3,"s3://crabby-images/b59de/b59dea72e452b422d9897956aafc3edbcdf98a57" alt=""
/art/runtime/dex_file.cc
std::unique_ptr<const DexFile> DexFile::OpenFile(int fd,
const std::string& location,
bool verify,
bool verify_checksum,
std::string* error_msg) {
**************************************
std::unique_ptr<DexFile> dex_file = OpenCommon(map->Begin(),
map->Size(),
location,
dex_header->checksum_,
kNoOatDexFile,
verify,
verify_checksum,
error_msg);
**************************************
}
data:image/s3,"s3://crabby-images/e66ee/e66ee0fe28393ed0d0be37681da58f6471786800" alt=""
2.Dex2Oat编译流程
data:image/s3,"s3://crabby-images/86760/86760a98c6f0fa48855b4098c48eeab9c35856ed" alt=""
/art/runtime/exec_utils.cc
bool Exec(std::vector<std::string>& arg_vector, std::string* error_msg) {
int status = ExecAndReturnCode(arg_vector, error_msg);
if (status != 0) {
const std::string command_line(android::base::Join(arg_vector, ' '));
*error_msg = StringPrintf("Failed execv(%s) because non-0 exit status",
command_line.c_str());
return false;
}
return true;
}
data:image/s3,"s3://crabby-images/4cd4b/4cd4b1660d191b0c35982de2a35a7cf696936f62" alt=""
/art/dex2oat/dex2oat.cc
int main(int argc, char** argv) {
int result = static_cast<int>(art::Dex2oat(argc, argv));
if (!art::kIsDebugBuild && (RUNNING_ON_MEMORY_TOOL == 0)) {
_exit(result);
}
return result;
/art/dex2oat/dex2oat.cc
static dex2oat::ReturnCode Dex2oat(int argc, char** argv) {
**************************************
dex2oat::ReturnCode setup_code = dex2oat->Setup();
dex2oat::ReturnCode result;
if (dex2oat->IsImage()) {
result = CompileImage(*dex2oat);
} else {
result = CompileApp(*dex2oat);
}
**************************************
}
data:image/s3,"s3://crabby-images/bccbc/bccbc2a78351f55dfb8758ba9ecc64aee9f37746" alt=""
3.类加载流程
1.隐式加载:
(1)创建类的实例,也就是new一个对象
(2)访问某个类或接口的静态变量,或者对该静态变量赋值
(3)调用类的静态方法
(4)反射Class.forName("android.app.ActivityThread")
(5)初始化一个类的子类(会首先初始化子类的父类)
2.显示加载:
(1)使用LoadClass()加载
(2)使用forName()加载
Class.forName 和 ClassLoader.loadClass加载有何不同:
(1)ClassLoader.loadClass也能加载一个类,但是不会触发类的初始化(也就是说不会对类的静态变量,静态代码块进行初始化操作)
(2)Class.forName这种方式,不但会加载一个类,还会触发类的初始化阶段,也能够为这个类的静态变量,静态代码块进行初始化操作
data:image/s3,"s3://crabby-images/4ae7c/4ae7cce10982a13cef857c03623cad5e283e8660" alt=""
data:image/s3,"s3://crabby-images/59799/597995deaa125c1d28a00f0bb49190cdfb366f2c" alt=""
data:image/s3,"s3://crabby-images/9c6ab/9c6ab73641a2cd2bef635479ed03ffa7c91a9099" alt=""
ConvertJavaArrayToDexFiles对cookie进行了处理
data:image/s3,"s3://crabby-images/26903/26903ab137a2ac2394c8ba742ecb636fe7021610" alt=""
art/runtime/class_linker.cc
mirror::Class* ClassLinker::DefineClass(Thread* self,
const char* descriptor,
size_t hash,
Handle<mirror::ClassLoader> class_loader,
const DexFile& dex_file,
const DexFile::ClassDef& dex_class_def) {
***************
LoadClass(self, *new_dex_file, *new_class_def, klass);
***************
}
art/runtime/class_linker.cc
void ClassLinker::LoadClass(Thread* self,
3120 const DexFile& dex_file,
3121 const DexFile::ClassDef& dex_class_def,
3122 Handle<mirror::Class> klass) {
3123 const uint8_t* class_data = dex_file.GetClassData(dex_class_def);
3124 if (class_data == nullptr) {
3125 return; // no fields or methods - for example a marker interface
3126 }
3127 LoadClassMembers(self, dex_file, class_data, klass);
3128}
art/runtime/class_linker.cc
void ClassLinker::LoadClassMembers(Thread* self,
const DexFile& dex_file,
const uint8_t* class_data,
Handle<mirror::Class> klass) {
***************
LoadMethod(dex_file, it, klass, method);
LinkCode(this, method, oat_class_ptr, class_def_method_index);
***************
}
art/runtime/class_linker.cc
void ClassLinker::LoadMethod(const DexFile& dex_file,
const ClassDataItemIterator& it,
Handle<mirror::Class> klass,
ArtMethod* dst) {
}
data:image/s3,"s3://crabby-images/cbf4a/cbf4aeac840eb31748509bb747ac143829a978ee" alt=""
art/runtime/interpreter/interpreter.cc
static inline JValue Execute(
Thread* self,
const DexFile::CodeItem* code_item,
ShadowFrame& shadow_frame,
JValue result_register,
bool stay_in_interpreter = false) REQUIRES_SHARED(Locks::mutator_lock_){
***************
ArtMethod *method = shadow_frame.GetMethod();
***************
}
4.DexFile详解
data:image/s3,"s3://crabby-images/d1eb5/d1eb50ab382c18a18b6701f52c846585efcbe057" alt=""
data:image/s3,"s3://crabby-images/5770f/5770f8a54606182bd463399a75f68b6dcf9069b0" alt=""
data:image/s3,"s3://crabby-images/bc59e/bc59ecab5fd699bbb806ff13bd75c0ebc24acdab" alt=""
(1)直接查找法
data:image/s3,"s3://crabby-images/f1ebf/f1ebfd29ac31d8cc9d9c2abfacc53c4c02104f03" alt=""
data:image/s3,"s3://crabby-images/f092a/f092ad15eb8da4d3db5f90038f520cc280ad39de" alt=""
(2)间接查找法
getDexFile()
getMethod()
5.ArtMethod详解
data:image/s3,"s3://crabby-images/b2b93/b2b93616fce07575a266efab011474975a7be3a4" alt=""
五
脱壳技术归纳
data:image/s3,"s3://crabby-images/a8a6b/a8a6b45b2391e2980f23d3c7b80d9e10f4e28f98" alt=""
1.现有工具脱壳法
data:image/s3,"s3://crabby-images/dd94c/dd94c7025948d69b535db59d92c7ff02931b762a" alt=""
(1)FRIDA-DEXDump
data:image/s3,"s3://crabby-images/bd3f7/bd3f74a16032faa3b9a04a855ed816f159ddbf86" alt=""
data:image/s3,"s3://crabby-images/2b2e6/2b2e66d137d3d65757262f3d2c147d6f72f20978" alt=""
(2)FDex2
(3)其他工具
data:image/s3,"s3://crabby-images/0d5bd/0d5bdcb0d071ddf293a9233beb58ca3cbc4720b2" alt=""
2.Hook脱壳法
data:image/s3,"s3://crabby-images/c6cf8/c6cf82671c8f13366b5b54f82fbfc4f3be0fc122" alt=""
data:image/s3,"s3://crabby-images/d41c5/d41c5ada2d830f67a30d84181e043f8c8446f189" alt=""
data:image/s3,"s3://crabby-images/e70e1/e70e1c7a9d9d10102f1102306c0409ee843fea30" alt=""
data:image/s3,"s3://crabby-images/b5ad6/b5ad6d3348529f1c5852f408b7eeed50c86fb8d0" alt=""
data:image/s3,"s3://crabby-images/1738a/1738a899021fd260be42648588c21bae0e551120" alt=""
data:image/s3,"s3://crabby-images/c8120/c812075baee85c75ec1b523fbd07c08014e78d61" alt=""
data:image/s3,"s3://crabby-images/3fcd1/3fcd112f8b3809998590f4a28dfa960d02805f7a" alt=""
data:image/s3,"s3://crabby-images/c1e2a/c1e2a18790d27947c13559fa82f2b69077a31a75" alt=""
3.插桩脱壳法
data:image/s3,"s3://crabby-images/217fe/217fee8e5c61157cb7973602cb29ace3d6121ca1" alt=""
//add
char dexfilepath[100]=0;
memset(dexfilepath,0,100);
sprintf(dexfilepath,"%d_%zu_LoadMethod.dex",getpid(),dex_file.Size());
int dexfd = open(dexfilepathm,O_CREAT|O_RDWR,666);
if(dexfd>0){
int result = write(dexfd,dex_file.Begin(),dex_file.Size());
if(result>0){
close(dexfd);
LOG(WARNING)<<"LoadMethod"<<dexfilepath;
}
}
//add
data:image/s3,"s3://crabby-images/89434/89434735cfe598eb772ac6050bf298cbe654ffdd" alt=""
data:image/s3,"s3://crabby-images/9128b/9128b58da6c31f2e5fac2b313fe1e800bbbdd561" alt=""
data:image/s3,"s3://crabby-images/c1e2a/c1e2a18790d27947c13559fa82f2b69077a31a75" alt=""
4.反射脱壳法
核心思路:反射 + mCookie
步骤:
1、找到加固apk的任一class,一般选择主Application或Activity
2、通过该类找到对应的Classloader
3、通过该Classloader找到BaseDexClassLoader
4、通过BaseDexClassLoader找到其字段DexPathList
5、通过DexPathList找到其变量Element数组dexElements
6、迭代该数组,该数组内部包含DexFile结构
7、通过DexFile获取其变量mCookie和mFileName
至此我们已经获取了mCookie
对该mCookie的解释:
#1、4.4以下好像,mCookie对应的是一个int值,该值是指向native层内存中的dexfile的指针
#2、5.0是一个long值,该值指向native层std::vector<const DexFile*>* 指针,注意这里有多个dex,你需要找到你要的
#3、8.0,该值也是一个long型的值,指向底层vector,但是vector下标0是oat文件,从1开始是dex文件
// 至于你手机是那个版本,如果没有落入我上面描述的,你需要自己看看代码
8、根据mCookie对应的值做转换,最终你能找到dexfile内存指针
9、把该指针转换为dexfile结构,通过findClassDef来匹配你所寻找的dex是你要的dex
10、dump写文件
data:image/s3,"s3://crabby-images/886c0/886c0d5aa8e7e5df454ada0d85460d0eb83f0e20" alt=""
data:image/s3,"s3://crabby-images/cbb68/cbb68b9730b2563df7bf45aaed71fb04471fc01a" alt=""
data:image/s3,"s3://crabby-images/6043c/6043c2a1e16625abae3ca63744535a2abd983dd5" alt=""
data:image/s3,"s3://crabby-images/acd57/acd579abdd14c5b1aefbceaca515b4e290ccea46" alt=""
5.动态调试脱壳法
data:image/s3,"s3://crabby-images/456da/456da825afb93e2920aada00c7a521985c260e2e" alt=""
data:image/s3,"s3://crabby-images/8cff3/8cff3824fc3e1f0c756893d3aff00fa31617e862" alt=""
data:image/s3,"s3://crabby-images/addf2/addf22ffa1a381e747035a4604a13c0c58c0e9b8" alt=""
data:image/s3,"s3://crabby-images/99ee4/99ee4fc426e8a635e1ea834b0b25dd148d589b36" alt=""
data:image/s3,"s3://crabby-images/04609/046097227c99ea98f92b99158d59b14442f987e4" alt=""
data:image/s3,"s3://crabby-images/a480a/a480a5b7737f45ea3c1472af7768bcb96756b116" alt=""
data:image/s3,"s3://crabby-images/3761c/3761c1f0b9556c85425d644f57b2759d014cbac8" alt=""
static main(void){
auto fp, begin, end, dexbyte;
fp = fopen("d:\\dump.dex", "wb+");
begin = 0x76FCD93020;
end = begin + 0x7EEC5600;
for ( dexbyte = begin; dexbyte<end;dexbyte++)
{
fputc(Byte(dexbyte), fp);
}
}
data:image/s3,"s3://crabby-images/908c7/908c752d1c81968ca29a88df797d479320010683" alt=""
data:image/s3,"s3://crabby-images/1d3f0/1d3f0e3a4399cc241a8fdfd9133de3bc1ea93dce" alt=""
data:image/s3,"s3://crabby-images/f8a21/f8a215b997c41e04627d25aa183929b8749eb6b8" alt=""
data:image/s3,"s3://crabby-images/95737/95737f7e00dfe8b8fec8119e1a5a00e0bc22ea12" alt=""
6.特殊API脱壳法
data:image/s3,"s3://crabby-images/269b9/269b9bcea811f957aeea72807513351db541d103" alt=""
data:image/s3,"s3://crabby-images/113e1/113e11578484bb148fe6d63538d99901f63dcd40" alt=""
data:image/s3,"s3://crabby-images/92721/927213026de60f49a133f7038d609da8d272f723" alt=""
data:image/s3,"s3://crabby-images/1a05b/1a05b7820275d18fd7a35089d8a8a6a8554d8aab" alt=""
data:image/s3,"s3://crabby-images/8ecb1/8ecb109aa0f0ec5efb89be8c4755b48645ea19d6" alt=""
data:image/s3,"s3://crabby-images/b3823/b382331a818434885d07a42ecbaee6ffb41739ac" alt=""
data:image/s3,"s3://crabby-images/f8e24/f8e24b6707fbcbd881d3b2b95226047ad28d1da8" alt=""
data:image/s3,"s3://crabby-images/66105/6610537b0653193fda0d1e9d504d7fc471b26fd4" alt=""
data:image/s3,"s3://crabby-images/192d0/192d01bb6f98640351f77da1468b7fdb43148043" alt=""
data:image/s3,"s3://crabby-images/4d028/4d028bdb8ac1b70d37f1688e6ebf8df2fba24994" alt=""
六
实验总结
参考文献
https://bbs.pediy.com/thread-252630.htm#msg_header_h2_4
https://bbs.pediy.com/thread-254555.htm#msg_header_h2_4
https://www.anquanke.com/post/id/221905?display=mobile
https://www.qj301.com/news/317.html
看雪ID:随风而行aa
https://bbs.pediy.com/user-home-905443.htm
# 往期推荐
data:image/s3,"s3://crabby-images/4bdce/4bdcef414a5579619c6ee7fd008091406293f28d" alt=""
data:image/s3,"s3://crabby-images/5dad0/5dad0742d87bde90311bd67b857093694ffa55e4" alt=""
球分享
data:image/s3,"s3://crabby-images/5dad0/5dad0742d87bde90311bd67b857093694ffa55e4" alt=""
球点赞
data:image/s3,"s3://crabby-images/5dad0/5dad0742d87bde90311bd67b857093694ffa55e4" alt=""
球在看
data:image/s3,"s3://crabby-images/42688/42688d648ccd96823d735634e7603c78899ab6a9" alt=""
点击“阅读原文”,了解更多!
[广告]赞助链接:
关注数据与安全,洞悉企业级服务市场:https://www.ijiandao.com/
让资讯触达的更精准有趣:https://www.0xu.cn/
data:image/s3,"s3://crabby-images/1019d/1019d51d7b80866d93c96b9071fd9c90b5d6fb8a" alt="公众号"
随时掌握互联网精彩
- ASF基金会披露由腾讯云鼎实验室通报的高危漏洞
- 科大讯飞回应薪酬回溯制度;OpenAI宣布开放API,开发人员可将ChatGPT集成到自己产品;Godot 4.0发布|极客头条
- 高通在MWC巴塞罗那展示领先的Wi-Fi 7发展势头
- 乔布斯离开后的苹果 | 历史上的今天
- C++、Rust 编译一样糟糕?我用 1.7 万行代码试了试
- 在Z|安芯网盾高薪诚招Linux应用层开发以及样本分析工程师
- 人物 | 宋士明:我们都需要抱团取暖
- 库克:苹果收取 30% 佣金很合理!
- 媲美光纤:5G固定无线接入
- 长肥管道(LFT)中TCP的艰难处境与打法
- 高朋满座话未来|专访vivo高级副总裁,首席技术官施玉坚
- 部署了SSL证书浏览器提示“不安全” 怎么办?