题目

  1. 最长递增子序列

给你一个整数数组 nums ,找到其中最长严格递增子序列的长度。

子序列 是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。例如,[3,6,2,7] 是数组 [0,3,1,6,2,2,7] 的子序列。

示例 1:

输入:nums = [10,9,2,5,3,7,101,18]
输出:4
解释:最长递增子序列是 [2,3,7,101],因此长度为 4 。

示例 2:

输入:nums = [0,1,0,3,2,3]
输出:4

示例 3:

输入:nums = [7,7,7,7,7,7,7]
输出:1

提示:

  • 1 <= nums.length <= 2500
  • -104 <= nums[i] <= 104

进阶:

你能将算法的时间复杂度降低到 O(n log(n)) 吗?

解题思路

递归暴力

这个一开始的思路是动态规划,但是动态规划的实现比较复杂,所以先用递归暴力的方法来实现。每一个单个的数字都可以作为一个递增子序列,把整个序列分为该以单个数字形成的子序列 a 和剩下数字组成的子序列 b 两部分。编写一个递归函数来计算 ab 能够组成的最长递增子序列的长度。每次递归都要判断 b序列的开头数字是否大于 a 的最后一个数字,如果大于,则需要比较比较加入 b 的开头数字和 a 的最后一个数字组成的子序列的长度和不加入 b 的开头数字组成的子序列的长度,取最大值。否则就直接计算去掉 b 的开头数字后的子序列与 a 能够组成的最长递增子序列的长度。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
func max(num1, num2 int) int {
if num1 > num2 {
return num1
}

return num2
}

func calculateLIS(currentSubsequence []int, remainingNums []int) int {
if len(remainingNums) == 0 {
return len(currentSubsequence)
} else if len(remainingNums) == 1 {
if remainingNums[0] > currentSubsequence[len(currentSubsequence)-1] {
return len(currentSubsequence) + 1
}
return len(currentSubsequence)
}

if remainingNums[0] > currentSubsequence[len(currentSubsequence)-1] {
include := calculateLIS(append(currentSubsequence, remainingNums[0]), remainingNums[1:])
exclude := calculateLIS(currentSubsequence, remainingNums[1:])
return max(include, exclude)
}
return calculateLIS(currentSubsequence, remainingNums[1:])
}

func lengthOfLIS(nums []int) int {
longestLength := 0

for i := 0; i < len(nums)-1; i++ {
currentLength := calculateLIS([]int{nums[i]}, nums[i:])
if currentLength > longestLength {
longestLength = currentLength
}
}

return longestLength
}

然后不出意外的超时了,时间复杂度是 O(n^2),以下面的代码为例:

1
2
3
4
5
6
func main() {
// Example usage
nums := []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95, 96, 97, 98, 99, 100, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 117, 118, 119, 120, 121, 122, 123, 124, 125, 126, 127, 128, 129, 130, 131, 132, 133, 134, 135, 136, 137, 138, 139, 140, 141, 142, 143, 144, 145, 146, 147, 148, 149, 150, 151, 152, 153, 154, 155, 156, 157, 158, 159, 160, 161, 162, 163, 164, 165, 166, 167, 168, 169, 170, 171, 172, 173, 174, 175, 176, 177, 178, 179, 180, 181, 182, 183, 184, 185, 186, 187, 188, 189, 190, 191, 192, 193, 194, 195, 196, 197, 198, 199, 200, 201, 202, 203, 204, 205, 206, 207, 208, 209, 210, 211, 212, 213, 214, 215, 216, 217, 218, 219, 220, 221, 222, 223, 224, 225, 226, 227, 228, 229, 230, 231, 232, 233, 234, 235, 236, 237, 238, 239, 240, 241, 242, 243, 244, 245, 246, 247, 248, 249, 250, 251, 252, 253, 254, 255, 256, 257, 258, 259, 260, 261, 262, 263, 264, 265, 266, 267, 268, 269, 270, 271, 272, 273, 274, 275, 276, 277, 278, 279, 280, 281, 282, 283, 284, 285, 286, 287, 288, 289, 290, 291, 292, 293, 294, 295, 296, 297, 298, 299, 300, 301, 302, 303, 304, 305, 306, 307, 308, 309, 310, 311, 312, 313, 314, 315, 316, 317, 318, 319, 320, 321, 322, 323, 324, 325, 326, 327, 328, 329, 330, 331, 332, 333, 334, 335, 336, 337, 338, 339, 340, 341, 342, 343, 344, 345, 346, 347, 348, 349, 350, 351, 352, 353, 354, 355, 356, 357, 358, 359, 360, 361, 362, 363, 364, 365, 366, 367, 368, 369, 370, 371, 372, 373, 374, 375, 376, 377, 378, 379, 380, 381, 382, 383, 384, 385, 386, 387, 388, 389, 390, 391, 392, 393, 394, 395, 396, 397, 398, 399, 400, 401, 402, 403, 404, 405, 406, 407, 408, 409, 410, 411, 412, 413, 414, 415, 416, 417, 418, 419, 420, 421, 422, 423, 424, 425, 426, 427, 428, 429, 430, 431, 432, 433, 434, 435, 436, 437, 438, 439, 440, 441, 442, 443, 444, 445, 446, 447, 448, 449, 450, 451, 452, 453, 454, 455, 456, 457, 458, 459, 460, 461, 462, 463, 464, 465, 466, 467, 468, 469, 470, 471, 472, 473, 474, 475, 476, 477, 478, 479, 480, 481, 482, 483, 484, 485, 486, 487, 488, 489, 490, 491, 492, 493, 494, 495, 496, 497, 498, 499, 500, 501, 502, 503, 504, 505, 506, 507, 508, 509, 510, 511, 512, 513, 514, 515, 516, 517, 518, 519, 520, 521, 522, 523, 524, 525, 526, 527, 528, 529, 530, 531, 532, 533, 534, 535, 536, 537, 538, 539, 540, 541, 542, 543, 544, 545, 546, 547, 548, 549, 550, 551, 552, 553, 554, 555, 556, 557, 558, 559, 560, 561, 562, 563, 564, 565, 566, 567, 568, 569, 570, 571, 572, 573, 574, 575, 576, 577, 578, 579, 580, 581, 582, 583, 584, 585, 586, 587, 588, 589, 590, 591, 592, 593, 594, 595, 596, 597, 598, 599, 600, 601, 602, 603, 604, 605, 606, 607, 608, 609, 610, 611, 612, 613, 614, 615, 616, 617, 618, 619, 620, 621, 622, 623, 624, 625, 626, 627, 628, 629, 630, 631, 632, 633, 634, 635, 636, 637, 638, 639, 640, 641, 642, 643, 644, 645, 646, 647, 648, 649, 650, 651, 652, 653, 654, 655, 656, 657, 658, 659, 660, 661, 662, 663, 664, 665, 666, 667, 668, 669, 670, 671, 672, 673, 674, 675, 676, 677, 678, 679, 680, 681, 682, 683, 684, 685, 686, 687, 688, 689, 690, 691, 692, 693, 694, 695, 696, 697, 698, 699, 700, 701, 702, 703, 704, 705, 706, 707, 708, 709, 710, 711, 712, 713, 714, 715, 716, 717, 718, 719, 720, 721, 722, 723, 724, 725, 726, 727, 728, 729, 730, 731, 732, 733, 734, 735, 736, 737, 738, 739, 740, 741, 742, 743, 744, 745, 746, 747, 748, 749, 750, 751, 752, 753, 754, 755, 756, 757, 758, 759, 760, 761, 762, 763, 764, 765, 766, 767, 768, 769, 770, 771, 772, 773, 774, 775, 776, 777, 778, 779, 780, 781, 782, 783, 784, 785, 786, 787, 788, 789, 790, 791, 792, 793, 794, 795, 796, 797, 798, 799, 800, 801, 802, 803, 804, 805, 806, 807, 808, 809, 810, 811, 812, 813, 814, 815, 816, 817, 818, 819, 820, 821, 822, 823, 824, 825, 826, 827, 828, 829, 830, 831, 832, 833, 834, 835, 836, 837, 838, 839, 840, 841, 842, 843, 844, 845, 846, 847, 848, 849, 850, 851, 852, 853, 854, 855, 856, 857, 858, 859, 860, 861, 862, 863, 864, 865, 866, 867, 868, 869, 870, 871, 872, 873, 874, 875, 876, 877, 878, 879, 880, 881, 882, 883, 884, 885, 886, 887, 888, 889, 890, 891, 892, 893, 894, 895, 896, 897, 898, 899, 900, 901, 902, 903, 904, 905, 906, 907, 908, 909, 910, 911, 912, 913, 914, 915, 916, 917, 918, 919, 920, 921, 922, 923, 924, 925, 926, 927, 928, 929, 930, 931, 932, 933, 934, 935, 936, 937, 938, 939, 940, 941, 942, 943, 944, 945, 946, 947, 948, 949, 950, 951, 952, 953, 954, 955, 956, 957, 958, 959, 960, 961, 962, 963, 964, 965, 966, 967, 968, 969, 970, 971, 972, 973, 974, 975, 976, 977, 978, 979, 980, 981, 982, 983, 984, 985, 986, 987, 988, 989, 990, 991, 992, 993, 994, 995, 996, 997, 998, 999, 1000, 1001, 1002, 1003, 1004, 1005, 1006, 1007, 1008, 1009, 1010, 1011, 1012, 1013, 1014, 1015, 1016, 1017, 1018, 1019, 1020, 1021, 1022, 1023, 1024, 1025, 1026, 1027, 1028, 1029, 1030, 1031, 1032, 1033, 1034, 1035, 1036, 1037, 1038, 1039, 1040, 1041, 1042, 1043, 1044, 1045, 1046, 1047, 1048, 1049, 1050, 1051, 1052, 1053, 1054, 1055, 1056, 1057, 1058, 1059, 1060, 1061, 1062, 1063, 1064, 1065, 1066, 1067, 1068, 1069, 1070, 1071, 1072, 1073, 1074, 1075, 1076, 1077, 1078, 1079, 1080, 1081, 1082, 1083, 1084, 1085, 1086, 1087, 1088, 1089, 1090, 1091, 1092, 1093, 1094, 1095, 1096, 1097, 1098, 1099, 1100, 1101, 1102, 1103, 1104, 1105, 1106, 1107, 1108, 1109, 1110, 1111, 1112, 1113, 1114, 1115, 1116, 1117, 1118, 1119, 1120, 1121, 1122, 1123, 1124, 1125, 1126, 1127, 1128, 1129, 1130, 1131, 1132, 1133, 1134, 1135, 1136, 1137, 1138, 1139, 1140, 1141, 1142, 1143, 1144, 1145, 1146, 1147, 1148, 1149, 1150, 1151, 1152, 1153, 1154, 1155, 1156, 1157, 1158, 1159, 1160, 1161, 1162, 1163, 1164, 1165, 1166, 1167, 1168, 1169, 1170, 1171, 1172, 1173, 1174, 1175, 1176, 1177, 1178, 1179, 1180, 1181, 1182, 1183, 1184, 1185, 1186, 1187, 1188, 1189, 1190, 1191, 1192, 1193, 1194, 1195, 1196, 1197, 1198, 1199, 1200, 1201, 1202, 1203, 1204, 1205, 1206, 1207, 1208, 1209, 1210, 1211, 1212, 1213, 1214, 1215, 1216, 1217, 1218, 1219, 1220, 1221, 1222, 1223, 1224, 1225, 1226, 1227, 1228, 1229, 1230, 1231, 1232, 1233, 1234, 1235, 1236, 1237, 1238, 1239, 1240, 1241, 1242, 1243, 1244, 1245, 1246, 1247, 1248, 1249, 1250, 1251, 1252, 1253, 1254, 1255, 1256, 1257, 1258, 1259, 1260, 1261, 1262, 1263, 1264, 1265, 1266, 1267, 1268, 1269, 1270, 1271, 1272, 1273, 1274, 1275, 1276, 1277, 1278, 1279, 1280, 1281, 1282, 1283, 1284, 1285, 1286, 1287, 1288, 1289, 1290, 1291, 1292, 1293, 1294, 1295, 1296, 1297, 1298, 1299, 1300, 1301, 1302, 1303, 1304, 1305, 1306, 1307, 1308, 1309, 1310, 1311, 1312, 1313, 1314, 1315, 1316, 1317, 1318, 1319, 1320, 1321, 1322, 1323, 1324, 1325, 1326, 1327, 1328, 1329, 1330, 1331, 1332, 1333, 1334, 1335, 1336, 1337, 1338, 1339, 1340, 1341, 1342, 1343, 1344, 1345, 1346, 1347, 1348, 1349, 1350, 1351, 1352, 1353, 1354, 1355, 1356, 1357, 1358, 1359, 1360, 1361, 1362, 1363, 1364, 1365, 1366, 1367, 1368, 1369, 1370, 1371, 1372, 1373, 1374, 1375, 1376, 1377, 1378, 1379, 1380, 1381, 1382, 1383, 1384, 1385, 1386, 1387, 1388, 1389, 1390, 1391, 1392, 1393, 1394, 1395, 1396, 1397, 1398, 1399, 1400, 1401, 1402, 1403, 1404, 1405, 1406, 1407, 1408, 1409, 1410, 1411, 1412, 1413, 1414, 1415, 1416, 1417, 1418, 1419, 1420, 1421, 1422, 1423, 1424, 1425, 1426, 1427, 1428, 1429, 1430, 1431, 1432, 1433, 1434, 1435, 1436, 1437, 1438, 1439, 1440, 1441, 1442, 1443, 1444, 1445, 1446, 1447, 1448, 1449, 1450, 1451, 1452, 1453, 1454, 1455, 1456, 1457, 1458, 1459, 1460, 1461, 1462, 1463, 1464, 1465, 1466, 1467, 1468, 1469, 1470, 1471, 1472, 1473, 1474, 1475, 1476, 1477, 1478, 1479, 1480, 1481, 1482, 1483, 1484, 1485, 1486, 1487, 1488, 1489, 1490, 1491, 1492, 1493, 1494, 1495, 1496, 1497, 1498, 1499, 1500, 1501, 1502, 1503, 1504, 1505, 1506, 1507, 1508, 1509, 1510, 1511, 1512, 1513, 1514, 1515, 1516, 1517, 1518, 1519, 1520, 1521, 1522, 1523, 1524, 1525, 1526, 1527, 1528, 1529, 1530, 1531, 1532, 1533, 1534, 1535, 1536, 1537, 1538, 1539, 1540, 1541, 1542, 1543, 1544, 1545, 1546, 1547, 1548, 1549, 1550, 1551, 1552, 1553, 1554, 1555, 1556, 1557, 1558, 1559, 1560, 1561, 1562, 1563, 1564, 1565, 1566, 1567, 1568, 1569, 1570, 1571, 1572, 1573, 1574, 1575, 1576, 1577, 1578, 1579, 1580, 1581, 1582, 1583, 1584, 1585, 1586, 1587, 1588, 1589, 1590, 1591, 1592, 1593, 1594, 1595, 1596, 1597, 1598, 1599, 1600, 1601, 1602, 1603, 1604, 1605, 1606, 1607, 1608, 1609, 1610, 1611, 1612, 1613, 1614, 1615, 1616, 1617, 1618, 1619, 1620, 1621, 1622, 1623, 1624, 1625, 1626, 1627, 1628, 1629, 1630, 1631, 1632, 1633, 1634, 1635, 1636, 1637, 1638, 1639, 1640, 1641, 1642, 1643, 1644, 1645, 1646, 1647, 1648, 1649, 1650, 1651, 1652, 1653, 1654, 1655, 1656, 1657, 1658, 1659, 1660, 1661, 1662, 1663, 1664, 1665, 1666, 1667, 1668, 1669, 1670, 1671, 1672, 1673, 1674, 1675, 1676, 1677, 1678, 1679, 1680, 1681, 1682, 1683, 1684, 1685, 1686, 1687, 1688, 1689, 1690, 1691, 1692, 1693, 1694, 1695, 1696, 1697, 1698, 1699, 1700, 1701, 1702, 1703, 1704, 1705, 1706, 1707, 1708, 1709, 1710, 1711, 1712, 1713, 1714, 1715, 1716, 1717, 1718, 1719, 1720, 1721, 1722, 1723, 1724, 1725, 1726, 1727, 1728, 1729, 1730, 1731, 1732, 1733, 1734, 1735, 1736, 1737, 1738, 1739, 1740, 1741, 1742, 1743, 1744, 1745, 1746, 1747, 1748, 1749, 1750, 1751, 1752, 1753, 1754, 1755, 1756, 1757, 1758, 1759, 1760, 1761, 1762, 1763, 1764, 1765, 1766, 1767, 1768, 1769, 1770, 1771, 1772, 1773, 1774, 1775, 1776, 1777, 1778, 1779, 1780, 1781, 1782, 1783, 1784, 1785, 1786, 1787, 1788, 1789, 1790, 1791, 1792, 1793, 1794, 1795, 1796, 1797, 1798, 1799, 1800, 1801, 1802, 1803, 1804, 1805, 1806, 1807, 1808, 1809, 1810, 1811, 1812, 1813, 1814, 1815, 1816, 1817, 1818, 1819, 1820, 1821, 1822, 1823, 1824, 1825, 1826, 1827, 1828, 1829, 1830, 1831, 1832, 1833, 1834, 1835, 1836, 1837, 1838, 1839, 1840, 1841, 1842, 1843, 1844, 1845, 1846, 1847, 1848, 1849, 1850, 1851, 1852, 1853, 1854, 1855, 1856, 1857, 1858, 1859, 1860, 1861, 1862, 1863, 1864, 1865, 1866, 1867, 1868, 1869, 1870, 1871, 1872, 1873, 1874, 1875, 1876, 1877, 1878, 1879, 1880, 1881, 1882, 1883, 1884, 1885, 1886, 1887, 1888, 1889, 1890, 1891, 1892, 1893, 1894, 1895, 1896, 1897, 1898, 1899, 1900, 1901, 1902, 1903, 1904, 1905, 1906, 1907, 1908, 1909, 1910, 1911, 1912, 1913, 1914, 1915, 1916, 1917, 1918, 1919, 1920, 1921, 1922, 1923, 1924, 1925, 1926, 1927, 1928, 1929, 1930, 1931, 1932, 1933, 1934, 1935, 1936, 1937, 1938, 1939, 1940, 1941, 1942, 1943, 1944, 1945, 1946, 1947, 1948, 1949, 1950, 1951, 1952, 1953, 1954, 1955, 1956, 1957, 1958, 1959, 1960, 1961, 1962, 1963, 1964, 1965, 1966, 1967, 1968, 1969, 1970, 1971, 1972, 1973, 1974, 1975, 1976, 1977, 1978, 1979, 1980, 1981, 1982, 1983, 1984, 1985, 1986, 1987, 1988, 1989, 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019, 2020, 2021, 2022, 2023, 2024, 2025, 2026, 2027, 2028, 2029, 2030, 2031, 2032, 2033, 2034, 2035, 2036, 2037, 2038, 2039, 2040, 2041, 2042, 2043, 2044, 2045, 2046, 2047, 2048, 2049, 2050, 2051, 2052, 2053, 2054, 2055, 2056, 2057, 2058, 2059, 2060, 2061, 2062, 2063, 2064, 2065, 2066, 2067, 2068, 2069, 2070, 2071, 2072, 2073, 2074, 2075, 2076, 2077, 2078, 2079, 2080, 2081, 2082, 2083, 2084, 2085, 2086, 2087, 2088, 2089, 2090, 2091, 2092, 2093, 2094, 2095, 2096, 2097, 2098, 2099, 2100, 2101, 2102, 2103, 2104, 2105, 2106, 2107, 2108, 2109, 2110, 2111, 2112, 2113, 2114, 2115, 2116, 2117, 2118, 2119, 2120, 2121, 2122, 2123, 2124, 2125, 2126, 2127, 2128, 2129, 2130, 2131, 2132, 2133, 2134, 2135, 2136, 2137, 2138, 2139, 2140, 2141, 2142, 2143, 2144, 2145, 2146, 2147, 2148, 2149, 2150, 2151, 2152, 2153, 2154, 2155, 2156, 2157, 2158, 2159, 2160, 2161, 2162, 2163, 2164, 2165, 2166, 2167, 2168, 2169, 2170, 2171, 2172, 2173, 2174, 2175, 2176, 2177, 2178, 2179, 2180, 2181, 2182, 2183, 2184, 2185, 2186, 2187, 2188, 2189, 2190, 2191, 2192, 2193, 2194, 2195, 2196, 2197, 2198, 2199, 2200, 2201, 2202, 2203, 2204, 2205, 2206, 2207, 2208, 2209, 2210, 2211, 2212, 2213, 2214, 2215, 2216, 2217, 2218, 2219, 2220, 2221, 2222, 2223, 2224, 2225, 2226, 2227, 2228, 2229, 2230, 2231, 2232, 2233, 2234, 2235, 2236, 2237, 2238, 2239, 2240, 2241, 2242, 2243, 2244, 2245, 2246, 2247, 2248, 2249, 2250, 2251, 2252, 2253, 2254, 2255, 2256, 2257, 2258, 2259, 2260, 2261, 2262, 2263, 2264, 2265, 2266, 2267, 2268, 2269, 2270, 2271, 2272, 2273, 2274, 2275, 2276, 2277, 2278, 2279, 2280, 2281, 2282, 2283, 2284, 2285, 2286, 2287, 2288, 2289, 2290, 2291, 2292, 2293, 2294, 2295, 2296, 2297, 2298, 2299, 2300, 2301, 2302, 2303, 2304, 2305, 2306, 2307, 2308, 2309, 2310, 2311, 2312, 2313, 2314, 2315, 2316, 2317, 2318, 2319, 2320, 2321, 2322, 2323, 2324, 2325, 2326, 2327, 2328, 2329, 2330, 2331, 2332, 2333, 2334, 2335, 2336, 2337, 2338, 2339, 2340, 2341, 2342, 2343, 2344, 2345, 2346, 2347, 2348, 2349, 2350, 2351, 2352, 2353, 2354, 2355, 2356, 2357, 2358, 2359, 2360, 2361, 2362, 2363, 2364, 2365, 2366, 2367, 2368, 2369, 2370, 2371, 2372, 2373, 2374, 2375, 2376, 2377, 2378, 2379, 2380, 2381, 2382, 2383, 2384, 2385, 2386, 2387, 2388, 2389, 2390, 2391, 2392, 2393, 2394, 2395, 2396, 2397, 2398, 2399, 2400, 2401, 2402, 2403, 2404, 2405, 2406, 2407, 2408, 2409, 2410, 2411, 2412, 2413, 2414, 2415, 2416, 2417, 2418, 2419, 2420, 2421, 2422, 2423, 2424, 2425, 2426, 2427, 2428, 2429, 2430, 2431, 2432, 2433, 2434, 2435, 2436, 2437, 2438, 2439, 2440, 2441, 2442, 2443, 2444, 2445, 2446, 2447, 2448, 2449, 2450, 2451, 2452, 2453, 2454, 2455, 2456, 2457, 2458, 2459, 2460, 2461, 2462, 2463, 2464, 2465, 2466, 2467, 2468, 2469, 2470, 2471, 2472, 2473, 2474, 2475, 2476, 2477, 2478, 2479, 2480, 2481, 2482, 2483, 2484, 2485, 2486, 2487, 2488, 2489, 2490, 2491, 2492, 2493, 2494, 2495, 2496, 2497, 2498, 2499, 2500}
result := lengthOfLIS(nums)
println(result)
}

