Gradle之Gradle 的基本使用(一)
【Android 修炼手册】Gradle 篇 -- Gradle 的基本使用
预备知识
- 基本的 android 开发知识
- 了解 Android Studio 基本使用
看完本文可以达到什么程度
- 掌握 gradle 的基本使用
- 了解 gradle 及 android gradle plugin
- 了解 gradle 构建阶段及生命周期回调
- 掌握 Task,Transform 等概念
- 学会自定义 task,自定义 gradle 插件
如果您已经达到上面的程度,那么可以不用再看下文了,直接看最后的总结即可
本文将从下面几个部分进行讲解:
一、gradle 是什么
可以从三个角度来理解
1. gradle 是一个自动化构建工具
gradle 是通过组织一系列 task 来最终完成自动化构建的,所以 task 是 gradle 里最重要的概念
我们以生成一个可用的 apk 为例,整个过程要经过 资源的处理,javac 编译,dex 打包,apk 打包,签名等等步骤,每个步骤就对应到 gradle 里的一个 task
gradle 可以类比做一条流水线,task 可以比作流水线上的机器人,每个机器人负责不同的事情,最终生成完整的构建产物
2. gradle 脚本使用了 groovy 或者 kotlin DSL
gradle 使用 groovy 或者 kotlin 编写,不过目前还是 groovy 居多
那什么是 DSL 呢?DSL 也就是 Domain Specific Language 的简称,是为了解决某一类任务专门设计的计算机语言
DSL 相对应的是 GPL (General-Purpose Language),比如 java
与 GPL 相比起来,DSL 使用简单,定义比较简洁,比起配置文件,DSL 又可以实现语言逻辑
对 gradle 脚本来说,他实现了简洁的定义,又有充分的语言逻辑,以 android {} 为例,这本身是一个函数调用,参数是一个闭包,但是这种定义方式明显要简洁很多
3. gradle 基于 groovy 编写,而 groovy 是基于 jvm 语言
gradle 使用 groovy 编写,groovy 是基于 jvm 的语言,所以本质上是面向对象的语言,面向对象语言的特点就是一切皆对象,所以,在 gradle 里,.gradle 脚本的本质就是类的定义,一些配置项的本质都是方法调用,参数是后面的 {} 闭包
比如 build.gradle 对应 Project 类,buildScript 对应 Project.buildScript 方法
二、gradle 项目分析
关于 gradle 的项目层次,我们新建一个项目看一下,项目地址在 EasyGradle
2.1 settings.gradle
settings.gradle 是负责配置项目的脚本
对应 Settings 类,gradle 构建过程中,会根据 settings.gradle 生成 Settings 的对象
对应的可调用的方法在文档里可以查找
其中几个主要的方法有:
- include(projectPaths)
- includeFlat(projectNames)
- project(projectDir)
一般在项目里见到的引用子模块的方法,就是使用 include,这样引用,子模块位于根项目的下一级
include ':app'
如果想指定子模块的位置,可以使用 project 方法获取 Project 对象,设置其 projectDir 参数
include ':app'
project(':app').projectDir = new File('./app')
2.2 rootproject/build.gradle
build.gradle 负责整体项目的一些配置,对应的是 Project 类
gradle 构建的时候,会根据 build.gradle 生成 Project 对象,所以在 build.gradle 里写的 dsl,其实都是 Project 接口的一些方法,Project 其实是一个接口,真正的实现类是 DefaultProject
build.gradle 里可以调用的方法在 Project 可以查到
其中几个主要方法有:
- buildscript // 配置脚本的 classpath
- allprojects // 配置项目及其子项目
- respositories // 配置仓库地址,后面的依赖都会去这里配置的地址查找
- dependencies // 配置项目的依赖
以 EasyGradle 项目来看
buildscript { // 配置项目的 classpath
repositories { // 项目的仓库地址,会按顺序依次查找
google()
jcenter()
mavenLocal()
}
dependencies { // 项目的依赖
classpath 'com.android.tools.build:gradle:3.0.1'
classpath 'com.zy.plugin:myplugin:0.0.1'
}
} allprojects { // 子项目的配置
repositories {
google()
jcenter()
mavenLocal()
}
}
2.3 module/build.gradle
build.gradle 是子项目的配置,对应的也是 Project 类
子项目和根项目的配置是差不多的,不过在子项目里可以看到有一个明显的区别,就是引用了一个插件 apply plugin "com.android.application",后面的 android dsl 就是 application 插件的 extension,关于 android plugin dsl 可以看 android-gradle-dsl
其中几个主要方法有:
- compileSdkVersion // 指定编译需要的 sdk 版本
- defaultConfig // 指定默认的属性,会运用到所有的 variants 上
- buildTypes // 一些编译属性可以在这里配置,可配置的所有属性在 这里
- productFlavor // 配置项目的 flavor
以 app 模块的 build.gradle 来看
apply plugin: 'com.android.application' // 引入 android gradle 插件 android { // 配置 android gradle plugin 需要的内容
compileSdkVersion 26
defaultConfig { // 版本,applicationId 等配置
applicationId "com.zy.easygradle"
minSdkVersion 19
targetSdkVersion 26
versionCode 1
versionName "1.0"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
compileOptions { // 指定 java 版本
sourceCompatibility 1.8
targetCompatibility 1.8
} // flavor 相关配置
flavorDimensions "size", "color"
productFlavors {
big {
dimension "size"
}
small {
dimension "size"
}
blue {
dimension "color"
}
red {
dimension "color"
}
}
} // 项目需要的依赖
dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar']) // jar 包依赖
implementation 'com.android.support:appcompat-v7:26.1.0' // 远程仓库依赖
implementation 'com.android.support.constraint:constraint-layout:1.1.3'
implementation project(':module1') // 项目依赖
}
2.4 依赖
在 gradle 3.4 里引入了新的依赖配置,如下:
新配置 | 弃用配置 | 行为 | 作用 |
---|---|---|---|
implementation | compile | 依赖项在编译时对模块可用,并且仅在运行时对模块的消费者可用。 对于大型多项目构建,使用 implementation 而不是 api/compile 可以显著缩短构建时间,因为它可以减少构建系统需要重新编译的项目量。 大多数应用和测试模块都应使用此配置。 | implementation 只会暴露给直接依赖的模块,使用此配置,在模块修改以后,只会重新编译直接依赖的模块,间接依赖的模块不需要改动 |
api | compile | 依赖项在编译时对模块可用,并且在编译时和运行时还对模块的消费者可用。 此配置的行为类似于 compile(现在已弃用),一般情况下,您应当仅在库模块中使用它。 应用模块应使用 implementation,除非您想要将其 API 公开给单独的测试模块。 | api 会暴露给间接依赖的模块,使用此配置,在模块修改以后,模块的直接依赖和间接依赖的模块都需要重新编译 |
compileOnly | provided | 依赖项仅在编译时对模块可用,并且在编译或运行时对其消费者不可用。 此配置的行为类似于 provided(现在已弃用)。 | 只在编译期间依赖模块,打包以后运行时不会依赖,可以用来解决一些库冲突的问题 |
runtimeOnly | apk | 依赖项仅在运行时对模块及其消费者可用。 此配置的行为类似于 apk(现在已弃用)。 | 只在运行时依赖模块,编译时不依赖 |
还是以 EasyGradle 为例,看一下各个依赖的不同: 项目里有三个模块:app,module1, module2
模块 app 中有一个类 ModuleApi
模块 module1 中有一个类 Module1Api
模块 module2 中有一个类 Module2Api
其依赖关系如下:
implementation 依赖
当 module1 使用 implementation 依赖 module2 时,在 app 模块中无法引用到 Module2Api 类
api 依赖
当 module1 使用 api 依赖 module2 时,在 app 模块中可以正常引用到 Module2Api 类,如下图
compileOnly 依赖
当 module1 使用 compileOnly 依赖 module2 时,在编译阶段 app 模块无法引用到 Module2Api 类,module1 中正常引用,但是在运行时会报错
反编译打包好的 apk,可以看到 Module2Api 是没有被打包到 apk 里的
runtimeOnly 依赖
当 module1 使用 runtimeOnly 依赖 module2 时,在编译阶段,module1 也无法引用到 Module2Api
2.5 flavor
在介绍下面的流程之前,先明确几个概念,flavor,dimension,variant
在 android gradle plugin 3.x 之后,每个 flavor 必须对应一个 dimension,可以理解为 flavor 的分组,然后不同 dimension 里的 flavor 两两组合形成一个 variant
举个例子
如下配置:
flavorDimensions "size", "color" productFlavors {
big {
dimension "size"
}
small {
dimension "size"
}
blue {
dimension "color"
}
red {
dimension "color"
}
}
那么生成的 variant 对应的就是 bigBlue,bigRed,smallBlue,smallRed
每个 variant 可以对应的使用 variantImplementation 来引入特定的依赖,比如:bigBlueImplementation,只有在 编译 bigBlue variant的时候才会引入
三、gradle wrapper
gradlew / gradlew.bat 这个文件用来下载特定版本的 gradle 然后执行的,就不需要开发者在本地再安装 gradle 了。这样做有什么好处呢?开发者在本地安装 gradle,会碰到的问题是不同项目使用不同版本的 gradle 怎么处理,用 wrapper 就很好的解决了这个问题,可以在不同项目里使用不同的 gradle 版本。gradle wrapper 一般下载在 GRADLE_CACHE/wrapper/dists 目录下
gradle/wrapper/gradle-wrapper.properties 是一些 gradlewrapper 的配置,其中用的比较多的就是 distributionUrl,可以执行 gradle 的下载地址和版本
gradle/wrapper/gradle-wrapper.jar 是 gradlewrapper 运行需要的依赖包
四、gradle init.gradle
在 gradle 里,有一种 init.gradle 比较特殊,这种脚本会在每个项目 build 之前先被调用,可以在其中做一些整体的初始化操作,比如配置 log 输出等等
使用 init.gradle 的方法:
- 通过 --init-script 指定 init.gradle 位置
eg: gradlew --init-script initdir/init.gradle - init.gradle 文件放在 USER_HOME/.gradle/ 目录下
- .gradle 脚本放在 USER_HOME/.gradle/init.d/ 目录下
- .gradle 脚本放在 GRDALE_HOME/init.d/ 目录下
五、gradle 生命周期及回调
gradle 构建分为三个阶段
初始化阶段
初始化阶段主要做的事情是有哪些项目需要被构建,然后为对应的项目创建 Project 对象
配置阶段
配置阶段主要做的事情是对上一步创建的项目进行配置,这时候会执行 build.gradle 脚本,并且会生成要执行的 task
执行阶段
执行阶段主要做的事情就是执行 task,进行主要的构建工作
gradle 在构建过程中,会提供一些列回调接口,方便在不同的阶段做一些事情,主要的接口有下面几个
gradle.addBuildListener(new BuildListener() {
@Override
void buildStarted(Gradle gradle) {
println('构建开始')
// 这个回调一般不会调用,因为我们注册的时机太晚,注册的时候构建已经开始了,是 gradle 内部使用的
} @Override
void settingsEvaluated(Settings settings) {
println('settings 文件解析完成')
} @Override
void projectsLoaded(Gradle gradle) {
println('项目加载完成')
gradle.rootProject.subprojects.each { pro ->
pro.beforeEvaluate {
println("${pro.name} 项目配置之前调用")
}
pro.afterEvaluate{
println("${pro.name} 项目配置之后调用")
}
}
} @Override
void projectsEvaluated(Gradle gradle) {
println('项目解析完成')
} @Override
void buildFinished(BuildResult result) {
println('构建完成')
}
}) gradle.taskGraph.whenReady {
println("task 图构建完成")
}
gradle.taskGraph.beforeTask {
println("每个 task 执行前会调这个接口")
}
gradle.taskGraph.afterTask {
println("每个 task 执行完成会调这个接口")
}
六、自定义 task
默认创建的 task 继承自 DefaultTask 如何声明一个 task
task myTask {
println 'myTask in configuration'
doLast {
println 'myTask in run'
}
} class MyTask extends DefaultTask {
@Input Boolean myInputs
@Output
@TaskAction
void start() {
}
} tasks.create("mytask").doLast {
}
Task 的一些重要方法分类如下:
Task 行为
Task.doFirst
Task.doLastTask 依赖顺序
Task.dependsOn
Task.mustRunAfter
Task.shouldRunAfter
Task.finalizedByTask 的分组描述
Task.group
Task.descriptionTask 是否可用
Task.enabledTask 输入输出
gradle 会比较 task 的 inputs 和 outputs 来决定 task 是否是最新的,如果 inputs 和 outputs 没有变化,则认为 task 是最新的,task 就会跳过不执行
Task.inputs
Task.outputsTask 是否执行
可以通过指定 Task.upToDateWhen = false 来强制 task 执行
Task.upToDateWhen
比如要指定 Task 之间的依赖顺序,写法如下:
task task1 {
doLast {
println('task2')
}
}
task task2 {
doLast {
println('task2')
}
}
task1.finalizedBy(task2)
task1.dependsOn(task2)
task1.mustRunAfter(task2)
task1.shouldRunAfter(task2)
task1.finalizedBy(task2)
七、Android transform
要实现 transform 需要继承 com.android.build.api.transform.Transform 并实现其方法,实现了 Transform 以后,要想应用,就调用 project.android.registerTransform()
public class MyTransform extends Transform {
@Override
public String getName() {
// 返回 transform 的名称,最终的名称会是 transformClassesWithMyTransformForDebug 这种形式
return "MyTransform";
} @Override
public Set<QualifiedContent.ContentType> getInputTypes() {
/**
返回需要处理的数据类型 有 下面几种类型可选
public static final Set<ContentType> CONTENT_CLASS = ImmutableSet.of(CLASSES);
public static final Set<ContentType> CONTENT_JARS = ImmutableSet.of(CLASSES, RESOURCES);
public static final Set<ContentType> CONTENT_RESOURCES = ImmutableSet.of(RESOURCES);
public static final Set<ContentType> CONTENT_NATIVE_LIBS = ImmutableSet.of(NATIVE_LIBS);
public static final Set<ContentType> CONTENT_DEX = ImmutableSet.of(ExtendedContentType.DEX);
public static final Set<ContentType> DATA_BINDING_ARTIFACT = ImmutableSet.of(ExtendedContentType.DATA_BINDING);
*/
return TransformManager.CONTENT_CLASS;
} @Override
public Set<? super QualifiedContent.Scope> getScopes() {
/**
返回需要处理内容的范围,有下面几种类型
PROJECT(1), 只处理项目的内容
SUB_PROJECTS(4), 只处理子项目
EXTERNAL_LIBRARIES(16), 只处理外部库
TESTED_CODE(32), 只处理当前 variant 对应的测试代码
PROVIDED_ONLY(64), 处理依赖
@Deprecated
PROJECT_LOCAL_DEPS(2),
@Deprecated
SUB_PROJECTS_LOCAL_DEPS(8);
*/
return Sets.immutableEnumSet(QualifiedContent.Scope.PROJECT);
} @Override
public boolean isIncremental() {
// 是否增量,如果返回 true,TransformInput 会包括一份修改的文件列表,返回 false,会进行全量编译,删除上一次的输出内容
return false;
} @Override
void transform(TransformInvocation transformInvocation) throws TransformException, InterruptedException, IOException {
// 在这里处理 class
super.transform(transformInvocation)
// 在 transform 里,如果没有任何修改,也要把 input 的内容输出到 output,否则会报错
for (TransformInput input : transformInvocation.inputs) {
input.directoryInputs.each { dir ->
// 获取对应的输出目录
File output = transformInvocation.outputProvider.getContentLocation(dir.name, dir.contentTypes, dir.scopes, Format.DIRECTORY)
dir.changedFiles // 增量模式下修改的文件
dir.file // 获取输入的目录
FileUtils.copyDirectory(dir.file, output) // input 内容输出到 output
}
input.jarInputs.each { jar ->
// 获取对应的输出 jar
File output = transformInvocation.outputProvider.getContentLocation(jar.name, jar.contentTypes, jar.scopes, Format.JAR)
jar.file // 获取输入的 jar 文件
FileUtils.copyFile(jar.file, output) // input 内容输出到 output
}
}
}
} // 注册 transform
android.registerTransform(new MyTransform())
在 transform 中的处理,一般会涉及到 class 文件的修改,操纵字节码的工具一般是 javasist 和 asm 居多,这两个工具在这里先不介绍了。后面有机会会展开说一下
八、自己写 plugin
gradle 的插件可以看作是一系列 task 的集合
在 android 工程的 build.gradle 脚本里,第一行就是 apply plugin: 'com.android.application',这个就是引入 android gradle 插件,插件里有 android 打包相关的 task
关于 android gradle plugin 的源码分析,在后面会讲到,现在先看看如何实现一个自己的 plugin
8.1 初始化工程
- 在 android studio 中创建一个 java module
- 在 src/main 目录下创建 groovy 目录,然后创建自己的包名和插件类
- 在 src/main 目录下创建 resources/META-INFO/gradle-plugins 目录,创建 ,myplugin.properties 文件,文件里内容是
implementation-class=com.zy.plugin.MyPlugin // 这里是自己的插件类
4.修改 build.gradle 文件
// 引入 groovy 和 java 插件
apply plugin: 'groovy'
apply plugin: 'java' buildscript {
repositories {
mavenLocal()
maven { url 'http://depot.sankuai.com/nexus/content/groups/public/' }
maven { url 'https://maven.google.com' }
jcenter()
}
} repositories {
mavenLocal()
maven {
url "http://mvn.dianpingoa.com/android-nova"
}
maven {
url 'http://depot.sankuai.com/nexus/content/groups/public/'
}
maven { url 'https://maven.google.com' }
} dependencies {
compile gradleApi()
compile localGroovy()
compile 'com.android.tools.build:gradle:3.0.1'
}
现在为止,项目结构是这个样子的
8.2 创建 Plugin
在刚才创建的插件类里,就可以写插件的代码了。插件类继承 Plugin,并实现 apply 接口,apply 就是在 build.gradle 里 apply plugin 'xxx' 的时候要调用的接口了
插件开发可以使用 groovy 和 java,使用 groovy 的话可以有更多的语法糖,开发起来更方便一些
package com.zy.plugin import org.gradle.api.Plugin
import org.gradle.api.Project class MyPlugin implements Plugin<Project> { @Override
void apply(Project project) {
println("apply my plugin")
}
}
8.3 创建插件的 task
我们再定义一个 task 类 MyTask,继承自 DefaultTask,简单的输出一些信息
package com.zy.plugin import org.gradle.api.DefaultTask
import org.gradle.api.tasks.TaskAction class MyTask extends DefaultTask { @TaskAction
void action() {
println('my task run')
}
}
然后在 plugin 中注册这个 task
class MyPlugin implements Plugin<Project> { @Override
void apply(Project project) {
println("apply my plugin")
project.tasks.create("mytask", MyTask.class)
}
}
8.4 本地安装插件
这样一个简单的插件就开发好了,如何使用呢
我们首先需要在 build.gradle 中引入 maven 插件,并且配置 install 相关的属性
apply plugin: 'maven' install {
repositories.mavenInstaller {
pom.version = '0.0.1' // 配置插件版本号
pom.artifactId = 'myplugin' // 配置插件标识
pom.groupId = 'com.zy.plugin' // 配置插件组织
}
}
之后执行 ./gradlew install 便会把插件安装在本地 maven 仓库
之后在使用的地方引入我们插件的 classpath
classpath 'com.zy.plugin:myplugin:0.0.1'
之后加载插件
apply plugin; 'myplugin' // 这里的 myplugin 是前面说的 myplugin.properties 的名字
然后运行 ./gradlew tasks --all | grep mytask,就可以看到我们在 plugin 里新增的 task 了
./gradlew mytasks 就可以执行 task 了
8.5 打包发布
在插件 build.gradle 里新增上传的配置如下
uploadArchives {
repositories {
mavenDeployer {
repository(url: "mavenUrl")
pom.version = '0.0.1'
pom.artifactId = 'myplugin'
}
}
}
运行 ./gradlew uploadArchives 就可以了
8.6 调试插件
那么开发插件的时候如何调试呢?
1.首先在 as 中新增一个 remote 配置
2.之后在执行 task 的时候增加下面的参数
./gradlew app:mytask -Dorg.gradle.debug=true
此时可以看到 gradle 在等待 debug 进程连接
3.之后在插件代码中打好断点,在 as 中点击 debug 按钮,就可以调试插件代码了
九、重点总结
主要要点如下图:
其中一定要掌握的如下:
- gradle dsl 查询地址 docs.gradle.org/current/dsl…
- android gradle plugin dsl 查询地址 google.github.io/android-gra…
- gradle 构建生命周期和回调
- implementation / api
- flavor
- 自定义 Task
- 自定义 Transform 和 自定义插件可以作为扩展内容
作者:ZYLAB
链接:https://juejin.im/post/5cd441f851882554b86d088b
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
Gradle之Gradle 的基本使用(一)的更多相关文章
- Gradle笔记——Gradle的简介与安装
本博客对Gradle进行一个简单的介绍,以及它的安装. Gradle介绍 Gradle是一个基于JVM的构建工具,它提供了: 像Ant一样,通用灵活的构建工具 可以切换的,基于约定的构建框架 强大的多 ...
- Chapter 3. Installing Gradle 安装gradle
3.1. Prerequisites Gradle requires a Java JDK or JRE to be installed, version 6 or higher (to check, ...
- Gradle、Gradle Wrapper与Android Plugin for Gradle
欢迎和大家交流技术相关问题: 邮箱: jiangxinnju@163.com 博客园地址: http://www.cnblogs.com/jiangxinnju GitHub地址: https://g ...
- 彻底搞懂Gradle、Gradle Wrapper与Android Plugin for Gradle的区别和联系
首先用一段通俗易懂但是不是非常专业的话描述一下三者的概念.区别和联系. Gradle是个构建系统,能够简化你的编译.打包.测试过程.熟悉Java的同学,可以把Gradle类比成Maven. Gradl ...
- AS 中 Plugin for Gradle 和 Gradle 之间的版本对应关系
Plugin for Gradle 和 Gradle 之间的版本对应关系 来源:https://developer.android.com/studio/releases/gradle-plugin. ...
- 【Gradle】 Gradle 综合
Gradle User Guide:http://www.gradle.org/docs/current/userguide/userguide.html 针对它的中文翻译:http://ask.an ...
- Gradle:Gradle入门
一.安装Gradle 1.首先确保你安装的JDK1.5或以上版本号. C:\Users\chengxiang.peng.QUNARSERVERS>java -version java ver ...
- 【Gradle】Gradle任务
Gradle任务 多种方式创建任务 1.直接以一个任务名字创建一个任务的方式: def Task task1 = task(task1) task1.doLast{ println 'task1' } ...
- 【Gradle】Gradle入门
Gradle入门 配置Gradle环境 安装之前确保已经安装配置好Java环境,要求JDK6以上,并且在环境变量里配置了JAVA_HOME,查看Java版本可以在终端输入如下命令: java -ver ...
- Gradle之Gradle 源码分析(四)
Gradle 的启动 constructTaskGraph runTasks finishBuild gradle 脚本如何编译和执 插件调用流程 一.Gradle 的启动 1.1 整体实现图 1.2 ...
随机推荐
- java数据结构03
1.求二叉树的深度 https://www.cnblogs.com/xudong-bupt/p/4036190.html class TreeNode { char val; TreeNode lef ...
- ui自动化之selenium操作(三)xpath定位
xpath 的定位方法,非常强大.使用这种方法几乎可以定位到页面上的任意元素. 1. 什么是xpath? xpath 是XML Path的简称, 由于HTML文档本身就是一个标准的XML页面,所以我们 ...
- 关于AP如何获取station的rssi
最近在研究一个问题:如何通过AP来获取station的rssi. 具体可以拆分为以下三种情况: 1.首先station如果已经连接到AP上,这种情况很容易就能够得到station的RSSI.这里就不讨 ...
- pytho xml
转载自:https://www.cnblogs.com/gouguoqilinux/p/9168332.html xml是实现不同语言或程序直接进行数据交换的协议,跟json差不多,单json使用起来 ...
- Python模块-requests模块使用
写在前面 这篇文章是我照着廖雪峰python网站学习的,大致内容差不多,多了我一丢丢的自己的想法.如果发现有什么不对的话请及时联系我.qq:472668561 参考链接:https://www.lia ...
- mac中登陆mysql忘记密码解决办法
1.打开终端,输入命令:cd /usr/local/mysql/bin 2.mysql -uroot -p,用这条命令登陆时报错信息: 报错:Enter password: ERROR 1045 (2 ...
- ZROI 19.08.10模拟赛
传送门 写在前面:为了保护正睿题目版权,这里不放题面,只写题解. A \(20pts:\) 枚举操作序列然后暴力跑,复杂度\(O(6^n)\). \([50,80]pts:\) 枚举改成dfs,每层操 ...
- chrome获取页面element的xPath
chrome真的是强大的工具 1.在chrome打开的页面点击F12,进入开发者模式 2.点击弹出的开发者工具左上角的跟踪箭头,再点击需要跟踪的页面元素,html程序定位到元素在程序中位置 3.选中的 ...
- 微信小程序中的自定义组件(components)
其实小程序开发很像vue和react的结合,数据绑定和setData 重新渲染页面的数据,最近发现连写组件都是很像,也是醉了,自我认为哈, 因为小程序可以将页面内的功能模块抽象成自定义组件,以便在 ...
- css---一个大div中套左右两个div,如何让最高的把最低的撑开?且把父级撑开呢?
到最后实现了效果,但是在理论上感觉还是很牵强,如果哪位大神有方法,请评论指出哦 Html: css: