描述

一个房间里有n盏灯泡,一开始都是熄着的,有1到n个时刻,每个时刻i,我们会将i的倍数的灯泡改变状态(即原本开着的现将它熄灭,原本熄灭的现将它点亮),问最后有多少盏灯泡是亮着的。

提示

范围:40%的数据保证,n<=maxlongint

100%的数据保证,n<=10^200

**********************************************************************

1.编个小程序,打表(1-30),可以看出规律         f(n)=sqrt(n)的下界。

2.思考如何求大整数的根号:两种思路:迭代法,例如二分法、牛顿迭代等;

打点法:精确计算,一步到位。笔算开平方法,我二姐教过我。真是太好了。

我只想简单说个大概:

i。从后往前,隔两位点一个小数点

ii。从前往后,试商,做差,落下来

3.打点法肯定速度非常快O(200*100)的复杂度:根最大是100位,每求一位最多200位的操作。这个过程需要用到大数乘法、减法、移位。

4.最后一点,在编大数运算时最好按照位数固定进行运算,这样虽然牺牲了一点效率,但可读性好、易于实现。

5.要背下来大数运算,不要只会抄scl,唯有如此,方能随用随写,剑心合一。

#include<iostream>
#include<string.h>
#include<stdio.h>
using namespace std;
#define size 205
int n[205], ni;
int ans[205];
int now[205];
int two[205];
int mu[205];
void getTwo(){//two=(ans*2)<<1;
	memset(two, 0, sizeof(two));
	int i;
	for (i = 0; i < size; i++){
		two[i] += (ans[i] << 1);
		two[i + 1] = two[i] / 10;
		two[i] %= 10;
	}
	for (i = size - 2; i >= 0; i--)
		two[i + 1] = two[i];
}
void getNow(){
	int i;
	for (i = size - 3; i >= 0; i--){
		now[i + 2] = now[i];
	}
	now[1] = n[ni--];
	now[0] = n[ni--];
}
void mul(int k){//mu=two_k*k
	memset(mu, 0, sizeof(mu));
	int i;
	for (i = 0; i < size; i++){
		mu[i] += two[i] * k;
		mu[i + 1] = mu[i] / 10;
		mu[i] %= 10;
	}
}
int cmp(){//now-mu
	int i;
	for (i = size - 1; i >= 0; i--)
	if (now[i] != mu[i])
		return now[i] - mu[i];
	return 0;
}
void sub(){//now-mu
	int i;
	for (i = 0; i < size - 1; i++){
		now[i] -= mu[i];
		if (now[i] < 0){
			now[i + 1]--;
			now[i] += 10;
		}
	}
}
void getAns(){
	int i;
	for (i = size - 2; i >= 0; i--){
		ans[i + 1] = ans[i];
	}
	for (i = 9; i >= 0; i--){
		two[0] = i;
		mul(i);
		if (cmp() >= 0){
			ans[0] = i;
			sub();
			return;
		}
	}
}
void go(){
	while (ni >= 0){
		getTwo();
		getNow();
		getAns();
	}
}
int main(){
	freopen("in.txt", "r", stdin);
	char a[205];
	int i;
	for (i = 0; a[i]; i++);
	ni = i;
	memset(n, 0, sizeof(n));
	for (i = 0; i < ni; i++)
		n[i] = a[ni - 1 - i] - '0';
	if ((ni & 1) == 0)ni--;
	memset(ans, 0, sizeof(ans));
	memset(now, 0, sizeof(now));
	go();
	for (i = size - 1; ans[i] == 0; i--);
	for (; i >= 0; i--)cout << ans[i];
	return 0;
}