而这其实可以考虑增加一些限制条件,当已经计算出的最大长度的子序列长度超过了剩下子串的长度时,就可以直接返回了。

1
2
3
4
5
6
7
8
9
10
11
func lengthOfLIS(nums []int) int {
longestLength := 1

for i := 0; i < len(nums)-1; i++ {
if longestLength >= len(nums) - i {
break
}
...
}
...
}

还有当 b0 子串的第一个元素比 a0 子串的最后一个元素大时,b 去掉该元素变为 b1,而 a0 有两种情况,一种是 a0 添加 b 子串的第一个元素变为 a1(子串长度加一),一种是 a0 子串保持不变。之后需要比较这两种情况才能得出最大值。而在源码中首先计算的是包含的情况 include := calculateLIS(append(currentSubsequence, remainingNums[0]), remainingNums[1:]) 这里其实可以在计算 include 之后就判断该情况得出的长度是否就是 a1 的长度加上 b1 的长度,如果是的话就可以直接返回了,因为这超过了另一种情况的最大长度(另一种情况的最大长度是 a0 的长度加上 b1 的长度 len(currentSubsequence) + len(remainingNums[1:]) )。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
func calculateLIS(currentSubsequence []int, remainingNums []int) int {

...

if remainingNums[0] > currentSubsequence[len(currentSubsequence)-1] {
include := calculateLIS(append(currentSubsequence, remainingNums[0]), remainingNums[1:])
if include == len(currentSubsequence) + len(remainingNums) {
return include
}
exclude := calculateLIS(currentSubsequence, remainingNums[1:])
return max(include, exclude)
}

...

}

