Android混淆那些事

前言

作为一个Android开发,大家或多或少都有一些关于混淆的了解(毕竟披个纱布也比裸奔要好的多吧)。混淆的概念虽然容易理解,但相信大多数开发可能还是在网上搜索通用配置后通过C-V大法接入到自己的项目中,这也使得混淆配置比较混乱,缺乏针对性。

来吧,让我们看看怎么才能穿好这件衣服!!

 

混淆的必要性

Java是一种广泛使用的计算机编程语言,拥有跨平台、面向对象、泛型编程的特性。但不同于一般的编译语言或解释型语言,它首先将源代码编译成字节码,再依赖各种不同平台上的虚拟机来解释执行字节码,从而具有“一次编写,到处运行”的跨平台特性。而这些字节码带有许多的语义信息,很容易被反编译成Java源代码。

WTFK!!

那么作为使用Java作为编程语言(别较劲,我知道Kotlin)的Android应用怎么办,不是动不动就被反编译还原啦!!没错,如果你不做些保护措施,那你的APP、SDK在别人眼里就是那么的赤果果。

混淆就是你现在需要的一顶保护伞。

混淆就是将原本正常的项目文件,对其类、方法、字段,重新命名a,b,c…之类的字母,使得处理后的代码在实现相同功能的同时,降低阅读性,从而增加反编译的难度。但是,但是,但是,重要的事情说三遍,混淆只能增加反编译的难度并不能保证觉得的安全,比较你也懂得,总有人的技术比你。。。。。。

 

ProGuard

官网:https://stuff.mit.edu/afs/sipb/project/android/sdk/android-sdk-linux/tools/proguard/docs/index.html#manual/introduction.html

在官网上是这样说的:
ProGuard is a Java class file shrinker, optimizer, obfuscator, and preverifier. The shrinking step detects and removes unused classes, fields, methods, and attributes. The optimization step analyzes and optimizes the bytecode of the methods. The obfuscation step renames the remaining classes, fields, and methods using short meaningless names. These first steps make the code base smaller, more efficient, and harder to reverse-engineer. The final preverification step adds preverification information to the classes, which is required for Java Micro Edition or which improves the start-up time for Java 6.

简单来说就是:
ProGuard 是一款针对于Java class的压缩、优化、混淆以及预校验工具;压缩环节检测并删除未使用的类、字段、方法和属性;优化环节分析和优化方法的字节码;混淆环节使用无意义的短名称重命名剩余的类、字段和方法。这些操作使代码库更小、更高效并且更难进行逆向工程。最后的预验证环节将预验证信息添加到类中,这是 Java Micro Edition 或者 Java 6及高版本所必需的,当然这一环节在Android开发者并不需要。

如流程图所示,ProGuard主要包含四个环节:压缩、优化、混淆、预检验;
在这里插入图片描述

  • 压缩(Shrink):检测并删除未使用的类、字段、方法和属性;
  • 优化(Optimize):优化字节码并删除未使用的指令;
  • 混淆(Obfuscate):使用无意义的短名称重命名剩余的类、字段和方法;
  • 预预检验(Preverify):对 class 文件进行预检验,确保虚拟机加载的 class 文件是安全并且可以执行的;

 

Android使用ProGuard

Android混淆说明:https://developer.android.com/studio/build/shrink-code#keep-code

Android使用ProGuard手册:https://www.guardsquare.com/manual/home

在Android Gradle 插件 3.4.0 或更高版本构建项目时,已不再使用 ProGuard 执行编译时代码优化,而是与 R8 编译器协同工作,完成编译任务。若想不想启用R8则可在gradle.properties中添加如下配置:

android.enableR8=false
android.enableR8.libraries=false

在正式使用ProGuard之前先让我们用几张图来直观的感受一下混淆的效果:

混淆前
在这里插入图片描述

混淆后
在这里插入图片描述

对比结果:

  • 包体积缩减,因为作为示例的SDK本身内容优先,资源文件极少,所以这里看来效果不明显,各位看官可以在自己的项目中看下实际的缩减效果;
  • 包名、类名、方法名替换为无意义的字母,增加理解难度;

既然看过了对比效果,那接下来就让我们撸起袖子开干吧。

 

开启ProGuard

新建module的时候系统会默认在build.gradle中添加以下配置,以开启混淆:

