hdu5014 构造b数列使得t最大(小想法)

题意:
     给你一个序列a,他有n+1个数,每个数的范围是ai >= 0 && a[i] <= n,同时任意两个数字都是不相同的,就是ai != aj (i!=j),然后让你构造一个序列b,这个序列的要求与a序列一样,最后要求构造的b是使 t = (a0 ^ b0) + (a1 ^ b1) + ...+ (an ^ bn)最大。

思路:

      一开始想的太简单了,因为题目要求的是每个数字都只能用一次,一开始我的想法是根据n的奇偶,来单独处理,处理的时候也是根据每个ai的奇偶来处理的,结果wa了,现在来说下正确的解法吧,这个题目我用的正确解法有点贪心的意思,我们每次先找到一个最大的2^k的数字,这个数字要小于等于n,从它开始枚举,往两边扩,如果不能扩了就找到当前没有被找出来的最大的那个2^k继续扩,不怎么好解释,自己在纸上多写几个连续的二进制数,然后模拟下很容易懂的,这里面有一个很容易让人怀疑的地方就是可能大数的那个方向会有剩余,其实不用担心这个,不可能有剩余,因为如果以某一个数为中心往两侧扩的话,只要前面扩到头了,也就是到1了,那么后面一定是mark完的了,因为扩到1相当于自己*2了,也就是2^(k+1)了,要是没有看懂什么意思,那么就自己找几个连续的二进制数模拟下就行了。


#include<stdio.h>
#include<string.h>
#include<stack>

#define N 110000

using namespace std;

int mark[N];
int num[N] ,Ans[N];

int main ()
{
   int n ,i ,tmp;
   while(~scanf("%d" ,&n))
   {
      for(i = 0 ;i <= n ;i ++)
      scanf("%d" ,&num[i]);
      stack<int>my_sk;
      tmp = 1;
      while(1)
      {
         my_sk.push(tmp);
         tmp <<= 1;
         if(tmp > n) break;
      }
      memset(mark ,0 ,sizeof(mark));
      memset(Ans ,0 ,sizeof(Ans));
      while(!my_sk.empty())
      {
         int L = my_sk.top() - 1;
         int R = L + 1;
         my_sk.pop();
         while(1)
         {
            if(L < 0 || R > n || mark[L] || mark[R])
            break;
            mark[L] = mark[R] = 1;
            Ans[L] = R ,Ans[R] = L;
            L -- ,R ++;
         }
      }
      __int64 ans = 0;
      for(i = 0 ;i <= n ;i ++)
      ans += (__int64)(num[i] ^ Ans[num[i]]);   
      printf("%I64d\n" ,ans);
      for(i = 0 ;i <= n ;i ++)
      if(i == n) printf("%d\n" ,Ans[num[i]]);
      else printf("%d " ,Ans[num[i]]);
   }
   return 0;
}

HDU2019的数列有序问题通常涉及到数组排序或搜索算法。这类题目一般会给出一个未排序的整数序列,然后需要检查这个序列是否能通过某种操作变得有序。常见的操作可能是交换两个元素、删除一个元素等。 例如,你可以考虑使用二分查找或者归并排序的思想。如果序列已经是升序排列,直接返回true;如果是降序排列,也需要检查能否通过一次交换将整个序列变为升序;对于其他情况,可以尝试从中间元素开始向两边遍历,看能否通过有限次的操作使序列有序。 下面是一个简单的Python示例,假设我们有一个函数`checkSorted(nums)`,它接受一个整数列表`nums`: ```python def checkSorted(nums): n = len(nums) # 如果只有一个元素或者已经有序 if n <= 1 or nums == sorted(nums): return True # 检查是否存在逆序对 for i in range(1, n): if nums[i] < nums[i - 1]: left, right = i, n - 1 while left < right: mid = (left + right) // 2 if nums[mid] > nums[i - 1]: left = mid + 1 else: right = mid # 如果找到了逆序对并且右边界小于等于左边界,说明可以通过一次交换修复 if right <= i - 1: nums[left], nums[i - 1] = nums[i - 1], nums[left] if checkSorted(nums): return True # 否则无法修复,返回false else: return False # 所有操作都尝试过了,还是有序的 return True # 测试 nums = [4, 2, 3, 1] # 这个例子应该返回True,因为可以通过一次交换变成升序 print(checkSorted(nums)) ``` 请注意,这只是一种基本思路,实际解题时可能需要根据题目给出的具体条件进行调整。如果你遇到具体的题目,请提供题目详细描述以便我能给出更精确的帮助。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值