二进制加法
二进制加法
原题链接:https://leetcode.cn/problems/JFETK5/
位运算是将数字用二进制形式表示之后,对每位上0或1的运算,二进制及其位运算是现代计算机科学的基石,
位运算种类:非~、与&、或|、异或^、左移<<、右移>>
按位与&:对应二进制位上,a与b必须同时为真
按位或|:对应二进制进位上,a与b有一边为真即可
异或运算^:对应二进制位上,相同则为0,不同则为1(逆运算首先满足交换律)
关于异或运算的补充:
异或运算是逆运算:由 a ^ b = c 可得 c ^ a = b; c ^ b = a;
相同的数异或运算为0:n ^ n = 0 ;
任何数与0异或运算值不变:0 ^ n = n;
问:在文件中有一堆整数,每一次数都出现了两次,其中有一个值只出现了一次,如何快速找到只出现一次的这个数的值?
答:利用异或运算解决,将所有整数互相异或,最后得到的值即为只出现一次的值
左移 m << n :二进制数整 m 左移n 位
右移 m >> n:二进制数整 m 右移 n 位
运算
按位与&
0 & 0 = 0
1 & 0 = 0
0 & 1 = 0
1 & 1 = 1
按位或|
0 | 0 = 0
1 | 0 = 1
0 | 1 = 1
1 | 1 = 1
按位异或^
0 ^ 0 = 0
1 ^ 0 = 1
0 ^ 1 = 1
1 ^ 1 = 0
1.数制转换主要思路:将输入的二进制转换为10进制,使用10进制进行运算,将得到的结果转化回二进制即可。
存在问题:outside the range of representable values of type ‘int’,将int类型改为long long 无法存储,
1234567891011121314151617181920212223242526272829303132333435class Solution {public: string _ ...
整数除法
整数除法
原题链接:https://leetcode.cn/problems/xoh6Oh/
编程语言可能会提供占据不同内存空间的整数类型,每种类型能表示的整数范围不同,
无符号整数无论二进制表示的最高位是0还是1,其都表示一个正数,32位整数的范围为 ~ 。由于整数范围的限制,如果计算结果超出了范围,就会产生溢出。产生溢出时运行不会出错,但是计算结果可能会出乎意料。
1.减法主要思路:为了求得dividend/divisor,可以不断的从dividend里减去divisor,当不能再减去更多的divisor时,循环执行的次数就是商
存在问题:
当除数很大时,减法操作执行的次数就会很多,导致程序超时,如
没有考虑除数与被除数,在32位int数据类型的范围溢出问题
没有考虑除法结果,在32位int数据类型的范围溢出问题
12345678910111213141516class Solution {public: int divide(int dividend, int divisor) { int flag = 0; int count = 0; if (dividend < 0 && divisor > 0 || dividend > 0 && divisor < 0) flag = 1; dividend = abs(dividend); divisor = abs(divisor); while (dividend - divisor >= 0) { dividend -= divisor; count++; } if (flag) count = -count; return count; }};
2.减法的优化主要思路:
当被除数大于除数时,继续比较判断被除数是否大于除数的2倍,如果是则继续判断是否大于除数的4倍、8倍、… 倍,
如果被除数最多大于除数的 倍,则将被除数减去除数的 倍,然后将剩余的被除数重复前面的步骤(由于每次 ...
奶牛碑文
奶牛碑文
原题链接:https://oj.haizeix.com/problem/516
约翰和他的奶牛在大草原漫游,在一块石头上发现了一些有趣的碑文。碑文似乎是一个神秘古老的语言,只包括三个大写字母 C,O,W。
尽管约翰看不懂,但是令他高兴的是,C,O,W的顺序形式构成了一句他最喜欢的奶牛单词 “COW”。
现在,他想知道有多少次 COW出现在文本中。
如果 COW内穿插了其他字符,只要COW字符出现在正确的顺序,约翰也不介意。
他也不介意出现不同的COW共享一些字母。例如CWOW出现了1次COW,CCOW算出现了2次COW,CCOOWW算出现了8次COW。
输入:
第一行一个整数 N。(1≤N≤105)
第二行为含有 N 个字符的字符串,字符只可能是 C,O,W。
输出:
输出 COW 作为输入字符串的子串出现的次数(不一定是连续的),答案可能会很大。
样例:
126COOWWW
16
参考代码
暴力法:三重for循环分别对C,O,W进行遍历,可能会导致超时。
利用前缀和思想:空间复杂度换时间复杂度,
1234567891011121314151617181920212223242526272829#include<iostream>using namespace std;char str[100005];int numc[100005], numw[100005], n;long long ans;int main() { cin >> n >> str; for (int i = 0; str[i]; ++i) {//从前向后遍历 计算有多少个C(前缀和C) if (i == 0) { numc[i] = (str[i] == 'C'); } else { numc[i] = numc[i - 1] + (str[i] == 'C'); } } for (int i = n - 1; i >= 0; --i) {//从后向前遍历 计算有多少个W(后缀和W) if (i == n - 1) { numw[i] = (str[i] == 'W' ...
三角形个数
三角形个数
原题链接:https://oj.haizeix.com/problem/517
输入一根木棒的长度 n,将其分成三段,每段的长度是正整数,输出由这三小段木棒组成的不一样的三角形个数。
输入:
第一行一个整数 n。(1 ≤ n ≤ 104)
输出:
输出组成的不一样的三角形的个数。
样例:
110
12
样例说明:
两个能组成的三角形边长分别为 2,4,4 和 3,3,4。
参考代码123456789101112131415161718#include<iostream>using namespace std;int main() { int ans = 0; int n; cin >> n; //注意枚举范围 for (int i = 1; i <= n / 3; ++i) {//枚举最小边 for (int j = i; j <= (n - i) / 2; ++j) {//枚举次小边 if ((i + j) > (n - i - j)) {//判断最长边的合法性 ans++; //cout << i << " " << j << " " << n - i - j << endl; } } } cout << ans << endl; return 0;}
优雅数
优雅数
原题链接:https://oj.haizeix.com/problem/519
优雅枚举
给定两个数 L,R 求 L,R 之间(含)有多少个“优雅数”。
优雅数的定义:把一个数看做一个长度为 n 的字符串,n 个字符中 n−1 个全相同,有且仅有一个字符不同。例如 33323,119 都是优雅的,99999, 2332 都是不优雅的。
输入:
输入一行两个整数 L, R。(100 ≤ L ≤ R ≤ 1016)
输出:
输出两数间优雅数的个数。
样例:
1110 133
113
样例说明:
1110,112,113,114,115,116,117,118,119,121,122,131,133
参考代码
思路1:暴力法,遍历两数区间,直接对优雅数进行判定。
思路2:构造优雅数,将区间之内的所有优雅数直接构造出来,判定是否构造的优雅数是否在给定区间范围内。
单独的数字是几
多个的数字是几
优雅数的长度
单独的数字在优雅数中的位置(构造优雅数)
若一堆数为0的话,单独的数字必须要在优雅数的首位(不允许前导零)
若一个数为0的话,单独的0数字必须不能在首位(不允许前导零)
123456789101112131415161718192021222324252627282930#include<iostream>using namespace std;int main() { long long ans = 0; long long a, b; cin >> a >> b; for (int i = 0; i < 10; ++i) {//枚举单个的数字 for (int j = 0; j < 10; ++j) {//枚举多个的数字 if (i == j) continue; for (int k = 3; k < 18; ++k) {//枚举优雅数的数字长度 for (int l = 1; l <= k; ++l) {//枚举单独的数字在优雅数中的位置 if (i == 0 && l == 1) continue; if (j == 0 && l != 1) bre ...
火柴棒等式
火柴棒等式
原题链接:https://oj.haizeix.com/problem/514
给出 n 根火柴棒,可以拼出多少个形如 a + b = c 的等式。
等式中的 a,b,c 是用火柴棒拼出的整数(不能有前导零),数字和符号使用的火柴棒数量如下:
1234567891011120:61:22:53:54:45:56:67:38:79:6+:2=:2
有以下注意事项:
加号和等号各自需要两根火柴棒
如果 a ≠ b,则 a + b = c 和 b + a = c 视为不同的等式,a,b,c 均不小于 0
n 根火柴棒必须全部用上。
输入:
共一行一个整数 n。(1 ≤ n ≤ 24)
输出:
输出能组成的等式数。
样例:
114
12
样例说明:两个等式如下
120 + 1 = 11 + 0 = 1
参考代码难点:枚举范围的确定,取火柴消耗数量最少的数字1举例(使得相加数值最大,由此确定最大范围),最大为1111+1111=2222。
123456789101112131415161718192021222324252627282930#include<iostream>using namespace std;int n;//火柴数量int ans;int num[10] = {6, 2, 5, 5, 4, 5, 6, 3, 7, 6};int func(int x) { int count = 0; if (x == 0) return 6; while (x) { count += num[x % 10]; x /= 10; } return count;} int main() { cin >> n; for (int i = 0; i <= 2000; ++i) { for (int j = 0; j <= 2000; ++j) { if ((func(i) + func(j) + func(i + j) + 4) == n) { ans++; cout << i << " + " << ...
比例简化
比例简化
原题链接:https://oj.haizeix.com/problem/515
在社交媒体上,经常会看到针对某一个观点同意与否的民意调查以及结果。例如,对某一观点表示支持的有 1498 人,反对的有 902 人,那么赞同与反对的比例可以简单的记为 1498:902
不过,如果把调查结果就以这种方式呈现出来,大多数人肯定不会满意。因为这个比例的数值太大,难以一眼看出它们的关系。对于上面这个例子,如果把比例记为 5:3,虽然与真实结果有一定的误差,但依然能够较为准确地反映调查结果,同时也显得比较直观。
现给出支持人数 A,反对人数 B,以及一个上限 L,请你将 A 比 B 化简为 A’ 比 B’,
要求在 A’ 和 B’ 均不大于L,且 A’ 和 B’ 互质的前提下,
A’ / B’ ≥ A / B
且 A’ / B’ − A / B的值尽可能小
输入:
共一行三个整数 A, B, L,每两个整数之间用一个空格隔开,分别表示支持人数、反对人数以及上限
输出:
共一行两个整数 A’,B’,中间用一个空格隔开,表示化简后的比例
样例:
11498 902 10
15 3
参考代码==思路==:
若从1开始枚举,
如果枚举出一个答案 A’,B’(互质),当在继续枚举出一个新的答案 A’,B’(不互质),必然存在一个原来就有的A’,B’(互质)
并且新的答案与原来的答案,比例应该是一样的。
即使用两个暴力for循环遍历,保证答案满足A’ / B’ ≥ A / B ,且A’ / B’ − A / B的值尽可能小即可。
1
3.OnlineJ516:奶牛碑文约翰和他的奶牛在大草原漫游,在一块石头上发现了一些有趣的碑文。碑文似乎是一个神秘古老的语言,只包括三个大写字母 C,O,W。
尽管约翰看不懂,但是令他高兴的是,C,O,W的顺序形式构成了一句他最喜欢的奶牛单词 “COW”。
现在,他想知道有多少次 COW出现在文本中。
如果 COW内穿插了其他字符,只要COW字符出现在正确的顺序,约翰也不介意。
他也不介意出现不同的COW共享一些字母。例如CWOW出现了1次COW,CCOW算出现了2次COW,CCOOWW ...
独木舟
独木舟
原题链接:https://oj.haizeix.com/problem/503
一群人去旅行要租用独木舟,每艘独木舟最多乘两人,且所有独木舟有一个统一的载重限度。
给出独木舟的载重限度和每个人的体重,现求最少需要租用多少独木舟。
输入
第一行一个整数 w,表示独木舟的载重量。(80 ≤ w ≤ 200)
第二行一个整数 n,表示旅游人数。 (1 ≤ n ≤ 30000)
接下来 n 行,每行一个数表示 ai,即每个人的重量 (5 ≤ ai ≤ w)
输出
接下来 n 行,每行一个数表示 ai,即每个人的重量 (5 ≤ ai ≤ w)
样例:
12345678910111009902020305060708090
16
参考代码123456789101112131415161718192021222324252627282930313233#include<iostream>#include<algorithm>using namespace std;int weight[30005];int main() { //1.数据录入 int w;//独木舟载重量 int n;//旅游的人数 cin >> w >> n; for (int i = 0; i < n; ++i) cin >> weight[i]; sort(weight, weight + n); //2.进行数据处理 int count = 0; if (weight[0] * 2 > w) {//特殊情况处理 count = n; } else {//一般情况处理 for (int i = 0; i < n; ++i) { for (int j = n - 1; j >= 0; --j) { if (weight[j] == 0) continue; if ((weight[i] + weight[j]) <= w || i == j) { count++; weight[j] = 0; weight[i] = 0; break; } } } ...
删数
删数
原题链接:https://oj.haizeix.com/problem/504
输入一个高精度的正整数 n(长度不大于 240 位),
去掉其中任意 s 个数字后,剩下的数字按原左右次序将组成一个新的正整数,现求一种方案,使得新的正整数数值最小。
输入
第一行一个整数 n
第二行一个正整数 s
输出
输出一个数表示最小值,输出时忽略数字的前导零
样例:
123//样例11795664
115
123//样例29030713
11
参考代码
让小的数字在前面、大的数字在后面,使得整体的数字变小。
删除违反该规则的数字,
如果没有违反规则的数字,则删除最后一个数字(最大的数字)
12345678910111213141516171819202122232425262728293031323334#include<iostream>#include<string>using namespace std;int main() { string str; cin >> str; int n; cin >> n; //1.数据操作 for (int i = 0; i < n; ++i) { int ind = str.size() - 1;//默认删除最后一位-1 for (int j = 1; j < str.size(); ++j) { if (str[j - 1] > str[j]) {//优先删除违反规则的数字 ind = j - 1; break; } } str.replace(ind, 1, ""); } //2.前置零的特殊处理 int flag = 0; for (int i = 0; i < str.size(); ++i) { if (flag == 1) { cout << str[i]; } else if (str[i] != '0') { cout << str[i ...
最大整数
最大整数
原题链接:https://oj.haizeix.com/problem/505
现在有 n 个正整数,将他们连成一排,组成一个最大的整数。
例如,现在有三个整数 13, 312, 343,连接成最大整数为 34331213。
输入:
第一行一个整数 n。(1 ≤ n ≤ 100000)
第二行 n 个不超过 int 类型范围的正整数。
输出:
输出一个数表示组成的最大整数。
样例:
123121 12 96
19612121
参考代码思路:按照某种规则进行排序(前面的字符串 连接上 后面的字符串的字典序,大于后面的字符串连接上前面的字符串的字典序)
123456789101112131415161718192021#include<iostream>#include<algorithm>#include<string>using namespace std;int n;string str[100005];bool cmp(const string &str1, const string &str2) { //前面连接上后面的字典序 大于 后面连接上前面的字典序 return str1 + str2 > str2 + str1;}int main() { cin >> n; for (int i = 0; i < n; ++i) cin >> str[i]; sort(str, str + n, cmp); for (int i = 0; i < n; ++i) cout << str[i]; cout << endl; return 0;}