当前位置: 首页 > news >正文

【算法导论】快速排序

文章目录

      • 1. 快速排序的描述
        • 1.1基本描述
        • 1.2 PARTITOION函数
        • 1.3 快速排序C++完整代码
      • 2. 快速排序的性能
        • 2.1 最坏时间复杂度
        • 2.2 平均时间复杂度

1. 快速排序的描述

1.1基本描述

  快速排序是一种时间复杂度为 O(n^2) 的排序算法。虽然最坏情况时间复杂度很差,但他的平均性能却很好,它的期望时间复杂度是 O(nlgn) 而且 O(nlgn) 中隐含的常数因子很小大约是1.44左右。
  快速排序与归并排序一样,也是基于归并的思想以下是对其子数组 A[ p…r ] 进行快速排序三步分治的过程:

分解:数组 A[ p…r ] 被划分为两个子数组 A[ p…q-1] 和 A[q+1…r],使得 A[ p…q-1]中的每一个元素都小于等于 A[ q ] ,而 A[ q ]也小于等于 A[ q+1…r]中的每一个元素。

解决:通过递归调用快速排序,对子数组A[ p…q-1] 和 A[q+1…r]进行排序。

合并:因为子数组都是原址排序的,所以并不需要合并操作:A[ p…r ] 已经有序。

快速排序伪代码:

QUICKSORT(A,p,r)if p < rq = PARTITION(A, p ,r)QUICKSORT(A, p ,q-1)QUICKSORT(A, q+1 ,r)

其中,q = PARTITION(A, p ,r) 所执行的操作就是分解操作,并返回 q。

1.2 PARTITOION函数


PARTITION函数伪代码:

PARTITION(A, p, r) x = A[r] i = p - 1 for j = p to r - 1 do if A[j] ≤ xthen i = i + 1 exchange A[i] with A[j] 
exchange A[i + 1] with A[r] 
return i + 1

PARTITION函数思路简介:

  其中 0 - i 维护的是小于等于A[ r ] 的序列,A[ i+1 ] 即为比 A[ r ] 大的第一个数,j从开始节点遍历到倒数第二个节点,遇到比 A[ r ] 小的数便进行 A[ j ] 与 A[ i+1 ] 的交换,以此维护了 0 - i 序列中比 A[ r ]小的特性。j 遍历完成后,即实现了A[0…i] 小于等于 A[ r ] ,A[i+1…j] 大于 A[ r ]。

PARTITION函数执行过程:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

1.3 快速排序C++完整代码
# include <iostream>
using namespace std;
int PARTITION(int A[],int p,int r)
{int x = A[r];int i = p - 1;for(int j = p; j <= r - 1; j++){if(A[j] <= x){i = i + 1;swap(A[i],A[j]);}}swap(A[i+1] , A[r]);return i+1;
}
void QUICKSORT(int A[],int p,int r)
{if(p < r){int q = PARTITION(A,p,r);QUICKSORT(A,p,q-1);QUICKSORT(A,q+1,r);}
}
int main()
{int A[100010];int n;cin>>n;for(int i=0;i<n;i++){cin>>A[i];}QUICKSORT(A,0,n-1);for(int i=0;i<n;i++){cout<<A[i]<<" ";}return 0;
}

2. 快速排序的性能

2.1 最坏时间复杂度