动态规划

当然上面可以在特定情况下减少一些不必要的计算,但是并不能完全避免重复计算的问题。这里其实通过递归就可以画出二叉树的形式来表示所有的可能性。先以 nums[0] 为根结点,左子树为包含 nums[1] 的情况,右子树为不包含 nums[1] 的情况。然后在左子树中又以 nums[1] 为根结点,左子树为包含 nums[2] 的情况,右子树为不包含 nums[2] 的情况。这样就可以一直递归下去,直到没有元素可选为止。而当到达叶子结点时,就可以返回当前的子串长度。而当以 nums[1] 为根结点的情况下,左子树和右子树的情况其实是重复的。以 [0,1,0,3,2,3] 为例:

longest-increasing-subsequence-binary-tree

在上面这颗二叉树中,每一层就表示了每判断一个元素是否该放入子串中后子串最后一个元素的状态:

  • 第一层只有根结点这样 1 种状态;
  • 第二层有 2 种状态;
  • 第三层也只有 2 种状态;
  • 第四层有 2 种状态;
  • 第五层有三种状态(num[3]:3, num[1]:1, nums[0]:0);
  • 第六层有 2 种状态了(num[4]:2, num[3]:3)

如果遍历 nums 依次以数组中的每一个元素为根结点来构建这颗二叉树的话,就可以得到所有的可能性了。所以可以两个点来考虑重构之前的递归代码:

  • 把计算一个元素是否放入子串作为一个阶段,每一个阶段的状态存储下来,用于下一个阶段的计算;
  • 按照上面的二叉树状态来计算,每一个根结点都需要计算出所有的可能性。但除了以第一个节点作为根结点来计算外,后面所有的元素作为根节点的时候构建的状态树其实都是重复的,所以可以把之前计算过的状态存储下来,避免重复计算。