vijos-1447 开关灯泡-大整数开方算法的更多相关文章

  1. 【老鸟学算法】大整数乘法——算法思想及java实现

    算法课有这么一节,专门介绍分治法的,上机实验课就是要代码实现大整数乘法.想当年比较混,没做出来,颇感遗憾,今天就把这债还了吧! 大整数乘法,就是乘法的两个乘数比较大,最后结果超过了整型甚至长整型的最大 ...

  2. vijos - P1447开关灯泡 (大数模板 + 找规律 + 全然数 + python)

    P1447开关灯泡 Accepted 标签:CSC WorkGroup III[显示标签] 描写叙述 一个房间里有n盏灯泡.一開始都是熄着的,有1到n个时刻.每一个时刻i,我们会将i的倍数的灯泡改变状 ...

  3. 【vijos】1447 开关灯泡(高精度+特殊的技巧)

    https://vijos.org/p/1447 一开始想了想似乎只想到了与约数个数有关,即约数个数为奇数那么显然是亮的. 竟然没想到完全平方数..sad.. 在正因子中,只有完全平方数的正因子才是奇 ...

  4. 大整数算法[11] Karatsuba乘法

    ★ 引子         前面两篇介绍了 Comba 乘法,最后提到当输入的规模很大时,所需的计算时间会急剧增长,因为 Comba 乘法的时间复杂度仍然是 O(n^2).想要打破乘法中 O(n^2) ...

  5. 大整数算法[09] Comba乘法(原理)

    ★ 引子          原本打算一篇文章讲完,后来发现篇幅会很大,所以拆成两部分,先讲原理,再讲实现.实现的话相对复杂,要用到内联汇编,要考虑不同平台等等. 在大整数计算中,乘法是非常重要的,因为 ...

  6. C# 基于大整数类的RSA算法实现(公钥加密私钥解密,私钥加密公钥解密)

    但是C#自带的RSA算法类RSACryptoServiceProvider只支持公钥加密私钥解密,即数字证书的使用. 所以参考了一些网上的资料写了一个RSA的算法实现.算法实现是基于网上提供的一个大整 ...

  7. [转]大整数算法[11] Karatsuba乘法

    ★ 引子         前面两篇介绍了 Comba 乘法,最后提到当输入的规模很大时,所需的计算时间会急剧增长,因为 Comba 乘法的时间复杂度仍然是 O(n^2).想要打破乘法中 O(n^2) ...

  8. 从大整数乘法的实现到 Karatsuba 快速算法

    Karatsuba 快速乘积算法是具有独特合并过程(combine/merge)的分治算法(Karatsuba 是俄罗斯人).此算法主要是对两个整数进行相乘,并不适用于低位数(如 int 的 32 位 ...

  9. 算法笔记_034:大整数乘法(Java)

    目录 1 问题描述 2 解决方案 2.1 蛮力法   1 问题描述 计算两个大整数相乘的结果. 2 解决方案 2.1 蛮力法 package com.liuzhen.chapter5; import ...

随机推荐

  1. Navicat for MySQL 工具注册码

    在使用 MySQL 客户端中,感觉 Navicat for MySQL 做的还是很不错的.鄙人比较懒,喜欢采用绿色安装的方式,百度网盘上备份的有个绿色的压缩包,解压打开需要注册,这里记录一下注册码,版 ...

  2. C++之STL

    5.子类模板访问基类模板在子类模板中访问那些在基类模板中声明且依赖于模板参数的符号,应该在它前面加上作用域限定符"::" 或者显示使用this指针否则,编译器将试图在全局域中寻找该 ...

  3. 单片机温度控制系统DS18B20

    单片机温度控制系统核心 由895X系列单片机来控制来驱动18b20温度传感器模块,通过编写C语言代码,来实现对模块的控制驱动,不断的接收读取18b20传过来的温度信号.将传过来的高低位字节经过个人代码 ...

  4. IO的多路复用和信号驱动

    Linux为多路复用IO提供了较多的接口,有select(),pselect(),poll()的方式,继承自BSD和System V 两大派系. select模型比较简单,“轮询”检测fd_set的状 ...

  5. SpringMVC从入门到精通之第二章

    这一章原本我是想写一个入门程序的,但是后来仔细想了一下,先从下面的图中的组件用代码来介绍,可能更效果会更加好一点.第一节:开发准备介绍之前先说下我的开发调试环境:JDK 1.7的64位 .Eclips ...

  6. Spring之IoC总结帖

    Spring是一个开源框架,Spring是于2003 年兴起的一个轻量级的Java 开发框架,由Rod Johnson 在其著作Expert One-On-One J2EE Development a ...

  7. Machine Learning Algorithms Study Notes(4)—无监督学习(unsupervised learning)

    1    Unsupervised Learning 1.1    k-means clustering algorithm 1.1.1    算法思想 1.1.2    k-means的不足之处 1 ...

  8. AC日记——数字统计 openjudge 1.5 41

    41:数字统计 总时间限制:  1000ms 内存限制:  65536kB 描述 请统计某个给定范围[L, R]的所有整数中,数字2出现的次数. 比如给定范围[2, 22],数字2在数2中出现了1次, ...

  9. Java中正则Matcher类的matches()、lookAt()和find()的区别

    在Matcher类中有matches.lookingAt和find都是匹配目标的方法,但容易混淆,整理它们的区别如下: matches:整个匹配,只有整个字符序列完全匹配成功,才返回True,否则返回 ...

  10. 使用gulp将移动端px转为rem

    使用gulp的插件可以很方便的将xp转为rem,在布局的时候使用@1x .@2x布局,即10rem=device-width:@1x即设计图为320px,1rem对应的10px像素,相对的@2x即为布 ...