android {
    ...
    buildTypes {
        release {
            //true - 开启混淆;false - 关闭混淆
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}

minifyEnabled:混淆开关;

proguardFiles:混淆配置文件列表;

getDefaultProguardFile:获取系统提供的默认混淆配置文件,文件路径为:\sdk\tools\proguard\proguard-android.txt;系统还提供了“proguard-android-optimize.txt”默认配置,如果在SDK中开启混淆这里建议使用“proguard-android.txt”;

proguard-rules.pro:编译器自动创建的配置文件,无内容,用于编写我们自定义的混淆规则;

小技巧:当针对SDK开启混淆可通过下面的设置将混淆配置直接打进SDK中,避免使用者对于SDK混淆配置的依赖:

android {
	defaultConfig{
	 	//将混淆配置打包进aar
        	consumerProguardFiles 'proguard-rules.pro'
	}
}

 

混淆规则

混淆开启后,编译器将根据配置文件进行处理,而需要我们自定义制定的规则则是需要明确告诉编译器哪些是我们需要保留的(编译器:老哥啊,你不告诉我哪些要留着,我怎么制定哪些是你需要的,我当然一股脑全给干了啊,毕竟我可是一个尽职尽责的编译器!!!);

下面让我们来看看一些关键的规则配置:

 

keep关键字

 

-keep [,modifier,…] class_specification

指定要保留的类和类成员(字段和方法),类不会被移除,但类成员若不声明保留且未被使用的话将会被移除(SDK开发尤其需要注意这一点);

例:

正确配置:

错误配置:

 

-keepclassmembers [,modifier,…] class_specification

指定要保留的类成员,其他类成员将会被混淆或移除,类名将会混淆,若类未被引用也将被移除;

例:

声明保留static属性及public构造方法;
在这里插入图片描述

可以看到,非static的name属性被混淆,test方法未引用被移除;
在这里插入图片描述

 

-keepclasseswithmembers [,modifier,…] class_specification

当指定的类成员都存在时,保留类和类成员,否则将被混淆或移除;

例:
声明保留具有指定构造方法及keepclasseswithmembers()方法的类

满足条件的类Keepclasseswithmembers1被保留,Keepclasseswithmembers2因缺少对应的构造方法不满足要求被移除
在这里插入图片描述

 

-keepnames、-keepclassmembernames、-keepclasseswithmembernames

这是三个规则对应 -keep(-keepclassmembers、-keepclasseswithmembers) allowshrinking class_specification 的缩写,即在原有混淆基础上若类或类成员未被使用则在压缩阶段允许移除;

 

keep规则总结对比

关键字目标对象说明
keep类、类成员保留类和类中的成员,防止它们被混淆或移除;
keepclassmembers类成员保留类成员,防止它们被混淆或移除
keepclasseswithmembers类、类成员指定的类成员都存在时,保留类和类成员,防止它们被混淆或移除
keepnames类、类成员保留类和类中的成员,防止它们被混淆,但当类或成员未被引用时会被移除;
keepclassmembernames类成员保留类成员,防止它们被混淆,但当类成员未被引用时会被移除;
keepclasseswithmembernames类、类成员指定的类成员都存在时,保留类和类成员,防止它们被混淆,但当类未被引用时会被移除;
从表格中我们可以知道每个规则都有自己针对的目标,我们必须根据实际场景来选择对应的关键字进行组合使用;尤其是必须选取合适的入口类进行keep,否则可能导致其他规则配置因为类或类成员未引用而被移除导致混淆配置无效;

 

通配符

通配符意义
<init>匹配任何构造函数
<fields>匹配任何字段
<methods>匹配任何方法
?匹配任何单个字符
%匹配任何基本类型(“ boolean”、“ int”等)或“ void”类型
  •   |	匹配任意长度字符,但是不包括类分隔符“.”
    
  •   |  	 匹配任意长度字符,包括类分隔符“.”
    

*** | 匹配任何类型(原始或非原始、数组或非数组)
… | 匹配任意数量的任意类型的参数

 

其他常见规则

 

-printmapping [filename]

指定生成混淆映射mapping文件;

 

-keepattributes[属性过滤器]

指定要保留的任何可选属性。属性可以指定一个或多个,如:

#保留Annotation不混淆
-keepattributes *Annotation*,InnerClasses
#避免混淆泛型
-keepattributes Signature
#抛出异常时保留代码行号
-keepattributes SourceFile,LineNumberTable

 

-dontwarn [class_filter]

指定不警告未解决的引用和其他问题;
引入三方library时,强烈建议配置相关包名,如:

#比如关闭Twitter sdk的警告,我们可以这样做
-dontwarn com.twitter.sdk.**

 

-allowaccessmodification

允许修改扩大访问修饰符,如default扩大为public,但是并不推荐配置该规则(尤其是SDK),因为这可能导致其他实现场景中的方法冲突;
还记得开头我们说的getDefaultProguardFile获取默认配置文件吗,正式因为这条规则的存在我并不推荐使用“proguard-android-optimize.txt”;

 

-dontshrink

关闭压缩,默认开启;
默认情况下,因为类或类成员未引用时将会被移除,但是程序猿小哥说:我不想管那么多,我只知道这些类和成员不能混淆,其他你(编译器)看着办;
这时候我们就只能加上这个配置了,保留所有类和成员只做Obfuscate(混淆);

 

-repackageclasses [package_name]

指定重新打包所有重命名的类文件,将它们移动到给定的包中;

 

-keepparameternames

保留参数名称和方法类型;

 

混淆产物

每次打包完成后将在 app/build/outputs/mapping/{flavor}/ 目录下生成一些混淆相关的文件;

文件名作用
configuration.txt所有混淆配置的汇总
mapping.txt原始与混淆过的类、方法、字段名称间的转换
resources.txt资源优化记录文件,哪些资源引用了其他资源,哪些资源在使用,哪些资源被移除
seeds.txt未进行混淆的类与成员
usage.txtAPK中移除的代码

 

哪些不应该混淆

  • 自定义控件不混淆;
  • 枚举类不混淆;
  • 第三方库中的类不混淆,需要根据三方的混淆规则进行配置,若无法确定则整个包名都进行保留;
  • 运用了反射的类不混淆;
  • 使用了 Gson 之类的工具要使 JavaBean 类即实体类不被混淆;
  • 有用到 WebView 的 JS 调用也需要保证写的接口方法不混淆;
  • 继承了Serializable接口的类不混淆;
  • Parcelable 的子类和 Creator 静态成员变量不混淆,否则会产生 Android.os.BadParcelableException 异常;
  • 使用的四大组件,自定义的Application* 实体类
  • JNI中调用的类
  • Layout布局使用的View构造函数(自定义控件)、android:onClick等。

 

常见问题

 

包冲突

因为混淆会使用无意义的a、b、c等字母对类、类成员进行替换,这就导致在多包集成的情况存在包名冲突的可能;
这里就需要我们在打包的时候添加 -repackageclasses 规则,将所有文件迁移至自定义的包名下,避免冲突;

 

运行中数据为null

这个没啥好说的,仔细检查下混淆配置是否完全吧,肯定是混淆了不该混淆的类;

 

Unsupported version number [55.0] (maximum 54.0, Java 10)

ProGuard版本不匹配;
在根目录的build.gradle中指定版本

buildscript{
	dependencies{
	classpath 'net.sf.proguard:proguard-gradle:6.1.1'
	}
}

 

通用配置

最后再提供一份通用的配置,这里的配置可能和系统默认混淆配置文件中存在重复(勿喷);

#---------------------------------基本指令区----------------------------------
# 设置混淆的压缩比率 0 ~ 7
-optimizationpasses 5
#混合时不使用大小写混合,混合后的类名为小写,windows下必须使用该选项
-dontusemixedcaseclassnames
#指定不去忽略非公共库的类和成员
-dontskipnonpubliclibraryclassmembers
#生成map文件
-printmapping proguardMapping.txt
##不做预校验,preverify是proguard的四个步骤之一,Android不需要preverify,去掉这一步能够加快混淆速度。
#-dontpreverify
# 混淆采用的算法.
-optimizations !code/simplification/cast,!field/*,!class/merging/*
#避免混淆注解类
-dontwarn android.annotation
-keepattributes *Annotation*
#保留内部类
-keepattributes InnerClasses
#避免混淆泛型
-keepattributes Signature
#抛出异常时保留代码行号
-keepattributes SourceFile,LineNumberTable
# 将.class信息中的类名重新定义为"Proguard"字符串
-renamesourcefileattribute Proguard
#----------------------------------------------------------------------------

#保留枚举类不被混淆
-keepclassmembers enum * {
    public static **[] values();
    public static ** valueOf(java.lang.String);
}

#保留Serializable序列化的类不被混淆
-keepclassmembers class * implements java.io.Serializable {
    static final long serialVersionUID;
    private static final java.io.ObjectStreamField[] serialPersistentFields;
    private void writeObject(java.io.ObjectOutputStream);
    private void readObject(java.io.ObjectInputStream);
    java.lang.Object writeReplace();
    java.lang.Object readResolve();
}
-keep public class * implements java.io.Serializable {*;}

#----------------------Android通用-----------------
# 避免混淆Android基本组件,下面是兼容性比较高的规则
-keep public class * extends android.app.Activity
-keep public class * extends android.app.Fragment
-keep public class * extends android.app.Application
-keep public class * extends android.app.Service
-keep public class * extends android.content.BroadcastReceiver
-keep public class * extends android.content.ContentProvider
-keep public class * extends android.app.backup.BackupAgentHelper
-keep public class * extends android.preference.Preference
-keep public class * extends android.view.View
#-keep public class com.android.vending.licensing.ILicensingService

# 保留support下的所有类及其内部类
-keep class android.support.** {*;}
-keep interface android.support.** {*;}
-keep public class * extends android.support.v4.**
-keep public class * extends android.support.v7.**
-keep public class * extends android.support.annotation.**
-dontwarn android.support.**

# 保留androidx下的所有类及其内部类
-keep class androidx.** {*;}
-keep public class * extends androidx.**
-keep interface androidx.** {*;}
-keep class com.google.android.material.** {*;}
-dontwarn androidx.**
-dontwarn com.google.android.material.**
-dontnote com.google.android.material.**

# 保持Activity中与View相关方法不被混淆
-keepclassmembers class * extends android.app.Activity{
        public void *(android.view.View);
}

# 避免混淆所有native的方法,涉及到C、C++
-keepclasseswithmembernames class * {
        native <methods>;
}

# 避免混淆自定义控件类的get/set方法和构造函数
-keep public class * extends android.view.View{
        *** get*();
        void set*(***);
        public <init>(android.content.Context);
        public <init>(android.content.Context,android.util.AttributeSet);
        public <init>(android.content.Context,android.util.AttributeSet,int);
}
-keepclasseswithmembers class * {
        public <init>(android.content.Context, android.util.AttributeSet);
        public <init>(android.content.Context, android.util.AttributeSet, int);
}

# 避免混淆枚举类
-keepclassmembers enum * {
        public static **[] values();
        public static ** valueOf(java.lang.String);
}

# 避免混淆序列化类
# 不混淆Parcelable和它的实现子类,还有Creator成员变量
-keep class * implements android.os.Parcelable {
        public static final android.os.Parcelable$Creator *;
}

# 不混淆Serializable和它的实现子类、其成员变量
-keep public class * implements java.io.Serializable {*;}
-keepclassmembers class * implements java.io.Serializable {
        static final long serialVersionUID;
        private static final java.io.ObjectStreamField[] serialPersistentFields;
        private void writeObject(java.io.ObjectOutputStream);
        private void readObject(java.io.ObjectInputStream);
        java.lang.Object writeReplace();
        java.lang.Object readResolve();
}

# 资源ID不被混淆
-keep class **.R$* {*;}

# 回调函数事件不能混淆
-keepclassmembers class * {
        void *(**On*Event);
        void *(**On*Listener);
}

# Webview 相关不混淆
-keepclassmembers class fqcn.of.javascript.interface.for.webview {
   public *;
}
-keepclassmembers class * extends android.webkit.WebViewClient {
        public void *(android.webkit.WebView, java.lang.String, android.graphics.Bitmap);
        public boolean *(android.webkit.WebView, java.lang.String);
}
-keepclassmembers class * extends android.webkit.WebViewClient {
        public void *(android.webkit.WebView, java.lang.String);
 }

# 使用GSON、fastjson等框架时,所写的JSON对象类不混淆,否则无法将JSON解析成对应的对象
-keepclassmembers class * {
         public <init>(org.json.JSONObject);
}

#kotlin 相关
-dontwarn kotlin.**
-keep class kotlin.** { *; }
-keep interface kotlin.** { *; }
-keepclassmembers class kotlin.Metadata {
    public <methods>;
}
-keepclasseswithmembers @kotlin.Metadata class * { *; }
-keepclassmembers class **.WhenMappings {
    <fields>;
}
-assumenosideeffects class kotlin.jvm.internal.Intrinsics {
    static void checkParameterIsNotNull(java.lang.Object, java.lang.String);
}

-keep class kotlinx.** { *; }
-keep interface kotlinx.** { *; }
-dontwarn kotlinx.**
-dontnote kotlinx.serialization.SerializationKt

-keep class org.jetbrains.** { *; }
-keep interface org.jetbrains.** { *; }
-dontwarn org.jetbrains.**

#需要log分析问题
#(可选)避免Log打印输出
-assumenosideeffects class android.util.Log {
    public static boolean isLoggable(java.lang.String,int);
    public static *** d(...);
    public static *** e(...);
    public static *** i(...);
    public static *** v(...);
    public static *** println(...);
    public static *** w(...);
    public static *** wtf(...);
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/289419.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

50、实战 - 利用 conv + bn + relu + add 写一个残差结构

上一节介绍了残差结构,还不清楚的同学可以返回上一节继续阅读。 到了这里,一个残差结构需要的算法基本都介绍完了,至少在 Resnet 这种神经网络中的残差结构是这样的。 本节我们做一个实战,基于之前几节中手写的 conv / bn 算法,来搭建一个残差结构。其中,relu 的实现和…

【linux】ufw 的基本使用

碎碎念 所有的云平台的网络流量的进出基本上有三层&#xff0c;首先是虚拟网的流量控制&#xff0c;一般是通过子网访问控制列表来控制vpc也好子网也好的流量出入&#xff0c;其次是安全组控制一层&#xff0c;通过安全组规则控制一类/一组主机&#xff08;指EC2/ECS/VM/CE这些…

SpringBoot整合ElasticSearch实现CRUD操作

本文来说下SpringBoot整合ES实现CRUD操作 文章目录 概述项目搭建ES简单的crud操作保存数据修改数据查看数据删除数据 本文小结 概述 SpringBoot支持两种技术和es交互。一种的jest&#xff0c;还有一种就是SpringData-ElasticSearch。根据引入的依赖不同而选择不同的技术。反正作…

SQL必知必会笔记(5~8章)

第五章 高级数据过滤 本章示例表为transcript成绩表&#xff0c;数据库软件选用SQLITE3&#xff0c;具体如下 1、高级过滤也是通过where条件子句实现&#xff0c;辅以and、or、in、not进行实现 2、and语句&#xff1a;and连接的多个条件为与的关系&#xff0c;例如 SELECT * FR…

洗地机怎么选?哪款洗地机好用?

选择洗地机前&#xff0c;我们需要对自己购买洗地机的需求做一个清洗的判断&#xff0c;吸尘器和扫地机智能解决地面基本的清洁问题&#xff0c;作为新兴的清洁工具洗地机越来越受大家的喜欢&#xff0c;洗地机的品类很多&#xff0c;洗地机到底该买哪款呢?我们先来看看挑选洗…

利用MATLAB绘制折线图

x20:20:140;%x轴上的数据&#xff0c;第一个值代表数据开始&#xff0c;第二个值代表间隔&#xff0c;第三个值代表终止a[0.85, 2.2, 3.45, 2.65, 1.5, 1.9, 1.25]; %a数据y值plot(x,a,-*b); %线性&#xff0c;颜色&#xff0c;标记 axis([0,160,0,4]) %确定x轴与y轴框图大小 …

安全防御之授权和访问控制技术

授权和访问控制技术是安全防御中的重要组成部分&#xff0c;主要用于管理和限制对系统资源&#xff08;如数据、应用程序等&#xff09;的访问。授权控制用户可访问和操作的系统资源&#xff0c;而访问控制技术则负责在授权的基础上&#xff0c;确保只有经过授权的用户才能访问…

K8S集群部署MySql

挂载MySQL数据卷 在k8s集群中挂载MySQL数据卷 需要安装一个NFS。 在主节点安装NFS yum install -y nfs-utils rpcbind 在主节点创建目录 mkdir -p /nfs chmod 777 /nfs 更改归属组与用户 chown -R nfsnobody:nfsnobody /nfs 配置共享目录 echo "/nfs *(insecure,rw,s…

半导体设备系列:半导体制造产能扩张,设备零部件需求旺盛

近年来国内半导体制造产能不断扩张&#xff0c;半导体设备厂商加速成长。我们认为下游发展将拉动上游本地化配套需求&#xff0c;半导体设备零部件迎来高增长阶段。 摘要 半导体设备零部件包含密封圈、EFEM、射频电源、静电吸盘、硅电极、真空泵、气体流量计、喷淋头等产品&a…

如何做好档案数字化前的鉴定工作

要做好档案数字化前的鉴定工作&#xff0c;可以按照以下步骤进行&#xff1a; 1. 确定鉴定目标&#xff1a;明确要鉴定的档案的内容、数量和性质&#xff0c;确定鉴定的范围和目标。 2. 进行档案清点&#xff1a;对档案进行全面清点和登记&#xff0c;包括数量、种类、状况等信…

【稳定检索|投稿优惠】2024年创新设计与经济发展国际学术会议(IACLDED 2024)

2024年创新设计与经济发展国际学术会议(IACLDED 2024) 2024 International Academic Conference on Innovative Design and Economic Development(IACLDED) 一、【会议简介】 2024年创新设计与经济发展国际学术会议(IACLDED 2024)&#xff0c;将于繁华的成都盛大召开。本次会议…

Vue3 使用路由 Router

Vue3 使用路由 Router 之前几篇博文说了一下 vue 的基本语法和 vue 的传参&#xff0c;今天这篇博文稍微说一下 vue3 里面使用路由。 介绍 众所周知&#xff0c;vue 是用来构建单页面应用的前端框架&#xff0c;大于大多数此类型应用来讲&#xff0c;都推荐使用官方支持的 vue…

阿赵UE学习笔记——6、免费资源获取

阿赵UE学习笔记目录 大家好&#xff0c;我是阿赵。   接下来准备要往UE引擎里面放美术资源了。美术资源可以自己做&#xff0c;不过也有一些免费的资源可以供我们使用的&#xff0c;这里介绍一些获得免费美术资源的方法。 一、Quixel 1、Quixel网站下载 Quixel资源库&#…

小H靶场笔记:DC-3

DC-3 January 3, 2024 4:11 PM Tags&#xff1a;Joomla owner&#xff1a;只惠摸鱼 信息收集 探测靶机ip&#xff1a; 192.168.199.133 nmap 扫描端口、 系统版本 漏洞 发现只有80端口开发&#xff0c; 且有cve-2017-8917漏洞存在是Joomla的SQL注入漏洞 Joomla版本为3.7.0…

【Java】接口和抽象类有什么共同点和区别?

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ 【Java】接口和抽象类有什么共同点和区别&…

李沐机器学习系列3---深度学习计算

1 层和块 1.1 定义块 用class表示层&#xff0c;并只需要实现构造函数和前向传播函数 class MLP(nn.Module):# 用模型参数声明层。这里&#xff0c;我们声明两个全连接的层def __init__(self):# 调用MLP的父类Module的构造函数来执行必要的初始化。# 这样&#xff0c;在类实…

流媒体学习之路(WebRTC)——GCC分析(4)

流媒体学习之路(WebRTC)——GCC分析&#xff08;4&#xff09; —— 我正在的github给大家开发一个用于做实验的项目 —— github.com/qw225967/Bifrost目标&#xff1a;可以让大家熟悉各类Qos能力、带宽估计能力&#xff0c;提供每个环节关键参数调节接口并实现一个json全配置…

代码审查那些事

代码审查(code review)是指对源代码进行系统化地审查&#xff0c;是软件开发中的最佳实践之一&#xff0c;代码合并之前必须审查通过才行&#xff0c;可及时发现隐藏问题&#xff0c;提高代码质量。 1 为什么要代码审查 代码审查环节&#xff0c;或者流于形式&#xff0c;或者…

依赖注入实现原理

依赖注入实现原理 一、Autowire 流程图&#xff1a; Autowire的后置处理器类是AutowiredAnnotationBeanPostProcessor&#xff0c;在启动时会执行这里面的方法 步骤一&#xff1a;寻找注入点 在创建一个Bean的过程中&#xff0c;Spring会利用AutowiredAnnotationBeanPostPro…

Go Lang Fiber介绍

利用GoLang Fiber进行高性能Web开发 在不断发展的Web开发世界中&#xff0c;选择合适的框架至关重要。速度、简洁性和强大的功能集是每个开发者都追求的品质。在使用Go构建Web应用时&#xff0c;“Fiber”作为一个强大且轻量级的框架在众多选择中脱颖而出。在这份全面的指南中…