码迷,mamicode.com
首页 > 其他好文 > 详细

【LeetCode】数学(共106题)

时间:2019-03-12 14:16:29      阅读:154      评论:0      收藏:0      [点我收藏+]

标签:精度   整数   括号   它的   假设   计算   完全   stl   其他   

【2】Add Two Numbers (2018年12月23日,review)

链表的高精度加法。

题解:链表专题:https://www.cnblogs.com/zhangwanying/p/9797184.html

 

【7】Reverse Integer (2018年12月23日, review)

给了一个32位的整数,返回它的reverse后的整数。如果reverse后的数超过了整数的范围,就返回 0. 

Example 1:
Input: 123
Output: 321

Example 2:
Input: -123
Output: -321

Example 3:
Input: 120
Output: 21

题解:见代码

技术图片
 1 class Solution {
 2 public:
 3     int reverse(int x) {
 4         long long num = x;
 5         bool nega = false;
 6         if (num < 0) {
 7             num = -num;
 8             nega = true;
 9         }
10         long long ret = 0;
11         while (num) {
12             int mod = num % 10;
13             ret = ret * 10 + mod;
14             num /= 10;
15         }
16         if (nega) {ret = -ret;}
17         if (ret > INT_MAX || ret < INT_MIN) {
18             ret = 0;
19         }
20         return (int) ret;
21     }
22 };
View Code

 

【8】String to Integer (atoi) (2018年12月23日,review)

按照题目给的规则实现 atoi

题解:corner cases 要注意,其他的没啥

技术图片
 1 class Solution {
 2 public:
 3     int myAtoi(string str) {
 4         const int n = str.size();
 5         if (n == 0) {return 0;}
 6         if (!isdigit(str[0]) && str[0] != - && str[0] != + && str[0] !=  ) {
 7             return 0;
 8         }
 9         int firstIdx = 0;
10         while (firstIdx < n && str[firstIdx] ==  ) {
11             ++firstIdx;
12         }
13         if (firstIdx == n || str[firstIdx] != + && str[firstIdx] != - && !isdigit(str[firstIdx])) { return 0; }
14         string number = string(1, str[firstIdx]);
15         int idx = firstIdx + 1;
16         while (idx < n && isdigit(str[idx])) {
17             number = number + string(1, str[idx]);
18             ++idx;
19         }
20         if (number == "-" || number == "+") {return 0;}
21         int startIdx = isdigit(number[0]) ? 0 : 1;
22         int p = startIdx;
23         while (p < number.size() && number[p] == 0) {
24             ++p;
25         }
26         number = number.substr(0, startIdx) + number.substr(p);
27         
28         if (number.size() > 11) {
29             if (number[0] == -) {
30                 return INT_MIN;
31             }    
32             return INT_MAX;
33         }
34         long long ret = 0;
35         for (int i = startIdx; i < number.size(); ++i) {
36             ret = ret * 10 + (number[i] - 0);    
37         }
38         if (number[0] == -) {
39             ret = -ret;
40         }
41         if (ret > INT_MAX) {
42             ret = INT_MAX;
43         } 
44         if (ret < INT_MIN) {
45             ret = INT_MIN;
46         }
47         return ret;
48     }
49 };
View Code

 

【9】Palindrome Number (2018年12月23日,review)

判断一个整数是不是回文。(不能转化成string)

题解:如果是负数,直接返回false。正数的话 reverse 一下,看reverse后是否相同。 

技术图片
 1 public:
 2     bool isPalindrome(int x) {
 3         if (x < 0) {return false;}
 4         long long num = x;
 5         long long reverse = reverseNum(num);
 6         return num == reverse;
 7     }
 8     long long reverseNum(long long num) {
 9         long long ret = 0;
10         while (num > 0) {
11             ret = ret * 10 + (num % 10);
12             num /= 10;
13         } 
14         return ret;
15     }
16 };
View Code

 

【12】Integer to Roman 

【13】Roman to Integer 

【29】Divide Two Integers 

 

【43】Multiply Strings (2018年11月27日,高精度乘法)(2019年1月23日复习,谷歌tag)

给了两个string类型的数字,num1 和 num2, 用string的形式返回 num1 * num2。(num1, num2 的长度都小于等于110)

题解:见 string 分类:https://www.cnblogs.com/zhangwanying/p/9885334.html

技术图片
 1 class Solution {
 2 public:
 3     string multiply(string num1, string num2) {
 4         size1 = num1.size(), size2 = num2.size();
 5         v1 = str2vecAndReverse(num1), v2 = str2vecAndReverse(num2);
 6         vector<int> v3(size1 + size2, 0);
 7         for (int i = 0; i < size1; ++i) {
 8             for (int j = 0 ; j < size2; ++j) {
 9                 v3[i+j] += v1[i] * v2[j];
10                 v3[i+j+1] += v3[i+j] / 10;
11                 v3[i+j] = v3[i+j] % 10;
12             }
13         }
14         string ret = "";
15         for (auto n : v3) {
16             ret = to_string(n) + ret;
17         }
18         auto p = ret.find_first_not_of(0);
19         if (p == string::npos) {
20             ret = "0";
21         } else {
22             ret = ret.substr(p);
23         } 
24         return ret;
25     }
26     vector<int> str2vecAndReverse(string num) {
27         int size = num.size();
28         vector<int> ret(size);
29         for (int i = size - 1; i >= 0; --i) {
30             ret[size-i-1] = num[i] - 0;
31         }
32         return ret;
33     }
34     int size1 = 0, size2 = 0;
35     vector<int> v1, v2;
36 };
View Code

 