因为无法举出使快速排序达到最坏情况的例子所以我们通过两步证明其最坏时间复杂度为 O( n^2 )

  1. 举一个例子证明其时间复杂度为 O( n 2 n^2 n2)
    当对快速排序进行分解的过程中得到的结果为一部分为 n 个元素,另一部分为0个元素时,该例的运行时间递归式为:
    T ( n ) = T ( n − 1 ) + T ( 0 ) + θ ( n ) = T ( n − 1 ) + θ ( n ) \begin{aligned}T(n) &= T(n-1)+T(0)+\theta(n)\\ &= T(n-1)+\theta(n) \end{aligned} T(n)=T(n1)+T(0)+θ(n)=T(n1)+θ(n)可以得到: T ( n ) = θ ( n 2 ) T(n)=\theta(n^2) T(n)=θ(n2)
    由此可知, Quicksort 的最坏运行时间为: Ω( n 2 n^2 n2)
  2. 证明其时间复杂度不超过 O( n 2 n^2 n2)
    设T(n)是对大小为 n 的输入进行快速排序的最坏情况时间,则有递归:
    T ( n ) = max ⁡ 0 ≤ q ≤ n − 1 ( T ( q ) + T ( n − q − 1 ) ) + C 1 n T(n)=\max_{0 \leq q\leq n-1}(T(q)+T(n-q-1))+C_1n T(n)=0qn1max(T(q)+T(nq1))+C1n我们猜测对于某个常数C使得T (n) ≤ C,将这个猜测代入上面的递推式,我们得到:
    T ( n ) ≤ max ⁡ 0 ≤ q ≤ n − 1 ( C q 2 + C ( n − q − 1 ) 2 ) + C 1 n = C ∗ max ⁡ 0 ≤ q ≤ n − 1 ( q 2 + ( n − 1 − q ) 2 ) + C 1 n \begin{aligned}T(n)&\leq \max_{0 \leq q\leq n-1}(Cq^2+C(n-q-1)^2)+C_1n\\&=C* \max_{0 \leq q\leq n-1}(q^2+(n-1-q)^2)+C_1n\end{aligned} T(n)0qn1max(Cq2+C(nq1)2)+C1n=C0qn1max(q2+(n1q)2)+C1n由于 q 2 + ( n − 1 − q ) 2 q^2+(n-1-q)^2 q2+(n1q)2 q q q 的二次函数,求导可得,在区间 [1… n ] 范围内该函数只可能在 q = 1 , q = n , q = n / 4 q = 1,q = n,q = n/4 q=1q=nq=n/4,等三个点处取极值,由此可知:
    ∑ 0 ≤ q ≤ n − 1 ( q 2 + ( n − 1 − q ) 2 ) ≤ n 2 \sum_{0 \leq q\leq n-1}(q^2+(n-1-q)^2) \leq n^2 0qn1(q2+(n1q)2)n2所以有:
    T ( n ) ≤ C ( n − 1 ) 2 + C 1 n = C ∗ n 2 − 2 C n + C 1 n + C T(n)\leq C(n-1)^2+C_1n=C*n^2-2Cn+C_1n+C T(n)C(n1)2+C1n=Cn22Cn+C1n+C当取 C > C1 时, T ( n ) < = C n 2 T (n)<=Cn^2 T(n)<=Cn2 对所有 n > = 1 n >=1 n>=1 成立,由此证得快速排序时间复杂度不超过 O( n 2 n^2 n2)。
2.2 平均时间复杂度

  设 T ( n ) T (n) T(n) 为输入规模为 n 时 QUICKSORT 算法的平均运行时间, T k ( n ) T_k(n) Tk(n)为所选划分元序号为 k+1 时 QUICKSORT 算法的平均运行时间,则 T ( n ) T (n) T(n) 满足以下递归方程:
T ( n ) = ∑ k = 0 n − 1 ( p ( k + 1 ) T k ( n ) ) T(n)=\sum_{k=0}^{n-1} (p(k+1)T_k(n)) T(n)=k=0n1(p(k+1)Tk(n))其中 p ( k + 1 ) p(k+1) p(k+1)为划分元素序号为 k + 1 k+1 k+1的概率,我们可以知道划分元素序号的概率是相同的故: p ( k + 1 ) = 1 n p(k+1)=\frac{1}{n} p(k+1)=n1,带入上式可得:
T ( n ) = ∑ k = 0 n − 1 1 n T k ( n ) ) = 1 n ∑ k = 0 n − 1 ( T ( k ) + T ( n − k − 1 ) + c n ) T(n)=\sum_{k=0}^{n-1} \frac{1}{n}T_k(n))= \frac{1}{n}\sum_{k=0}^{n-1} (T(k)+T(n-k-1)+cn) T(n)=k=0n1n1Tk(n))=n1k=0n1(T(k)+T(nk1)+cn)继续化简可得:
1 n ( ∑ k = 0 n − 1 T ( k ) + ∑ k = 0 n − 1 T ( n − k − 1 ) ) + c n = 2 n ∑ k = 0 n − 1 T ( k ) + c n \frac{1}{n}(\sum_{k=0}^{n-1} T(k)+\sum_{k=0}^{n-1}T(n-k-1))+cn=\frac{2}{n}\sum_{k=0}^{n-1}T(k)+cn n1(k=0n1T(k)+k=0n1T(nk1))+cn=n2k=0n1T(k)+cn解递归方程可得:
n T ( n ) = 2 ∑ k = 0 n − 1 T ( k ) + c n 2 nT(n)=2\sum_{k=0}^{n-1}T(k)+cn^2 nT(n)=2k=0n1T(k)+cn2 ( n − 1 ) T ( n − 1 ) = 2 ∑ k = 0 n − 1 T ( k ) + c n 2 (n-1)T(n-1)=2\sum_{k=0}^{n-1}T(k)+cn^2 (n1)T(n1)=2k=0n1T(k)+cn2两式相减,可得:
n T ( n ) − ( n − 1 ) T ( n − 1 ) = 2 T ( n − 1 ) + c ( 2 n − 1 ) nT(n)-(n-1)T(n-1)=2T(n-1)+c(2n-1) nT(n)(n1)T(n1)=2T(n1)+c(2n1) T ( n ) n + 1 < = T ( n − 1 ) n + 2 c n \frac{T(n)}{n+1}<=\frac{T(n-1)}{n}+\frac{2c}{n} n+1T(n)<=nT(n1)+n2c
G ( n ) = T ( n ) ( n + 1 ) G(n) = \frac{T(n)}{(n+1)} G(n)=(n+1)T(n)则有:
G ( n ) ≤ C ( n − 1 ) + 2 c n = G ( n − 2 ) + 2 c ( 1 n − 1 + 1 n ) = G ( n − 3 ) + 2 c ( 1 n − 2 + 1 n − 1 + 1 n ) = G ( n − k ) + 2 c ( 1 n − k + 1 + . . . + 1 n − 1 + 1 n ) \begin{aligned}G(n)& \leq C(n-1)+\frac{2c}{n}\\&=G(n-2)+2c(\frac{1}{n-1}+\frac{1}{n})\\&=G(n-3)+2c(\frac{1}{n-2}+\frac{1}{n-1}+\frac{1}{n})\\&=G(n-k)+2c(\frac{1}{n-k+1}+...+\frac{1}{n-1}+\frac{1}{n}) \end{aligned} G(n)C(n1)+n2c=G(n2)+2c(n11+n1)=G(n3)+2c(n21+n11+n1)=G(nk)+2c(nk+11+...+n11+n1)整理后得到: G ( 1 ) + 2 c ∑ k = 0 n − 2 1 n − k = 2 c ∑ k = 2 n 1 k < = 2 c ∗ H n < = 2 c l o g n G(1)+2c\sum_{k=0}^{n-2}\frac{1}{n-k}=2c\sum_{k=2}^{n}\frac{1}{k}<=2c*H_n<=2clogn G(1)+2ck=0n2nk1=2ck=2nk1<=2cHn<=2clogn
所以,Quicksort 算法的平均时间复杂度为: T ( n ) = G ( n ) ( n + 1 ) = θ ( n l o g n ) T(n)=G(n)(n+1)=\theta(nlogn) T(n)=G(n)(n+1)=θ(nlogn)

相关文章:

【算法导论】快速排序

文章目录 1. 快速排序的描述 1.1基本描述1.2 PARTITOION函数1.3 快速排序C完整代码 2. 快速排序的性能2.1 最坏时间复杂度2.2 平均时间复杂度 1. 快速排序的描述 1.1基本描述 快速排序是一种时间复杂度为 O(n^2) 的排序算法。虽然最坏情况时间复杂度很差&#xff0c;但他的平…...

QT之QScriptEngine的用法介绍

QT之QScriptEngine的用法介绍 成员函数用法举例 成员函数 1&#xff09;QScriptEngine::evaluate(const QString &program, const QString &fileName QString(), int lineNumber 1) 执行 JavaScript 代码并返回结果。 2&#xff09;QScriptEngine::evaluate(const…...

vim 工具的使用

注&#xff1a;以下操作都在普通模式下进行 光标的移动操作 gg 定位到代码的第一行 shiftg 定位到代码的最后一行 nshiftg 定位到第n行 shift6: 特定一行的开始 shift4 特定一行的结尾 上下左右的移动光标 h: 向左移动光标 j: 向下移动光标 k: 向上移动光标 l: 向右移动光标 …...