下面首先来实现用数组存储每一个阶段的状态。这里用一个 Subsequence 结构体来存储当前子串的最后一个元素和长度。然后用一个动态数组 dp 来存储每一个阶段的子串状态。每次遇到一个新的数字 nums[currentIndex],如果它可以扩展现有的子序列,就会向 dp 中追加一个新的 Subsequence

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
type Subsequence struct {
value int
length int
}

func lengthOfLIS(nums []int) int {
if len(nums) < 2 {
return len(nums)
}

maxLength := 1

for startIndex := 0; startIndex < len(nums)-1; startIndex++ {
dp := []Subsequence{{value: nums[startIndex], length: 1}}
for currentIndex := startIndex + 1; currentIndex < len(nums); currentIndex++ {
for _, subsequence := range dp {
if nums[currentIndex] <= subsequence.value {
continue
} else {
newLength := subsequence.length + 1
if newLength > maxLength {
maxLength = newLength
}
dp = append(dp, Subsequence{value: nums[currentIndex], length: newLength})
}
}
}
}

return maxLength
}

当然这个算法还不是动态规划,并且这个数组的内存占用是非常大的。主要原因如下:

  1. 动态数组 dp 的增长:在每次迭代中,dp 会存储当前起始索引 startIndex 下的所有可能的子序列状态。
    每次遇到一个新的数字 nums[currentIndex],如果它可以扩展现有的子序列,就会向 dp 中追加一个新的 Subsequence
    由于 dp 是一个动态数组,随着输入数组 nums 的长度增加,dp 的大小可能会呈指数级增长,尤其是在输入数组接近递增序列时。
    示例: 对于输入 [1, 2, 3, 4]:
    dp 会存储所有可能的子序列状态:
    • 长度为 1 的子序列:[1]
    • 长度为 2 的子序列:[1, 2], [1, 3], [1, 4]
    • 长度为 3 的子序列:[1, 2, 3], [1, 2, 4], [1, 3, 4]
    • 长度为 4 的子序列:[1, 2, 3, 4]
      这些子序列的状态会被存储在 dp 中,导致内存占用快速增加。
  2. 重复计算:每次从 startIndex 开始重新计算子序列,而没有利用之前计算的结果。例如,对于输入 [1, 2, 3, 4],从 startIndex = 0 开始计算时,已经计算了所有以 1 开头的子序列,但从 startIndex = 1 开始时,又会重新计算以 2 开头的子序列,导致大量重复计算和冗余存储。
  3. Subsequence 结构体的存储开销:每个 Subsequence 结构体存储两个字段:valuelength。这些结构体会被频繁创建并存储在 dp 中,进一步增加了内存占用。
  4. 嵌套循环的复杂性:外层循环遍历 startIndex,内层循环遍历 currentIndex,最内层循环遍历 dp。这种三层嵌套循环会导致算法的时间复杂度和空间复杂度都非常高,接近 O(n^3)