【50】Pow(x, n)  (算法群,2018年10月25日)

求double 类型 x 的 n 次幂,注意 x 和 n的大小

  • -100.0 < x < 100.0
  • n is a 32-bit signed integer, within the range [−2^31, 2^31 − 1 ]

题解:快速幂的基本解法,时间复杂度是 O(logN)。本题需要注意 n 的极小值,当 n =  -INT_MIN 的时候,转换成 -n int 类型会溢出,因此 WA 了一次。

技术图片
 1 class Solution {
 2 public:
 3     double myPow(double x, int n) {
 4         double res = 0.0;
 5         long long N = (long long)n;  // 当 n = —INT_MIN 的时候,int会溢出
 6         if (N == 0) {return 1;}
 7         if (N < 0) {
 8             res = fastPow(x, -N);
 9             res = 1.0 / res;
10         } else {
11             res = fastPow(x, N);
12         }
13         return res;
14     }
15     // 快速幂 O(logN)
16     double fastPow(double x, long long n) {
17         if (n == 1) { return x; }
18         double temp = fastPow(x, n / 2);
19         return n & 1 ? temp * temp * x : temp * temp; 
20     }
21 };
View Code

 

【60】Permutation Sequence 

【65】Valid Number 

【66】Plus One 

【67】Add Binary 

【69】Sqrt(x)  (2018年10月26日, 算法群)

返回x的平方根的整数部分。

题解:直接二分法。(这题有人从 1 开始枚举都能过。)stl 的 lower_bound 自己实现了一个, 12ms

技术图片
 1 //lower_bound 的实现
 2 class Solution {
 3 public:
 4     int mySqrt(int x) {
 5         int left = 1, right = 1 << 16;
 6         int mid = 0;
 7         while (left < right) {
 8             mid = (left + right) >> 1;
 9             if ((long long)mid * mid < (long long)x) {
10                 left = mid + 1;
11             } else {
12                 right = mid;
13             }
14         }
15         return left * left == x ? left : left -1;
16     }
17 };
View Code

顺便提一下 lower_bound 和 upper_bound 的写法。

 1 int my_lower_bound(vector<int>& nums, int target) {
 2     int left = 0, right = nums.size();
 3     int mid = 0;
 4     while (left < right) {
 5         mid = (left + right) >> 1;
 6         if (nums[mid] < target) {
 7             left = mid + 1;
 8         } else {
 9             right = mid;
10         }
11     }
12     return left;
13 }
 1 int my_upper_bound(vector<int>& nums, int target) {
 2     int left = 0, right = nums.size();
 3     int mid = 0;
 4     while (left < right) {
 5         mid = (left + right) >> 1;
 6         if (nums[mid] > target) {
 7             right = mid;
 8         } else {
 9             left = mid + 1;
10         }
11     }
12     return left;
13 }

 

【149】Max Points on a Line (2018年11月10号,算法群)

给了 2D 平面上的 n 个点,两个点能组成一条直线,返回一条直线上最多能有几个点。

Input: [[1,1],[3,2],[5,3],[4,1],[2,3],[1,4]]
Output: 4
Explanation:
^
|
|  o
|     o        o
|        o
|  o        o
+------------------->
0  1  2  3  4  5  6

题解:就是暴力解法,我们先固定一个点,然后用其他点作为另外一个点,确定一条直线的斜率,有 y = kx+b, 有 k 的情况下再加上一个固定的点坐标,这条直线就是确定的。用hash-table来数一条线上最多几个点。

要AC还有两个点需要注意,(1)输入可能有完全重复的点坐标,(2)当有两个点(A, B)足够近但是他们两个距离第三个点(C)足够远的时候,这个时候AC 和 BC 的斜率可能会爆double,就是 A B C并不在同一条直线上,但是AC 和BC算出来的斜率一样。这种可用求gcd约分处理。

技术图片
 1 /**
 2  * Definition for a point.
 3  * struct Point {
 4  *     int x;
 5  *     int y;
 6  *     Point() : x(0), y(0) {}
 7  *     Point(int a, int b) : x(a), y(b) {}
 8  * };
 9  */
