5711. 有界数组中指定下标处的最大值

本文介绍了一种利用二分查找解决特定序列问题的方法。针对一个由正整数组成的长度为n的序列,该序列中相邻元素之差不超过1,并且序列总和不超过maxSum的条件下,如何找到序列中最大值。文章详细解释了二分查找的过程及其应用,并给出了具体的实现代码。

思路

首先很容易想到,答案具有单调性。
所以做法就是二分答案
当前a[index]=mid的时候,如果整个序列总和小于等于maxSum,那么如果a[index]<mid的时候也一定满足。
那么我们就尝试增大一下mid,如果满足增大下限,不满足则减小上限。

根据题目条件,注意两点
1.都是正整数
2.相邻两个差值最多为1

那么我们知道当前a[index]=mid,就是尽可能让这个位置为最值点,其他位置尽量小。
我们去计算左边部分,左边可以放index+1个数字(包括第index这个位置)
那么其实就是往左边一直递减放就可以了。
如果减到了1,因为要为正整数,剩下位置也还是都是1就行。

右边也是同理计算。

代码

class Solution {
public:
    int maxValue(int n, int index, int maxSum) {
            int L=index+1,R=n-index;
            int l=0,r=1e9+1;
            while(l+1<r){
                int mid=l+r>>1;
                long long ans=0;
                if(L>=mid) ans+=1ll*mid*(mid+1)/2+L-mid;//能放的个数≥mid  剩下位置放1
                else ans+=1ll*L*(mid+mid-L+1)/2;//减不到1,即求[mid-L+1,mid]这个区间的和
                //下面右部分同理
                if(R>=mid) ans+=1ll*mid*(mid+1)/2+R-mid;
                else ans+=1ll*R*(mid+mid-R+1)/2;
                ans-=mid;
                if(ans<=maxSum) l=mid;
                else r=mid;

            }
        return l;
    }
};
先展示下效果 https://pan.quark.cn/s/a4b39357ea24 遗传算法 - 简书 遗传算法的理论是根据达尔文进化论而设计出来的算法: 人类是朝着好的方向(最优解)进化,进化过程中,会自动选择优良基因,淘汰劣等基因。 遗传算法(英语:genetic algorithm (GA) )是计算数学中用于解决最佳化的搜索算法,是进化算法的一种。 进化算法最初是借鉴了进化生物学中的一些现象而发展起来的,这些现象包括遗传、突变、自然选择、杂交等。 搜索算法的共同特征为: 首先组成一组候选解 依据某些适应性条件测算这些候选解的适应度 根据适应度保留某些候选解,放弃其他候选解 对保留的候选解进行某些操作,生成新的候选解 遗传算法流程 遗传算法的一般步骤 my_fitness函数 评估每条染色体所对应个体的适应度 升序排列适应度评估值,选出 前 parent_number 个 个体作为 待选 parent 种群(适应度函数的值越小越好) 从 待选 parent 种群 中随机选择 2 个个体作为父方和母方。 抽取父母双方的染色体,进行交叉,产生 2 个子代。 (交叉概率) 对子代(parent + 生成的 child)的染色体进行变异。 (变异概率) 重复3,4,5步骤,直到新种群(parentnumber + childnumber)的产生。 循环以上步骤直至找到满意的解。 名词解释 交叉概率:两个个体进行交配的概率。 例如,交配概率为0.8,则80%的“夫妻”会生育后代。 变异概率:所有的基因中发生变异的占总体的比例。 GA函数 适应度函数 适应度函数由解决的问题决定。 举一个平方和的例子。 简单的平方和问题 求函数的最小值,其中每个变量的取值区间都是 [-1, ...
答案: 1.构造随机数组: #include <stdio.h> #include <stdlib.h> #include <time.h> int main() { int n, i, j, temp, flag; int arr[100]; srand((int)time(NULL)); //用系统时间初始化rand()函数 printf("请输入n的值:"); scanf("%d", &n); for (i = 0; i < n; i++) { flag = 1; temp = rand() % 100 + 1; //生成1-100内的随机数 for (j = 0; j < i; j++) { if (arr[j] == temp) { //如果与已生成的元素重复,重新生成 flag = 0; break; } } if (flag) { arr[i] = temp; } else { i--; } } printf("生成的随机数组为:\n"); for (i = 0; i < n; i++) { printf("%d ", arr[i]); } printf("\n"); return 0; } 2.使用指针法输出数组元素值: #include <stdio.h> int main() { int arr[] = {1, 2, 3, 4, 5}; int* p; for (p = arr; p < arr + 5; p++) { printf("%d ", *p); } printf("\n"); return 0; } 3.数组中所有偶数求和: #include <stdio.h> int main() { int arr[] = {1, 2, 3, 4, 5}; int n = sizeof(arr) / sizeof(arr[0]); int i, sum = 0; for (i = 0; i < n; i++) { if (arr[i] % 2 == 0) { sum += arr[i]; } } printf("所有偶数的和为%d\n", sum); return 0; } 4.数组中所有奇数求和: #include <stdio.h> int main() { int arr[] = {1, 2, 3, 4, 5}; int n = sizeof(arr) / sizeof(arr[0]); int i, sum = 0; for (i = 0; i < n; i++) { if (arr[i] % 2 == 1) { sum += arr[i]; } } printf("所有奇数的和为%d\n", sum); return 0; } 5.数组中的所有数扩大k倍: #include <stdio.h> int main() { int arr[] = {1, 2, 3, 4, 5}; int n = sizeof(arr) / sizeof(arr[0]); int i, k; printf("请输入k的值:"); scanf("%d", &k); for (i = 0; i < n; i++) { arr[i] *= k; } printf("扩大%d倍后的数组为:\n", k); for (i = 0; i < n; i++) { printf("%d ", arr[i]); } printf("\n"); return 0; } 6.数组中所有偶数位置的元素加起来: #include <stdio.h> int main() { int arr[] = {1, 2, 3, 4, 5}; int n = sizeof(arr) / sizeof(arr[0]); int i, sum = 0; for (i = 0; i < n; i += 2) { if (arr[i] % 2 == 0) { sum += arr[i]; } } printf("偶数位置的元素加起来为%d\n", sum); return 0; } 7.找出数组中的最大值: #include <stdio.h> int main() { int arr[] = {1, 2, 3, 4, 5}; int n = sizeof(arr) / sizeof(arr[0]); int i, max = arr[0]; for (i = 1; i < n; i++) { if (arr[i] > max) { max = arr[i]; } } printf("数组中的最大值为%d\n", max); return 0; } 8.找出数组中的最小值: #include <stdio.h> int main() { int arr[] = {1, 2, 3, 4, 5}; int n = sizeof(arr) / sizeof(arr[0]); int i, min = arr[0]; for (i = 1; i < n; i++) { if (arr[i] < min) { min = arr[i]; } } printf("数组中的最小值为%d\n", min); return 0; } 9.n在数组中搜索指定元素,看其是否存在: #include <stdio.h> int main() { int arr[] = {1, 2, 3, 4, 5}; int n = sizeof(arr) / sizeof(arr[0]); int i, x; printf("请输入要搜索的元素:"); scanf("%d", &x); for (i = 0; i < n; i++) { if (arr[i] == x) { printf("%d存在于数组中\n", x); return 0; } } printf("%d不存在于数组中\n", x); return 0; }
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

我不会c语言

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值