继续以优化该算法,要减少结构体的存储开销、嵌套循环的复杂行和动态数组的增长等问题,重新审视这个问题。其实从刚才的二叉树表示的状态图就可以看出,可以从动态规划的角度考虑问题。所以需要去思考状态转移的过程。

动态转移方程

动态规划最重要的就是要理解动态转移方程,而这个题的动态规划转移之所以一开始比较难理解,其核心就是要想到:所有的递增子序列都可以看作是一个以 nums[i] 为结尾的子串。假设 nums 这个数组的长度为 n,那么就可以表示为:

在上面的公式中,P_i 表示以 nums[i] 为结尾的递增子串的集合,P0 就表示以 nums[0] 为结尾的递增子串的集合,P1 就表示以 nums[1] 为结尾的递增子串的集合,依次类推。而 P 表示所有的递增子串的集合。也就是说,P 是所有 P_i 的并集。

这非常重要,因为之前的算法和思维定势会让人不自觉地想到要从头开始计算每一个子串的状态,也就是在不自觉中把所有的递增子序列都可以看作是一个以 nums[i] 为开头的子串。

但是如果以 nums[i] 为开头的话,就会出现当需要判定某个元素 nums[j] 能够添加到该子串的时候,仍然需要去比较 nums[j] 和该子串的最后一个元素的大小,所以每一个以 nums[i] 开头的子串都需要存储其末尾的元素值,最终就会额外的内存开销过大的问题。比如在上面的二叉树中,nums[:1]nums[:2] 都是以 nums[0] 为开头的子串,而 nums[1:2]nums[1:3] 都是以 nums[1] 为开头的子串,但每一个子串都需要存储其末尾的元素值。

