LA2965侏罗纪(异或和为0的最大数字个数)
题意:
给你n个字符串,让你在里面找到一个字符串集合使得这些字符串中所有的字母出现的次数和为偶数,输出集合的最大个数,和ASCII最小的解。
思路:
考虑到每个字符串中所有的字符都是有大写字母组成的,我们可以把每个字符串都用一个26位长的二进制数表示,比如第一位表示A,那么当第一位为0的时候就是说明A出现了偶数次,1表示出现了奇数次(直接异或),那么我们要找到满足题意的集合也就是可以转化成我们在n个26位长的二进制数中找到最多的数字,使得他们异或一起最后等于0(等于0表示所有的字母都是偶数次),这样我们有两种方式,一种是我自己写的暴搜,时间复杂度O(2^n)没有超时,如果是要完全的暴力建议去写搜索,不要写for循环枚举,因为for的枚举在判断的时候时间复杂度还要*n,这样估计就超时了,写搜索可以在状态转换的时候把值算出来,这样最后的时候不用可以去算什么,直接if判断,减少了一个n,时间复杂度应该是妥妥的O(2^n),还有就是再说下白书上的方法,用的是中途相遇法,这个方法感觉很好,让我想起了双向广搜,这个题目我想是不是应该叫双向深搜,大体思路就是把要枚举的东西分成两部分分别枚举,然后在结合一起去判断,时间复杂度可以降低不少,白书给的时间复杂度我感觉算的有问题,我算的是O(1.44^n*n/2),两个方法我都试了,中途相遇法的时间复杂度优化了很多,下面是两个方法的AC代码。
直接暴力深搜 时间复杂度
O(2^n) runtime 2.292
#include<map>
#include<string>
#include<stdio.h>
#include<string.h>
using namespace std;
int num[30];
char str[1100];
int Ans ,Anszt;
void DFS(int nowid ,int nows ,int nownum ,int nowzt)
{
if(nowid == 0)
{
if(nownum == 0)
{
if(Ans <= nows)
{
Ans = nows;
Anszt = nowzt;
}
}
return ;
}
DFS(nowid - 1 ,nows ,nownum ,nowzt * 2);
DFS(nowid - 1 ,nows + 1 ,nownum ^ num[nowid] ,nowzt * 2 + 1);
}
int main ()
{
int n ,i ,j ,now;
while(~scanf("%d" ,&n))
{
for(i = 1 ;i <= n ;i ++)
{
scanf("%s" ,str);
now = 0;
int l = strlen(str) - 1;
for(j = 0 ;j <= l ;j ++)
now = now ^ (1 << (str[j] - 'A'));
num[i] = now;
}
Ans = Anszt = 0;
DFS(n ,0 ,0 ,0);
printf("%d\n" ,Ans);
int nowid = 1 ,mk = 0;
while(Anszt)
{
if(Anszt&1)
{
if(mk) printf(" %d" ,nowid);
else printf("%d" ,nowid);
mk = 1;
}
Anszt /= 2;
nowid ++;
}
printf("\n");
}
return 0;
}
中途相遇法 时间复杂度
O(2^(n/2)*n/2) => 1.44^n*n/2 runtime 0.029
#include<map>
#include<string>
#include<stdio.h>
#include<string.h>
using namespace std;
map<int ,int>mark;
int num[30];
char str[1100];
int bitcount(int x)
{
return x == 0 ? 0 : bitcount(x / 2) + (x & 1);
}
int main ()
{
int n ,i ,j ,now ,l;
while(~scanf("%d" ,&n))
{
for(i = 1 ;i <= n ;i ++)
{
scanf("%s" ,str);
now = 0 ,l = strlen(str) - 1;
for(j = 0 ;j <= l ;j ++)
now = now ^ (1 << (str[j] - 'A'));
num[i] = now;
}
mark.clear();
int n1 = n / 2;
int n2 = n - n1;
for(i = 0 ;i < (1 << n1) ;i ++)
{
int x = 0;
for(j = 1 ;j <= n1 ;j ++)
if(i & (1 << (j-1))) x ^= num[j];
if(!mark.count(x) || bitcount(i) > bitcount(mark[x]))
mark[x] = i;
}
int Ans = 0;
for(i = 0 ;i < (1<<n2) ;i ++)
{
int x = 0;
for(j = 1 ;j <= n2 ;j ++)
if(i & (1 << (j-1))) x ^= num[n1 + j];
if(mark.count(x) && bitcount(i) + bitcount(mark[x]) > bitcount(Ans))
Ans = (i << n1) ^ mark[x];
}
printf("%d\n" ,bitcount(Ans));
int mk = 0,Anszt = Ans ,nowid = 1;
while(Anszt)
{
if(Anszt&1)
{
if(mk) printf(" %d" ,nowid);
else printf("%d" ,nowid);
mk = 1;
}
Anszt /= 2;
nowid ++;
}
printf("\n");
}
return 0;
}
LA2965侏罗纪(异或和为0的最大数字个数)的更多相关文章
- LA2965 n个数中选出最多个数异或和为0
intput n 1<=n<=24 n串只有大写字母的字符串 output 选出最多个字符串且每个大写字母出现的次数为偶数 第一行输出个数x 第二行输出x个字符串的下标 做法:将每个字符串 ...
- 0..n去掉一个数,给你剩下的数,找出去掉的那个数
转载请注明转自blog.csdn.net/souldak , 微博@evagle 首先,考虑没有去掉那些数,如果n是奇数,n+1个最低位肯定是0101...01,count(0)=count(1),如 ...
- 在0~N个数字中,取指定个数的不重复数字,要求这些数字的和为指定值,求所有结果
using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace Cons ...
- TabIndex 属性 Tabindex="-1" 与Tabindex="0"、任意数字 (收录)
TabIndex 属性 Tabindex="-1" 与Tabindex="0".任意数字 html中的tabIndex属性可以设置键盘中的TAB键在控件中的移动 ...
- GridView导出成Excel字符"0"丢失/数字丢失的处理方式 收藏
GridView导出成Excel字符"0"丢失/数字丢失的处理方式 收藏 GridView 导出成Excel文件,这个代码在网上比较多.但是发现存在一个问题,导出的数据中如果有&q ...
- 获取数组中多个相加等于0的一组数字 javascript
//获取数组中两个相加等于0的一对数字,比如[ [ -10, 10 ], [ -5, 5 ] ] var arr=[-5,10,1,-10,3,4,5,9] //对数组进行排序 arr.sort(fu ...
- Easyui的numberbox无法输入以0开头的数字编号(转载)
1.问题 项目中碰到这样一个问题,Easyui的numberbox在输入数字编号的时候不能以0开头 在我输入以0开头的数字编号后,离开输入框的时候,那个前缀0就自动去掉了. 接下来,我们查看API说明 ...
- x^a=b(mod c)求解x在[0,c-1]上解的个数模板+原根求法
/************************************* 求解x^a=b(mod c) x在[0,c-1]上解的个数模板 输入:1e9>=a,b>=1,1e9>= ...
- python计算1~2008中0和1的个数
计算1~2008中所有自然数中1和0的个数总数. 通过自然数的大小划分区间,将自然数每位上的数载入列表,循环计数. list = [] onecount = 0 zerocount = 0 for i ...
随机推荐
- #String类简述(小白理解,小白编写,欢迎大神指点,小白跪谢)
@ 目录 一.前言(可忽略) 二.String变量的认知 三.String类的构造方法 四.String类的基本方法 4.1 toString()方法 4.2 equals()方法 4.3 equal ...
- Django之缓存、信号和图片验证码、ORM性能
一. 缓存 1. 介绍 缓存通俗来说:就是把数据先保存在某个地方,下次再读取的时候不用再去原位置读取,让访问速度更快. 缓存机制图解 2.Django中提供了6种缓存方式 1. 开发调试 2. 内存 ...
- Android | 玩转AppBarLayout,设置scrollFlags滑动属性详解
CoordinatorLayout与AppBarLayout的配合使用,在之前的文章中我们也经常使用,主要是专门用来打造各种炫酷的效果. 有童鞋看了之前的文章反馈对AppBarLayout中的scro ...
- [Azure Devops] 使用 Azure Boards 管理工作
1. 什么是 Azure Boards 通过 Azure Boards 网络服务,团队可以管理其软件项目.它提供了丰富的功能,包括 Scrum 和看板的本地支持.可定制的仪表板和集成报告.这些工具可以 ...
- P1164_小A点菜(JAVA语言)
思路 简单动态规划问题 题目背景 uim神犇拿到了uoi的ra(镭牌)后,立刻拉着基友小A到了一家--餐馆,很低端的那种. uim指着墙上的价目表(太低级了没有菜单),说:"随便点" ...
- 《逆向工程核心原理》——IAThook
hook逻辑写入dll中,注入dll. #include "pch.h" #include <tchar.h> #include "windows.h&quo ...
- 攻防世界 reverse SignIn
SignIn 2019_SUCTF __int64 __fastcall main(__int64 a1, char **a2, char **a3) { char mod; // [rsp+0 ...
- django 自带的用户系统
首先,我要说明一下,下面内容不是必须品,如果各位大神喜欢手写也是可以的,你也可以选择自带的功能来缩减你的代码量,提高效率! 第一步 系统配置用户表 首先,在models中创建用户表,导包 from d ...
- 深入理解Java并发框架AQS系列(三):独占锁(Exclusive Lock)
一.前言 优秀的源码就在那里 经过了前面两章的铺垫,终于要切入正题了,本章也是整个AQS的核心之一 从本章开始,我们要精读AQS源码,在欣赏它的同时也要学会质疑它.当然本文不会带着大家逐行过源码(会有 ...
- 互联网开发工具之idea项目打jar包
一.idea打jar包 步骤一:创建一个简单的java项目:如下图所示 `public class Main { public static void main(String[] args) { Sy ...