10 class Solution {
11 public:
12     int maxPoints(vector<Point>& points) {
13         const int n = points.size();
14         if (n < 3) {return n;}
15         map<pair<int, int>, int> slope_cnt; //在固定一个点的情况下,选择周围所有的点,计算斜率。
16         int ret = 0;
17         for (int i = 0; i < n - 1; ++i) {  //固定一个点
18             int numOfi = 0;
19             slope_cnt.clear();
20             Point p1 = points[i];
21             int samePoint = 0;  //与 p1 重合点的个数
22             for (int j = i + 1; j < n; ++j) { //选另外一个点
23                 Point p2 = points[j];
24                 pair<int, int> slope;
25                 if (p1.x == p2.x) { //没有斜率,两个点与y轴平行; 或者两个点重合
26                     if (p1.y == p2.y) { samePoint++; continue;}
27                     slope = make_pair(std::numeric_limits<int>::infinity(), std::numeric_limits<int>::infinity());
28                 } else {
29                     int dy = p1.y - p2.y, dx = p1.x - p2.x;
30                     int gcd = __gcd(dy, dx);
31                     slope = make_pair(dy/gcd, dx/gcd);
32                 }
33                 if (slope_cnt.find(slope) != slope_cnt.end()) {
34                     slope_cnt[slope]++;
35                 } else {
36                     slope_cnt[slope] = 2;
37                 }
38             }
39             numOfi = samePoint + 1;
40             for (auto p : slope_cnt) {
41                 numOfi = max(numOfi, p.second + samePoint);
42             }
43             ret = max(ret, numOfi);
44         }
45         return ret;
46     }
47 };
View Code

 

【166】Fraction to Recurring Decimal 

 

【168】Excel Sheet Column Title (2018年12月23日,review)

给了一个数字 n,返回 n 代表的 excel title。

For example:
    1 -> A
    2 -> B
    3 -> C
    ...
    26 -> Z
    27 -> AA
    28 -> AB 
    ...

Example 1:
Input: 1
Output: "A"

Example 2:
Input: 28
Output: "AB"

Example 3:
Input: 701
Output: "ZY"

 题解:26进制

技术图片
 1 class Solution {
 2 public:
 3     string convertToTitle(int n) {
 4         string ret  = "";
 5         while (n > 0) {
 6             n--;
 7             int mod = n % 26;
 8             char c = A + mod;
 9             ret = string(1, c) + ret;
10             n /= 26;
11         }
12         return ret;
13     }
14 };
View Code

 

【171】Excel Sheet Column Number (2018年12月23日,review)

与168题相反,本题是给了一个 string s 代表 excel 的title,问这个 s 是第几列。

题解:26进制

技术图片
 1 class Solution {
 2 public:
 3     int titleToNumber(string s) {
 4         int ret = 0;
 5         const int n = s.size();
 6         int pow = 1;
 7         for (int i = n-1; i >= 0; --i) {
 8             ret += pow * (s[i] - A + 1);
 9             pow *= 26;
10         }
11         return ret;
12     }
13 };
View Code

 

【172】Factorial Trailing Zeroes (2018年12月23日,周日,算法群)

给了一个数字 n,返回 n! 的结尾 0 有几个。

题解:结尾 0 的个数只和因子里面的 5 的个数有关系。比如 5! 后面有 1 个结尾 0, 10!后面有两个结尾 0。(其实和 2 的个数也有关系,但是 2 的个数肯定比 5 的个数多。)

技术图片
 1 class Solution {
 2 public:
 3     int trailingZeroes(int n) {
 4         int ret = 0;
 5         while (n > 0) {
 6             ret += n / 5;
 7             n /= 5;
 8         }
 9         return ret;
10     }
11 };
View Code

 

【202】Happy Number 

 

【204】Count Primes 

 

【223】Rectangle Area 

 

【224】Basic Calculator  (2018年11月16日,算法群)