而如果将 所有的递增子序列看作是一个以 nums[i] 为结尾的递增子串的集合 的话,问题就会迎刃而解。因为这样我们写出的动态规划方程就会非常简单且易于实现。求 nums 这个数组的最长递增子序列的长度的过程就可以转换为:比较每一个以 nums[i] 为结尾的最长递增子串的长度,求出其最大值即可。而求出每一个以 nums[i] 为结尾的最长递增子串的长度就可以通过动态转移方程来实现:

令长度为 i+1、j=i-1,也就是说数组最后一个元素为 nums[i]、倒数第二个元素为 nums[j]

  • 判断 nums[i] 这个元素是否可以添加到 nums[0:j] 这个数组的所有递增子串中(而判断 nums[i] 是否可以添加到 “nums[0:j] 这个数组的所有递增子串” 中,其实就是从以 nums[0],nums[1]nums[j] 为结尾的最长递增子串中去找出所有可以添加 nums[i] 的子串,也就是分别判断 nums[i] 是否大于这些子串的结尾元素 nums[0]nums[1]nums[j]);
  • 如果存在可以添加的子串的话,nums 这个数组的最长递增子序列长度就等于 “所有可添加元素的子串长度 + 1” 后的最大值;
  • 如果所有子串都不可以添加的话,nums 这个数组的最长递增子序列长度就等于 1 (也就是 nums[i] 这个元素本身);

所以这里就可以尝试写出状态转移方程:

根据状态转移方程,接下来就要思考动态规划的实现了:

  • 目标:存储每一个以 nums[i] 为结尾的最长递增子串的长度,然后求出其最大值,假设这个数组是 dp 的话,最后结果也就是 max(dp(i))
  • 从何处开始:因为这个问题的状态转移是从后往前进行的。也就是在计算“以 nums[i] 为结尾的最长递增子串”长度的时候,已经计算过“以 nums[:i-1] 这个数组的所有递增子序列”了,也就是说已经计算过 “以 nums[0], nums[1]nums[i-1] 为结尾的最长递增子串” 的长度了,所以就从 nums[0:1] 这个数组开始计算;
  • 如何判断:判断 nums[i] 是否可以添加到 nums[0:j] 这个数组的所有递增子串中,其实就是从以 nums[0],nums[1]nums[j] 为结尾的最长递增子串中去找出所有可以添加 nums[i] 的子串,也就是分别判断 nums[i] 是否大于这些子串的结尾元素 nums[0]nums[1]nums[j]
  • 如何存储:用一个数组 dp 来存储每一个以 nums[i] 为结尾的最长递增子串的长度,dp[i] 就表示以 nums[i] 为结尾的最长递增子串的长度;
  • 如何更新:如果 nums[i] 可以添加到 nums[0:j] 这个数组的所有递增子串中,那么就需要更新 dp[i] 的值,dp[i] = max(dp(k) + 1),也就是在 dp 中找到所有可以添加 nums[i] 的子串的长度,然后加上 1(也就是 nums[i] 本身);
  • 如何结束:当 nums 这个数组的所有元素都计算完毕后,就可以返回 max(dp) 了。