RPA有什么优势?RPA的8大优势!建议学习!

随着科技的不断发展&#xff0c;越来越多的企业开始寻求数字化转型&#xff0c;以提高生产力和效率。在这个过程中&#xff0c;RPA&#xff08;Robotic Process Automation&#xff09;机器人流程自动化技术逐渐成为企业数字化转型的重要工具之一。本文将从八个方面阐述RPA的优…...

初级篇—第二章SELECT查询语句

文章目录 什么是SQLSQL 分类SQL语言的规则与规范阿里巴巴MySQL命名规范数据导入指令 显示表结构 DESC基本的SELECT语句SELECTSELECT ... FROM列的别名 AS去除重复行 DISTINCT空值参与运算着重号查询常数过滤数据 WHERE练习 运算符算术运算符加减符号乘除符号取模符号 符号比较运…...

PostMan的学习

PostMan的学习 目录 环境变量和全局变量接口关联内置动态参数以及自定义动态参数实现业务闭环Postman断言批量运行collection数据驱动之CSV文件和JSON文件测试必须带请求头的接口Mock Serviers 服务器Cookie鉴权NewmanPostManNewManjenkins实现接口测试持续集成 参考资料&am…...

配置OSPF路由

OSPF路由 1.OSPF路由 1.1 OSPF简介 OSPF(Open Shortest Path First&#xff0c;开放式最短路径优先&#xff09;路由协议是另一个比较常用的路由协议之一&#xff0c;它通过路由器之间通告网络接口的状态&#xff0c;使用最短路径算法建立路由表。在生成路由表时&#xff0c;…...

CCF CSP认证 历年题目自练Day17

CCF CSP认证 历年题目自练Day17 题目一 试题编号&#xff1a; 201803-1 试题名称&#xff1a; 跳一跳 时间限制&#xff1a; 1.0s 内存限制&#xff1a; 256.0MB 问题描述&#xff1a; 问题描述   近来&#xff0c;跳一跳这款小游戏风靡全国&#xff0c;受到不少玩家的喜爱…...

基于Matlab实现多因子选股模型(附上源码+数据)

本文将介绍如何使用MATLAB实现多因子选股模型。我们将使用市盈率和市净率两个因子来进行选股&#xff0c;并通过简单的代码案例来演示该过程。 文章目录 引言简单案例总结源码数据下载 引言 多因子选股模型是一种常用的股票选股方法&#xff0c;通过综合考虑多个因子的信息来…...

【中秋国庆不断更】OpenHarmony多态样式stateStyles使用场景

Styles和Extend仅仅应用于静态页面的样式复用&#xff0c;stateStyles可以依据组件的内部状态的不同&#xff0c;快速设置不同样式。这就是我们本章要介绍的内容stateStyles&#xff08;又称为&#xff1a;多态样式&#xff09;。 概述 stateStyles是属性方法&#xff0c;可以根…...

Qt扩展-QCustomPlot绘图基础概述

QCustomPlot绘图基础概述 一、概述二、改变外观1. Graph 类型2. Axis 坐标轴3. 网格 三、案例1. 简单布局两个图2. 绘图与多个轴和更先进的样式3. 绘制日期和时间数据 四、其他Graph&#xff1a;曲线&#xff0c;条形图&#xff0c;统计框图&#xff0c;… 一、概述 本教程使用…...

二、局域网联机

目录 1.下载资源包 2.配置NetworkManager 3.编写测试UI 1.下载资源包 2.配置NetworkManager &#xff08;1&#xff09;在Assets/Prefabs下创建Network Prefabs List 相应设置如下&#xff1a; &#xff08;2&#xff09; 创建空物体“NetworkManager”并挂载NetworkMan…...

决策树剪枝:解决模型过拟合【决策树、机器学习】

如何通过剪枝解决决策树的过拟合问题 决策树是一种强大的机器学习算法&#xff0c;用于解决分类和回归问题。决策树模型通过树状结构的决策规则来进行预测&#xff0c;但在构建决策树时&#xff0c;常常会出现过拟合的问题&#xff0c;即模型在训练数据上表现出色&#xff0c;…...

Ubuntu部署运行ORB-SLAM2

ORB-SLAM2是特征点法的视觉SLAM集大成者&#xff0c;不夸张地说是必学代码。博主已经多次部署运行与ORB-SLAM2相关的代码&#xff0c;所以对环境和依赖很熟悉&#xff0c;对整个系统也是学习了几个月&#xff0c;一行行代码理解。本次在工控机上部署记录下完整的流程。 ORB-SLA…...

二十,镜面IBL--打印BRDF积分贴图

比起以往&#xff0c;这节应该是最轻松的了&#xff0c; 运行结果如下 代码如下&#xff1a; #include <osg/TextureCubeMap> #include <osg/TexGen> #include <osg/TexEnvCombine> #include <osgUtil/ReflectionMapGenerator> #include <osgDB/Re…...

自动驾驶:未来的道路上的挑战与机遇

自动驾驶&#xff1a;未来的道路上的挑战与机遇 文章目录 引言安全与道路事故的减少交通拥堵的缓解城市规划的变革技术和法律挑战结语 2023星火培训【专项营】Apollo开发者社区布道师倾力打造&#xff0c;包含PnC、新感知等的全新专项课程上线了。理论与实践相结合&#xff0c;…...

Go 语言 iota 的神奇力量

前言 当你深入研究官网库、开源库或者任何一个 Go 项目时&#xff0c;你都会发现 iota 这个神奇的标识符无处不在。它扮演着一种重要的角色&#xff0c;让代码变得更加简洁、清晰&#xff0c;并提高了可读性和可维护性。它的应用范围广泛&#xff0c;从枚举类型到位运算&#…...

前端开发和后端开发的一些建议

前端开发和后端开发是Web开发的两个方向 前端开发主要负责实现用户在浏览器上看到的界面和交互体验&#xff0c;包括HTML、CSS和JavaScript等技术。后端开发主要负责处理服务器端的逻辑和数据&#xff0c;包括数据库操作、服务器配置和接口开发等技术。 前端开发 前端开发需…...

基于 SpringBoot+Vue 的教室人事档案管理系统

1 简介 教师人事档案管理系统利用信息的合理管理&#xff0c;动态的、高效的、安全的实现了教师的各种需求&#xff0c;改变了传统的网上查看方式&#xff0c;使教师可以足不出户的在线查看最适合自己个人档案、奖惩信息、档案变动、培训报名或者新闻资讯。 1、教师后台功能模…...

Lua学习笔记:require非.lua拓展名的文件

前言 本篇在讲什么 Lua的require相关的内容 本篇需要什么 对Lua语法有简单认知 对C语法有简单认知 依赖Visual Studio工具 本篇的特色 具有全流程的图文教学 重实践&#xff0c;轻理论&#xff0c;快速上手 提供全流程的源码内容 ★提高阅读体验★ &#x1f449; ♠…...

Python 编程基础 | 第三章-数据类型 | 3.3、浮点数

一、浮点数...

beego---ORM相关操作

Beego框架是go语言开发的web框架。 **那什么是框架呢&#xff1f;**就是别人写好的代码&#xff0c;我们可以直接使用&#xff01;这个代码是专门针对某一个开发方向定制的&#xff0c;例如&#xff1a;我们要做一个网站&#xff0c;利用 beego 框架就能非常快的完成网站的开发…...

【网络原理】初始网络,了解概念

文章目录 1. 网络通信1.1 局域网LAN1.2 广域网WAN 2. 基础概念2.1 IP2.2 端口号 3. 认识协议4. 五元组5. 协议分层5.1 分层的作用5.2 OSI七层模型5.3 TCP/IP五层&#xff08;四层&#xff09;模型 6. 封装和分用 1. 网络通信 计算机与计算机之间是互相独立&#xff0c;是独立模…...

对象存储,从单机到分布式的演进

关于数据存储的相关知识,请大家关注“数据存储张”,各大平台同名。 通过《什么是云存储?从对象存储说起》我们对对象存储的历史、概念和基本使用有了一个大概的认识。而且我们以Minio为例,通过单机部署的模式实际操作了一下对象存储的GUI,感受了一下对象存储的用法。 在上…...

结构型设计模式——桥接模式

摘要 桥接模式(Bridge pattern): 使用桥接模式通过将实现和抽象放在两个不同的类层次中而使它们可以独立改变。 一、桥接模式的意图 将抽象与实现分离开来&#xff0c;使它们可以独立变化。 二、桥接模式的类图 Abstraction: 定义抽象类的接口Implementor: 定义实现类接口 …...

keepalived的vip实现nginx节点的主备

nginx wget http://nginx.org/download/nginx-1.18.0.tar.gz tar zxvf nginx-1.18.0.tar.gzyum install -y gcc gcc-c pcre pcre-devel zlib zlib-devel openssl openssl-devel libnl3-develcd nginx-1.18.0 mkdir -p /usr/local/nginx #需要使用https&#xff0c;在编译时启用…...

C++之std::atomic解决多线程7个问题(二百四)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…...

tailwindcss 如何在 uniapp 中使用

直接使用https://tailwindcss.com/docs/guides/vite这篇官方教程的写法是跑不通的&#xff0c;摸索以后整理了一下&#xff0c;最关键的是第6步 npm install -D tailwindcss postcss autoprefixernpx tailwindcss init -p在 tailwind.config.js 中写入 export default {conten…...

oracle-使用PLSQL工具自行修改用户密码

1、使用PLSQL工具&#xff0c;输入用户名和原密码登录&#xff0c;如下图 2、登录后&#xff0c;在会话下拉菜单中找到”Change password..” 3、在跳出的窗口中配置新密码&#xff0c;修改完成后单击”确认”&#xff0c;后退出PLSQL 4、重新打开PLSQL&#xff0c;使用新密码登…...

自动驾驶技术:现状与未来

自动驾驶技术&#xff1a;现状与未来 文章目录 引言自动驾驶技术的现状自动驾驶技术的挑战自动驾驶技术的未来结论结论 2023星火培训【专项营】Apollo开发者社区布道师倾力打造&#xff0c;包含PnC、新感知等的全新专项课程上线了。理论与实践相结合&#xff0c;全新的PnC培训不…...

资源库建设网站/爱站网seo工具包

定义和用法 defined() 函数检查某常量是否存在。 若常量存在&#xff0c;则返回 true&#xff0c;否则返回 false。 语法 defined(name) 参数描述name必需。规定要检查的常量的名称。例子 <?php define("GREETING","Hello world!"); echo defined(&quo…...

做网站能做职业吗/seo博客教程

f5 刷新页面 page——load事件&#xff0c;用if&#xff08;!IsPostBack&#xff09;也可以避免F5 刷新...

东莞保安公司投诉电话/seo标签怎么优化

之前因为懒&#xff0c;没有针对otter做更多的解释和说明&#xff0c;在使用过程中&#xff0c;也发现了一些问题&#xff0c;此次补上一个完整的文档&#xff0c;方便大家使用。 Otter是基于cannal开源的&#xff0c;canal又是基于mysql binlog的产品。我们就从binlog说起 bin…...

网站开发软件 d/如何在网上推广自己的产品

文章目录ARM裸机开发&#xff1a;主频与时钟一、时钟系统1.1 外部时钟电路1.2 7路PLL时钟源1.3 时钟树概览二、时钟配置2.1 内核时钟设置2.2 PFD时钟设置2.3 AHB、IPG 和 PERCLK 根时钟设置三、配置代码ARM裸机开发&#xff1a;主频与时钟 本章了解一下 IMX 的系统时钟主频配置…...

怎么看一个网站是用什么程序做的/搜索引擎营销有哪些

分享如何设置 macOS 暗夜模式&#xff0c;需要的朋友可以试试这个教程 如何在 macOS 中设置手动暗模式调度 1、单击菜单栏中的Apple 标志&#xff0c;然后单击系统偏好设置。 2、单击常规。 3、在外观下&#xff0c;选择自动。 4、单击返回以返回到系统偏好设置。 5、单击显示…...

wordpress 静态化 linux/免费推广有哪些

说明 这次 IO 给开发者带来了很多惊喜, ConstraintLayout 是其中较为实用的之一. Google 第一时间发布了官方的代码实验室指导教程, 从样例项目和实验操作出发一步步理解 ConstraintLayout. 这里是我的翻译. 同步于我的博客: http://quanqi.org/2016/05/20/code-labs-constrain…...