给了一个字符串算式,里面有数字,有“+”, “-” , “ ” , 和 “(” , ")"。问算式最后结果是多少,返回int。

题解:本题的题解写在了 stack 专题里面:https://www.cnblogs.com/zhangwanying/p/9886577.html

 

【231】Power of Two (2018年11月16日,随手复习了一下)

给了一个整数 n, 返回 n 是不是2的幂。

题解:复习一把。注意 n 如果是负数直接返回false。还有括号能多加就多加,不然可能发生你意想不到的情况orz

技术图片
1 class Solution {
2 public:
3     bool isPowerOfTwo(int n) {
4         return n > 0 && (n & (n-1)) == 0;
5     }
6 };
View Code

 

【233】Number of Digit One 

 

【246】Strobogrammatic Number 

 

【247】Strobogrammatic Number II 

 

【248】Strobogrammatic Number III 

 

【258】Add Digits 

 

【263】Ugly Number 

 

【264】Ugly Number II 

 

【268】Missing Number 

 

【273】Integer to English Words (2019年2月12日)

把一个int类型的数字给读出来。

Input: 1234567891
Output: "One Billion Two Hundred Thirty Four Million Five Hundred Sixty Seven Thousand Eight Hundred Ninety One"

题解:从右边到左边,三个一段。(代码写的一般==)

技术图片
 1 class Solution {
 2 public:
 3     vector<string> list = {"", " Thousand", " Million", " Billion"};
 4     string numberToWords(int num) {
 5         initMp();
 6         int level = 0;
 7         string res = "";
 8         if (num == 0) {return mp[0];}
 9         while (num) {
10             int temp = num % 1000;
11             string tempAns = trans(temp);
12             if (!tempAns.empty()) { 
13                 tempAns += list[level]; 
14                 res = tempAns + (res.empty() ? res : " " + res);
15             }
16             num /= 1000;
17             ++level;
18         }
19         return res;
20     }
21     unordered_map<int, string> mp;
22     string trans(int num) {
23         string str = to_string(num);
24         if (str.size() < 3) { str = string((3 - str.size()), 0) + str; }
25         string res = "";
26         if (str[0] != 0) { res = mp[str[0]-0] + " Hundred"; }
27         if (str[1] != 0) {
28             int temp = 0;
29             if (str[1] == 1) {
30                 temp = (str[1]-0) * 10 + (str[2]-0);
31             } else {
32                 temp = (str[1]-0) * 10;
33             }
34             res += res.empty() ? mp[temp] : " " + mp[temp];
35         }
36         if (str[2] != 0 && str[1] != 1) {
37             res += res.empty() ?  mp[(str[2] - 0)] : " " + mp[(str[2] - 0)];
38         }
39         return res;
40     }
41     void initMp() {
42         mp[0] = "Zero", mp[1] = "One", mp[2] = "Two", mp[3] = "Three", mp[4] = "Four", mp[5] = "Five", mp[6] = "Six", 
43         mp[7] = "Seven", mp[8] = "Eight", mp[9] = "Nine", mp[10] = "Ten", mp[11] = "Eleven", mp[12] = "Twelve", 
44         mp[13] = "Thirteen", mp[14] = "Fourteen", mp[15] = "Fifteen", mp[16] = "Sixteen", mp[17] = "Seventeen", 
45         mp[18] = "Eighteen", mp[19] = "Nineteen",  mp[20] = "Twenty", mp[30] = "Thirty", mp[40] = "Forty", 
46         mp[50] = "Fifty", mp[60] = "Sixty", mp[70] = "Seventy", mp[80] = "Eighty", mp[90] = "Ninety";
47     }
48 };
View Code

 

【279】Perfect Squares 

这题 dp/bfs 出现过。

 

【296】Best Meeting Point (2018年11月22日,开始冲hard)

给了一个二维的 0/1 矩阵grid, grid[i][j] = 1 代表 1 位置上有个人,现在矩阵上至少有两个人,这几个人想挑一个点 (x, y) 相会,问矩阵上的哪个点到这几个人的曼哈顿距离和最近,返回最近的距离之和。

题解可以见 sort分类里面的题解:https://www.cnblogs.com/zhangwanying/p/9914941.html

 

【313】Super Ugly Number 

 

【319】Bulb Switcher 

 

【326】Power of Three 

 

【335】Self Crossing 

 

【343】Integer Break 

 

【356】Line Reflection 

 

【357】Count Numbers with Unique Digits 

 

【360】Sort Transformed Array 

 

【365】Water and Jug Problem 

 

【367】Valid Perfect Square  (算法群,2018年10月27日)

给了一个正整数 num,判断是不是完全平方数。不能直接开根号。

题解:(1)我的解法非常挫,把 root 从 1 开始遍历,如果 root * root == num 就是,否则 root++。 如果 root * root > num,就直接退出。

技术图片
 1 // O(N) 
 2 class Solution {
 3 public:
 4     bool isPerfectSquare(int num) {
 5         int root = 1;
 6         while (root <= num/2+1)  {
 7             if (root *  root == num) { return true; }
 8             else if ((long long)root * root > num) { return false; }
 9             root++;
10         }
11         return false;
12     }
13 };
View Code

(2)本题也可以用二分解。 (直接lower_bound)

技术图片
 1 //binary search 妈呀,WA了好几次。
 2 class Solution {
 3 public:
 4     bool isPerfectSquare(int num) {
 5         int left = 1, right = num / 2 + 2;
 6         int mid = 0;
 7         while (left < right) {
 8             mid = (left + right) / 2;
 9             long long temp = (long long) mid * mid;
10             if (temp < num) {
11                 left = mid + 1;
12             } else {
13                 right = mid;
14             }
15         }
16         return left * left == num;   
17     }
18 };
View Code

(3)数学方法,一个完全平方数等于从 1 开始的连续奇数和。

证明就是这张图: 技术图片

技术图片
 1 //奇数和
 2 class Solution {
 3 public:
 4     bool isPerfectSquare(int num) {
 5         int i = 1;
 6         while (num > 0) {
 7             num -= i;
 8             i += 2;
 9         }
10         return num == 0;
11     }
12 };
View Code

 

【368】Largest Divisible Subset 

 

【372】Super Pow 

 

【396】Rotate Function

 

【397】Integer Replacement (2019年2月3日,算法群打卡题)

给了一个 int n,如果 n 是奇数的话,就可以把 n 加一或者减一,如果 n 是偶数的话,就可以把 n 除以 2. 返回把 n 变成 1 的最小步数。

题解:我自己是用 bfs 解的。本题也能用 dfs 或者 数学解法。下面说下数学解法。如果说当前数字 n 的二进制结尾是 11,那么就把它加一,凑成 4 的倍数。(但是 3 是个例外)。如果是其他的奇数的话,就全部 减一。 

技术图片
 1 class Solution {
 2 public:
 3     int integerReplacement(int m) {
 4         int cnt = 0;    
 5         long long n = m;
 6         while (n > 1) {
 7             if (n & 1) {
 8                 if ((n & 2) && (n != 3)) {
 9                     n += 1;
10                 } else {
11                     n -= 1;
12                 }
13             } else {
14                 n >>= 1;
15             }
16             ++cnt;
17         }
18         return cnt;
19     }
20 };
View Code

 

【400】Nth Digit 

【413】Arithmetic Slices 

【415】Add Strings 

【423】Reconstruct Original Digits from English 

【441】Arranging Coins 

【453】Minimum Moves to Equal Array Elements 

【462】Minimum Moves to Equal Array Elements II 

【469】Convex Polygon 

【478】Generate Random Point in a Circle 

【483】Smallest Good Base 

 

【507】Perfect Number (2018年11月25日)

We define the Perfect Number is a positive integer that is equal to the sum of all its positive divisors except itself.

Now, given an integer n, write a function that returns true when it is a perfect number and false when it is not.

Example:
Input: 28
Output: True
Explanation: 28 = 1 + 2 + 4 + 7 + 14

Note: The input number n will not exceed 100,000,000. (1e8)

题解:我第一次Ac的时候是用了遍历到 num/2 这个范围,但是其实遍历到 sqrt(num) 这个范围就行了。(本来 beats 10%, 现在beats 100%)

技术图片
 1 class Solution {
 2 public:
 3     bool checkPerfectNumber(int num) {
 4         if (num == 0) {return false;}
 5         int summ = 0;
 6         for (int k = 1; k * k <= num; ++k) {
 7             if (num % k == 0) {
 8                 summ += k;
 9                 summ += k == num / k ? 0 : num / k;
10             }
11         }
12         return summ == 2 * num;
13     }
14 };
View Code

 

【517】Super Washing Machines 

【523】Continuous Subarray Sum 

【535】Encode and Decode TinyURL 

 

【537】Complex Number Multiplication (2018年11月27日)

给了两个字符串,代表两个 complex, 返回这两个complex 的乘积(用字符串表示)。

题解:先分解这两个字符串分解出实数和虚数部分,然后分别求乘积的实数部分和虚数部分。

技术图片
 1 class Solution {
 2 public:
 3     string complexNumberMultiply(string a, string b) {
 4         vector<int> num1 = split(a), num2 = split(b);
 5         if (num1.empty() || num2.empty()) {return "err";}
 6         vector<int> num(2);
 7         num[0] = num1[0] * num2[0] - num1[1] * num2[1];
 8         num[1] = num1[0] * num2[1] + num1[1] * num2[0];
 9         string ret = to_string(num[0]) + "+" + to_string(num[1]) + "i";
10         return ret;
11     }
12     vector<int> split(string s) {
13         vector<int> ans(2);
14         auto pos = s.find("+");
15         if (pos == string::npos) {
16             return vector<int>();
17         }
18         string s1 = s.substr(0, pos);
19         string s2 = s.substr(pos+1, s.size()-pos-1);
20         //printf("s1 = %s, s2 = %s \n", s1.c_str(), s2.c_str());
21         ans[0] = stoi(s1), ans[1] = stoi(s2);
22         return ans;
23     }
24 };
View Code

 

【553】Optimal Division 

【573】Squirrel Simulation 

【592】Fraction Addition and Subtraction 

【593】Valid Square 

【598】Range Addition II 

【625】Minimum Factorization 

【628】Maximum Product of Three Numbers 

 

【633】Sum of Square Numbers (算法群,2018年10月29日)

给了一个非负整数 c,判断他是否能表达成 两个整数的平方和。

Input: 5
Output: True
Explanation: 1 * 1 + 2 * 2 = 5

题解:我一开始想的是 2 pointers 直接搞一下,结果是直接AC了,后来看了下群里大家的答案,都是二分,所以我也写个二分好了。2 pointers 的时间复杂度是 O(sqrt(c)), 二分的时间复杂度是 O(sqrt(c) * log(c)) 

技术图片
 1 class Solution {
 2 public:
 3     bool judgeSquareSum(int c) {
 4         int p1 = 0, p2 = sqrt(c) + 1;
 5         int summ = p1 * p1 + p2 * p2;
 6         while (p1 <= p2) {
 7             if (summ == c) {
 8                 return true;
 9             } else if (summ > c) {
10                 p2--;
11             } else if (summ < c) {
12                 p1++;
13             }
14             summ = p1 * p1 + p2 * p2;
15         }
16         return false;
17     }
18 };
2 pointers 
技术图片
 1 class Solution {
 2 public:
 3     bool judgeSquareSum(int c) {
 4         for (int p1 = 0; p1 < sqrt(c) + 1; ++p1) {
 5             int target = c - p1 * p1;
 6             int left = 0, right = sqrt(target) + 1;
 7             int t = my_lower_bound(left, right + 1, target);
 8             if (t * t == target) {
 9                 return true;
10             }
11         }
12         return false;
13     }
14     int my_lower_bound(int left, int right, int target) {
15         while (left < right) {
16             int mid = (left + right) / 2;
17             if (mid * mid < target) {
18                 left = mid + 1;
19             } else {
20                 right = mid;
21             }
22         }
23         return left;
24     }
25 };
binary search

本题还有一种解法是用了「费马平方和定理」,有空可以研究一下。https://leetcode.com/problems/sum-of-square-numbers/solution/ (solution #5)

 

【634】Find the Derangement of An Array 

【640】Solve the Equation 

【645】Set Mismatch 

【651】4 Keys Keyboard 

【660】Remove 9 

【670】Maximum Swap 

【672】Bulb Switcher II 

【728】Self Dividing Numbers 

【753】Cracking the Safe 

【754】Reach a Number 

【775】Global and Local Inversions 

【780】Reaching Points 

 

【781】Rabbits in Forest (2019年2月14日,Medium)

在一个森林中,每只兔子都有一个颜色,给了一个数组代表兔子的子集,数组的一个元素 nums[i] = x 代表除了它本身,森林里面还有多少只兔子跟他一样颜色。返回一个数值,代表森林里面最少多少只兔子。

题解:if a rabbit answer x, then it will have x+1 rabbits in its group. now n rabbits answers x. 

if n % (x+1) == 0, we need n / (x+1) groups of x + 1 rabbits.

if n % (x+1) != 0, we need n / (x+1)  + 1 groups of x + 1 rabbits.

技术图片
 1 class Solution {
 2 public:
 3     int numRabbits(vector<int>& answers) {
 4         const int n = answers.size();
 5         unordered_map<int, int> mp;
 6         //统计每个兔子报数的频次。
 7         for (auto num : answers) { mp[num]++; }
 8         // if x rabbits answer x, then we have x + 1 rabbits have the same color.
 9         int res = 0;
10         for (auto& p : mp) {
11             int key = p.first, value = p.second;
12             int k = value / (key + 1);
13             res += k * (key + 1);
14             int left = value % (key + 1);
15             res += left > 0 ? (key + 1) : 0;
16         }
17         return res;
18     }
19 };
View Code

 

【782】Transform to Chessboard 

【789】Escape The Ghosts 

【794】Valid Tic-Tac-Toe State  (Oct 15th, 2018 每日一题)

通过字符串数组给定一个Tic-Tac-Toe(三连棋游戏,两人轮流在九格方盘上画‘X‘或者‘O‘,谁先把三个相同记号排成横线、直线、斜线,即是胜者)状态board。 返回True如果当且仅当这个状态是一个有效的状态。 board是3x3数组,包含字符" ", "X", "O"。" "字符代表空的格。 

Tic-Tac-Toe游戏规则:

    1. 玩家只能轮流在空格(" ")里面画字符。
    2. 第一个玩家总是画"X",第二个玩家总是画"O"。
    3. "X"和"O"只能画在空白的格里面,不能画在已经存在"O"和"X"的格里。
    4. 三个相同记号排成横线、直线、斜线,即游戏结束。
    5. 如果没有空的格,游戏也结束。
    6. 游戏结束不能再移动。 
Example 1:
Input: board = ["O  ", "   ", "   "]
Output: false
Explanation: The first player always plays "X".

Example 2:
Input: board = ["XOX", " X ", "   "]
Output: false
Explanation: Players take turns making moves.

Example 3:
Input: board = ["XXX", "   ", "OOO"]
Output: false

Example 4:
Input: board = ["XOX", "O O", "XOX"]
Output: true

题解:参考知乎:https://zhuanlan.zhihu.com/p/34216982

首先我们要先熟悉下Tic-Tac-Toe三连棋游戏规则,就是两人轮流在九格方盘上画‘X‘或者‘O‘,谁先把三个相同记号排成横线、直线、斜线,即游戏结束。 
那么我们从游戏规则中来找处所有不合理的状态。 
根据规则1和2,假设X的数目为countX, O的数目为countO,那么我们可以得到countX==countO,或者countX - countO == 1。 
根据游戏结束后则不能再画O和X,那么当countX==count时,如果存在三个X排成横线、直线、斜线,那么即是不合理的,因为X先画,当三个X排成横线、直线、斜线时, 此时游戏结束,不能再画O,所以O的数目应该比X的数目少1。 
当countX - countO == 1时,如果存在三个O排成横线、直线、斜线,那么是不合理的,因为当三个O排成横线、直线、斜线时,游戏结束,不能再画X,所以此时X的数目应该和O的数目相等。 

技术图片
 1 class Solution {
 2 public:
 3     bool validTicTacToe(vector<string>& board) {
 4         int cntX = 0, cntO = 0;
 5         
 6         //1. X的个数要么和O个数相同,要么X个数比O的个数多一个
 7         for (int i = 0; i < 3; ++i) {
 8             for (int j = 0; j < 3; ++j) {
 9                 board[i][j] == X ? cntX++ : board[i][j] == O ? cntO++ : 1;
10             }
11         }
12         if (cntX != cntO && cntX != cntO+1) {
13             return false;
14         }
15         
16         //2. 判断最后一个放棋的是X,还是O
17         if (cntX - cntO == 1) {
18             //X最后放的棋子,所以每行,每列,每个对角线O不能有三连。
19             if(check(board, O) == false) {
20                 return false;
21             }
22         } else if (cntX == cntO) {
23             //O最后放的棋子,所以每行,每列,每个对角线X不能有三连。
24             if(check(board, X) == false) {
25                 return false;
26             }
27         }
28         return true;
29     }
30     bool check(vector<string>& board, char c) {
31         for (int i = 0; i < 3; ++i) {
32             string target = string(3, c);
33             if (board[i] == target) { return false;} // rows
34             if (board[0][i] == board[1][i] && board[1][i] == board[2][i] && board[0][i] == c) {  return false; } // cols
35         }
36         if (board[0][0] == board[1][1] && board[0][0] == board[2][2] && board[0][0] == c) { return false; }
37         if (board[0][2] == board[1][1] && board[1][1] == board[2][0] && board[0][2] == c) { return false; }
38         return true;
39     }
40 };
View Code

 

【800】Similar RGB Color 

【805】Split Array With Same Average 

【810】Chalkboard XOR Game 

【812】Largest Triangle Area 

【829】Consecutive Numbers Sum 

 

【836】Rectangle Overlap (2019年3月12日)

判断两个矩形是否有重叠部分。重叠部分是说重叠的面积大于0,如果是顶点重合或者边重合,则不算重叠。如果这两个矩形重合就返回 true,否则返回 false。

我们用矩形的 左上角 和 右下角 坐标表示一个矩形。

题解:我们想一想如何判断两条 1D 的线段重叠。一定存在一个 x,

left1 < x < right1 && left2 < x < right2

======>  left1 < x < right2 && left2 < x < right1

======>  left1 < righ2 && left2 < right1

同理,变成2D之后,矩形的 x 方向和 y 方向分别是 独立的 1D。所以代码如下:

1 //如果是1D,判断两个线段是否相交,就是 l1 < r2 && l2 < l1
2 class Solution {
3 public:
4     bool isRectangleOverlap(vector<int>& rec1, vector<int>& rec2) {
5         return rec1[0] < rec2[2] && rec2[0] < rec1[2] && rec1[1] < rec2[3] && rec2[1] < rec1[3];
6     }
7 };

 

 

 

 

 

【858】Mirror Reflection 

 

【866】Prime Palindrome (2018年11月3日, 和 906 超级回文数一起复习了一下)

找出大于等于 N 的第一个回文素数。

题解:先介绍一下我的方法,怎么求 比 N 大的第一个回文数,然后判断它是不是素数,不是素数,就获取下一个回文数。

技术图片
  1 //先构造回文,再判断是不是素数,比 N 大的回文数怎么构造。
  2 class Solution {
  3 public:
  4     int primePalindrome(int N) {
  5         if (isPalindrome(N) && isPrime(N)) {
  6             return N;
  7         }
  8         int palin = N;
  9         do {
 10             //printf("palin = %d \n", palin);
 11             palin = getNextGreaterPalindrome(palin); 
 12             
 13         } while (!isPrime(palin));
 14         return palin;
 15     }
 16     bool isPalindrome(int num) {
 17         string strN = to_string(num);
 18         int start = 0, end = strN.size()-1;
 19         while (start < end) {
 20             if (strN[start] != strN[end]) {
 21                 return false;
 22             }
 23             start++, end--;
 24         }
 25         return true;
 26     }
 27     bool isPrime(int num) {
 28         if (num == 1) {return false;}
 29         for (int i = 2; i <= sqrt(num); ++i) {
 30             if (num % i == 0) {
 31                 return false;
 32             }
 33         }
 34         return true;
 35     }
 36     int getNextGreaterPalindrome(int num) {
 37         if (num <= 9) {
 38             return num == 9 ? 11 : num + 1;
 39         }
 40         string strNum = to_string(num);
 41         const int n = strNum.size();
 42         int halfIdx = (n - 1) / 2;  //n = 3, halfIdx = 1; n = 4, halfIdx = 1;
 43         string strHalf = "", strNew = "";
 44         if (n & 1) { //odd size
 45             int p1 = halfIdx - 1, p2 = halfIdx + 1;
 46             while (p1 >= 0 && p2 < n) {
 47                 if (strNum[p1] != strNum[p2]) {
 48                     break;
 49                 }
 50                 p1--, p2++;
 51             }
 52             if (p1 == -1 || strNum[p1] < strNum[p2]) {  //increase by 1
 53                 strHalf = strNum.substr(0, halfIdx + 1);
 54                 int intHalf = atoi(strHalf.c_str());
 55                 ++intHalf;
 56                 string strHalfNew = to_string(intHalf);
 57                 string t = strHalfNew.substr(0, strHalfNew.size()-1);
 58                 reverse(t.begin(), t.end());
 59                 strNew = strHalfNew + t;
 60                 
 61             } else {
 62                 strHalf = strNum.substr(0, halfIdx);
 63                 string t = strHalf;
 64                 reverse(t.begin(), t.end());
 65                 strNew = strHalf + strNum[halfIdx] + t;
 66             }
 67         } else { // even size
 68             int p1 = halfIdx, p2 = halfIdx + 1;
 69             while (p1 >= 0 && p2 < n) {
 70                 if (strNum[p1] != strNum[p2]) {
 71                     break;
 72                 }
 73                 p1--, p2++;
 74             }
 75             //printf("p1 = %d, p2 = %d ,strNum[p1] = %c, strNum[p2] = %c \n", p1, p2, strNum[p1], strNum[p2]);
 76             if (p1 == -1 || strNum[p1] < strNum[p2]) {  //increase by 1
 77                 strHalf = strNum.substr(0, halfIdx + 1);
 78                 int intHalf = atoi(strHalf.c_str());
 79                 ++intHalf;
 80                 string strHalfNew = to_string(intHalf);
 81                 string t = strHalfNew;
 82                 reverse(t.begin(), t.end());
 83                 
 84                 // 是否有进位。。。
 85                 if ((int)strHalfNew.size() != (int)strHalf.size()) {
 86                     strNew = strHalfNew + t.substr(1, (int)strHalfNew.size()-1);
 87                 } else {
 88                     strNew = strHalfNew + t;
 89                 }           
 90             } else {
 91                 strHalf = strNum.substr(0, halfIdx  + 1);
 92                 string t = strHalf; 
 93                 reverse(t.begin(), t.end());
 94                 strNew = strHalf  + t;
 95             }
 96         }
 97         
 98         //string -> int
 99         int res = atoi(strNew.c_str());
100         return res;
101     }
102     
103     
104 };
View Code

genNextPalindrome 这个函数应该可以写的更短一些。等到更新之后放上来。

2019年3月12日更新,如何生成比num大的下一个回文数字。这次写的更短了一些。

 

技术图片
 1 class Solution {
 2 public:
 3     int primePalindrome(int N) {
 4         if (!isPalindrome(N)) {
 5             N = genNextPalindrome(N);
 6         }
 7         do {
 8             if (isPrime(N)) { break; }
 9         } while (N = genNextPalindrome(N));
10         return N;
11     }
12     bool isPrime(int num) {
13         if (num <= 1) {return false;}
14         for (int k = 2; k <= sqrt(num); ++k) {
15             if (num % k == 0) {return false;}
16         }
17         return true;
18     }
19     bool isPalindrome(int N) {
20         string str = to_string(N);
21         int start = 0, end = str.size() -1 ;
22         while (start < end) {
23             if (str[start++] != str[end--]) {return false;}
24         }
25         return true;
26     }
27     int genNextPalindrome(int num) {
28         string str = to_string(num);
29         int size = str.size();
30         string strHalf = str.substr(0, (size + 1)/2);
31         int iHalf = stoi(strHalf);
32         set<int> candidate;
33         for (int k = 0; k <= 1; ++k) {
34             string temp = to_string(iHalf + k);
35             temp += size & 1? string(temp.rbegin()+1, temp.rend()) : string(temp.rbegin(), temp.rend()); 
36             candidate.insert(stoi(temp));
37         }
38         int size2 = strHalf.size();
39         int base = pow(10, size) + 1;
40         candidate.insert(base);
41         int diff = INT_MAX;
42         int res = -1;
43         for (auto& c : candidate) {
44             if (c > num && c - num < diff) {
45                 diff = c - num;
46                 res = c;
47             }
48         }
49         return res;
50     }
51 };
View Code

 

 

【868】Binary Gap 

【869】Reordered Power of 2 

【877】Stone Game 

【878】Nth Magical Number 

【883】Projection Area of 3D Shapes 

【885】Spiral Matrix III 

【887】Super Egg Drop 

【891】Sum of Subsequence Widths 

【892】Surface Area of 3D Shapes 

【899】Orderly Queue 

【LeetCode】数学(共106题)

标签:精度   整数   括号   它的   假设   计算   完全   stl   其他   

原文地址:https://www.cnblogs.com/zhangwanying/p/9790007.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!