最后实现代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
func max(a, b int) int {
if a > b {
return a
}
return b
}

func lengthOfLIS(nums []int) int {
n := len(nums)
if n < 2 {
return n
}
dp := make([]int, n)
dp[0] = 1
maxLength := 1
for i := 1; i < n; i++ {
dp[i] = 1
for j := 0; j < i; j++ {
if nums[j] < nums[i] {
dp[i] = max(dp[i], dp[j]+1)
}
}
if dp[i] > maxLength {
maxLength = dp[i]
}
}
return maxLength
}

贪心算法&二分查找

在动态规划的实现中,时间复杂度是 O(n^2),而空间复杂度是 O(n)。但是这个题其实可以用贪心算法和二分查找来实现,时间复杂度可以降到 O(nlogn),空间复杂度也可以降到 O(n)

贪心算法的核心思想是:在每一步选择中都采取在当前状态下最好或最优的选择,从而希望通过每个阶段的最优选择达到全局的最优。而这个题的贪心算法就是要找到一个递增子序列的最小结尾值。也就是,要让每一个递增子序列的最后一个元素尽可能小,这样就可以为后面的元素留出更多的空间来扩展这个递增子序列。比如在 [10, 9, 2, 5, 3, 7, 101, 18] 中:

  • 长度为 1 的递增子串中,[10][9][2][5][3][7][101][18] 都是长度为 1 的递增子串,显然这些递增子串中末尾元素最小的就是 2,也就是 nums[2]
  • 长度为 2 的递增子串就可以以上一步的最优子串([2])为基础进行拓展,这其中中 [2, 5][2, 3][2, 7][2, 101][2, 18] 都是长度为 2 的递增子串。因为长度为 1 的递增子串中,最后一个元素的值越小,长度为 2 的递增子串中,最后一个元素的值可选范围就越大。那么此时选出长度为 2 的递增子串中最后一个元素最小的就是 5,也就是 nums[3],子串为 [2, 5]
  • 而长度为 3 的递增子串就可以以上一步的最优子串([2, 5])为基础进行拓展,这其中中 [2, 5, 7][2, 5, 101][2, 5, 18] 都是长度为 3 的递增子串。选出长度为 3 的递增子串中最后一个元素最小的就是 7,也就是 nums[5],子串为 [2, 5, 7]
  • 依次类推,最终得到的最长递增子序列就是 [2, 3, 7, 18],长度为 4

所以这里用贪心算法的核心就是要找到一个递增子序列的最小结尾值。也就是要让每一个递增子序列的最后一个元素尽可能小,这样就可以为后面的元素留出更多的空间来扩展这个递增子序列。

就编码而言,可以假设 tails 数组存储当前最长递增子序列的最小结尾值,且因为 tails 数组是有序的,那么就可以用二分查找来找到 tails 中第一个大于 num 的元素,然后更新 tails 中该元素为 num,而如果 num 大于 tails 中的所有元素的话,就扩展 tails 数组。最后,tails 数组的长度就是最长递增子序列的长度。下面是贪心算法和二分查找的实现:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
func lengthOfLIS(nums []int) int {
if len(nums) == 0 {
return 0
}

// 使用 tails 数组存储当前最长递增子序列的最小结尾值
tails := make([]int, 0)

// 遍历 nums 数组
for _, num := range nums {
left, right := 0, len(tails)
var mid int

// 二分查找 num 应插入的位置
for left < right {
mid = (left + right) >> 1
if tails[mid] < num {
// 如果 tails[mid] 小于 num,则在右半部分查找,也就是在 tails[mid+1:right] 中查找
// left = mid + 1 是因为 mid 已经被排除在外了,即便是 tails[mid] == num 也要排除在外,因为要找的是第一个大于 num 的元素
left = mid + 1
} else {
// 如果 tails[mid] 大于等于 num,则在左半部分查找,也就是在 tails[left:mid] 中查找
right = mid
}
}

// 如果 num 大于 tails 中的所有元素,则扩展 tails
if left >= len(tails) {
tails = append(tails, num)
} else {
// 否则更新 tails[left] 为 num
tails[left] = num
}
}

return len(tails)
}

最后更新 tails 的值是通过 left 来判断的,left 就是 tails 中第一个大于 num 的元素的下标,如果 left 大于等于 tails 的长度的话,就说明 num 大于 tails 中的所有元素了,就需要扩展 tails 数组。否则就更新 tails[left]num。这样就可以保证 tails 数组是有序的。