BNUOJ-1065或运算的简单解法
http://www.bnuoj.com/bnuoj/problem_show.php?pid=1065
-----------------------------------------------
#include<stdio.h>
int main()
{
int n,a[10001];
int T;
int i,j,k;
int ans=0;
scanf("%d",&T);
while(T--)
{
scanf("%d",&n);
ans=0;
for(i=0;i<n;++i)
scanf("%d",&a[i]);
for(i=0;i<n;++i)
for(j=0;j<n;++j)
ans+=(a[i]|a[j]);
printf("%d\n",ans);
}
return 0;
}
-----------------------------------------------
上面这个程序的时间复杂度就是O(n^2)的,输入规模增长到原来的n倍,运行时间将会是原来的n^2倍(两重循环内部的操作的次数变为原来的n^2倍)。这样的程序对于n高达10000的数据规模运行时间显然太长了,无法达到我们的要求。所以请你帮忙修改一下这个程序(只是两重循环的部分),降低算法的时间复杂度,但是程序的功能不能改变。
Input
每组数据的第一行有一个正整数 n (1≤n≤10000)。
接下来同一行有n个非负整数,每个数都不超过 2^16范围。两个数之间用空格分开。
Output
Sample Input
1
2 18467 6334
Sample Output
70239
思路:这道题目,必须感谢队友刘庆的想法......
比如现在有n(n==3)个数,分别是:2 5 7
将它们转化为二进制: 0010 0101 0111
那么,现在只看二进制,我们将这三个二进制加起来,但不进位,只统计二进制各个位上面的1的个数会得到:
1 1 1 1
0 2 2 2
这表示,二进制第三位没有1,第二位有2个1,第一位有2个1,第0位有2个1,这些位对应:2^3 2^2 2^1 2^0
那么会发现当2|2+2|5+2|7==2*2^0+n*2^1+2*2^2+0*2^3
5|2+5|5+5|7==n*2^0+2*2^1+n*2^2+0*2^3
7|2+7|5+7|7==n*2^0+n*2^1+n*2^2+0*2^3
发现没有,当一个数,比如7的二进制0111去或其他数的时候,若有一位本身是1,那么这一位或上与之对应的那一位(不管是0还是1)形成的新的数字的这一位都会是1;
就比如000110 与010000 或 变成010110 ,会发现只要有1的位,所形成新的二进制那一位也必然有1,那么当它与n个二进制数字或操作,所形成新的二进制数字,当原数字有1那么它对应的那一位也必然是1的.....这样,我们就只需要统计所以的二进制加起来不产生进位的情况下,每一位有多少个1就好 。
代码:
#include<iostream>
#include<stdio.h>
#include<string.h>
using namespace std;
struct node
{
int a[20];
int cnt;
int num;
}s[10005];
int t[20],f=0;
int p[20]={1,2,4,8,16,32,64,128,256,512,1024,2048,4096,8192,16384,32768,65536,131072};
int main()
{
int text;
scanf("%d",&text);
while(text--)
{
memset(s,0,sizeof(s));
memset(t,0,sizeof(t));
int n;
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%d",&s[i].num);
int tmp=s[i].num;
while(tmp>0)
{
int tmp1=s[i].cnt;
s[i].a[tmp1]=tmp%2;
tmp/=2;
s[i].cnt++;
}
for(int j=0;j<=15;j++)
if(s[i].a[j]==1)
t[j]++;
//for(int i=0;i<=15;i++)
//printf("%d",t[i]);
}
int ans=0;
for(int i=1;i<=n;i++)
{
for(int j=0;j<=15;j++)
{
if(s[i].a[j]==1)
{
ans+=p[j]*n;
}
if(s[i].a[j]==0)
{
ans+=t[j]*p[j];
}
}
}
printf("%d\n",ans);
}
return 0;
}
BNUOJ-1065或运算的简单解法的更多相关文章
- c++复习一:复数运算的简单实现。
复数运算的简单实现. 程序很简单了.基本忘光了复数,重新了解了基本概念.如何在平面表示一个复数,复数的长度|x|=开根 a^2+b^2.和四则运算. 程序基本点: 封装和抽象: 1)封装成员数据,私有 ...
- js中,三元运算的简单应用(?:)
js中,三元运算的简单应用: var sinOrMul = ""; sinOrMul =(subType=="single")?("<span ...
- [PHP][位转换积累]之异或运算的简单加密应用
异或的符号是^.按位异或运算, 对等长二进制模式按位或二进制数的每一位执行逻辑按位异或操作. 操作的结果是如果某位不同则该位为1, 否则该位为0. xor运算的逆运算是它本身,也就是说两次异或同一个数 ...
- 关于PHP位运算的简单权限设计
写在最前面 最近想写一个简单的关于权限处理的东西,之前我也了解过用二进制数的位运算可以出色地完成这个任务.关于二进制数 的位运算,常见的就是“或.与.非”这三种简单运算了,当然,我也查看了下PHP手册 ...
- Java中的位运算及简单的算法应用介绍
众所周知,计算机底层是二进制.而java作为一门计算机编程语言,也对二进制的位运算提供了完整的支持. 在java中,int是32位的,也就是说可以用来实现32位的位运算.方便起见,我们一般用16进制对 ...
- java位运算之简单了解
1.十进制转化为二进制 将正整数转化为二进制的方法“除2取余,逆序排列”. 2.二进制转化为十进制 表示整数的二进制,第一位为标志位,0代表为正整数,位数从右开始,第一个位数为0,各位位数记作n,取各 ...
- c# 如何利用异或运算进行简单加密解密
利用“^”异或运算对字符串进行加密 原理:按位做“异或”运算是->位值相同得1,不同得0,如下计算 1 ^ 1 = 0 1 ^ 0 = 1 0 ^ 1 = 1 0 ^ 0 = 0 例如: < ...
- VS Code mac版全局搜索失效最简单解法
网上百度到的一些说法,说是添加以下命令行 "search.exclude": { "system/": true, "!/system/**/*.ps ...
- 使用redux简单的实现加法运算(简单的状态改变)
描述该做啥?(action)!具体怎么做(reducer)!统一规划(store:包含reducer+所有的state) 上代码:index.ios.js import React, { Compon ...
随机推荐
- import 导包三种方法
# -*- coding: utf-8 -*- #python 27 #xiaodeng #导包三种方法 #(常用)完整的导入,也是最基本的方法 import re #自己定义别名,一般情况下尽量少用 ...
- DevExpress.XtraLayout.LayoutControl 动态添加控件
// Create an item within a specified group,// bound to a specified data field with the specified edi ...
- Centos7安装Openresty
通过yum安装 在 /etc/yum.repos.d/ 下新建 OpenResty.repo 内容 [openresty] name=Official OpenResty Repository bas ...
- linux 文件操作库函数
fopen :打开文件 fread :读文件 fwrite : 写文件 fgetc : 读字符 fputc : 写字符 fscanf : 格式化读 fprintf : 格式化写 fseek : 文件偏 ...
- 【cookie】cookie和session的终极区别
如果浏览器使用的是 cookie,那么所有的数据都保存在浏览器端,比如你登录以后,服务器设置了 cookie用户名(username),那么,当你再次请求服务器的时候,浏览器会将username一块发 ...
- dom 解析xml文件
JAXP技术 JAXP即Java Api for Xml Processing该API主要是SUN提供的用于解析XML数据的一整套解决方案,主要包含了DOM和SAX解析技术.大家可以参见SUN的以下两 ...
- Makefile 中:= ?= += =的区别【转】
转自:http://www.cnblogs.com/wanqieddy/archive/2011/09/21/2184257.html 在Makefile中我们经常看到 = := ?= +=这几个赋值 ...
- INFORMATION_SCHEMA数据库介绍
删除mysql数据库某一张主键表的所有外键关系 SELECT CONCAT('alter table ', TABLE_NAME , ' drop foreign key ', constraint_ ...
- Python学习笔记014——迭代器 Iterator
1 迭代器的定义 凡是能被next()函数调用并不断返回一个值的对象均称之为迭代器(Iterator) 2 迭代器的说明 Python中的Iterator对象表示的是一个数据流,被函数next()函数 ...
- PLSQL_统计信息系列02_统计信息的对象
20150505 Created By BaoXinjian