一个超复杂的间接递归——C语言初学者代码中的常见错误与瑕疵(6)
问题:
问题出处见 C语言初学者代码中的常见错误与瑕疵(5) 。
在该文的最后,曾提到完成的代码还有进一步改进的余地。本文完成了这个改进。所以本文讨论的并不是初学者代码中的常见错误与瑕疵,而是对我自己代码的改进和优化。标题只是为了保持系列的连续性。
改进
程序的总体思想没有改变,所以main()函数不需要任何改动。
int main( void )
{
unsigned n ; puts( "数据组数=?" );
scanf( "%u" , &n ); while ( n -- > )
{
int x ; puts( "整数X=?" );
scanf( "%d", & x ); printf("%d\n" , get_nearest( x ) ); //求最接近x的素数
} return ;
}
进一步的改进体现在
typedef
struct prime_list
{
unsigned prime;
struct prime_list * next;
}
Node; int get_nearest( int x )
{
int step = ; //步长增量
int sign = -; //符号
Node * head = NULL ; //素数链表 while ( ! be_prime( x , & head ) )
x += ( sign = - sign ) * ++ step ; my_free(head) ;
return x ;
}
这里增加了一个链表head,用于存储素数表。这样,在判断素数时只用较小的素数试除就可以了,这可以使计算数量大为减少。因为与自然数相比,素数的数量很少(≈ln n / n 个)。此外,在判断完x是否为素数之后,如果需要判断下一个数(x += ( sign = - sign ) * ++ step ;)是否为素数,这个素数表还可以重复使用,最多再向其中添加一个素数就可以了。(注意最初素数表是空的)
判断素数的方法很简单,小学生都懂。
bool be_prime( int x , Node * * pp ) //根据素数表pp判断x是否为素数
{
if ( x <= )
return false ; if ( x == )
return true ; if ( get_remainder( x , pp ) == ) // x对素数表pp中素数逐个求余有0值
return false ; return true ;
}
但是由于素数表(*pp==NULL)可能是空的,因此
int get_remainder( int x , Node * * pp )//x对素数表pp中素数逐个求余
{
while ( * pp == NULL || sqr_less( (*pp) -> prime , x ) )//表中素数个数不足
add_1_prime ( pp ) ; //表中增加一个素数 Node * p = * pp ; while ( p != NULL )
{
if ( x % p -> prime == )
return ;
p = p -> next ;
} return ! ;
} bool sqr_less ( int n , int x )
{
return n * n < x ;
}
需要先向其中添加素数
add_1_prime ( pp ) ;
直到
sqr_less( (*pp) -> prime , x )
依从小到大次序最后加入的那个素数的平方不小于x为止。
对于
void add_1_prime( Node * * pp )
{
if ( * pp == NULL )
{
add ( , pp ); //第一个素数
return ;
} int next_p = ( * pp )->prime + ; //从最后一个素数之后开始找下一个素数 while ( !be_prime( next_p , pp ) )
next_p ++ ; add( next_p , pp ); //将下一个素数加入素数表
}
来说,加入第一个素数——2很容易,但是寻找素数表中最大素数后的下一个素数时,却需要判断一个整数是否是素数
be_prime( next_p , pp )
这样,就会发现,这个过程最初是由判断某个数x是否是素数开始,
be_prime( x , & head )
在判断过程中需要建立素数表,
add_1_prime ( pp ) ;
而建立素数表,又需要判断某个数是否是素数
be_prime( next_p , pp )
这样就形成了一个极其复杂的间接递归调用。更为复杂的是,在调用的过程中,素数表本身即不断地被使用,而自身也处于不断的变化状态之中,即不断地被添加进新的素数,与复杂的间接递归一道,构成了比复杂更复杂的复杂的代码结构与复杂的数据结构的复杂的结合体。有兴趣的话可以自己算一下圈复杂度,如此复杂的情况通常并不容易遇到。
这种局面完全是由于精打细算造成的,由于对速度的斤斤计较,从而形成了一幅小猫在拼命咬自己尾巴同时小猫自己又在不断变化的复杂无比的动态画面。由此我们不难理解,为什么有人说,“不成熟的优化是万恶之源”(Premature optimization is the root of all evil!- Donald Knuth)。因为优化往往意味着引人复杂。复杂也是一种成本,而且是一种很昂贵的成本。
就这个题目而言这种成本应该算是值得,因为对于求一个较大的最接近的素数问题而言(例如对于109这个量级),两套代码的速度有天壤之别。
增强可读性?
如果把建立素数表的要求写在get_nearest()函数中,可能会使代码可读性变得更好些。
int get_nearest( int x )
{
int step = ; //步长增量
int sign = -; //符号
Node * head = NULL ; //素数链表 while ( 建立最大素数平方不小于x的素数表() , ! be_prime( x , & head ) )
x += ( sign = - sign ) * ++ step ; my_free(head) ;
return x ;
}
但这里的这个这个“,”是免不掉的,且圈复杂度不变。
至于这种写法是否真的改善了可读性,恐怕是见仁见智。
进一步提高效率
没什么更好的办法,只能用点“赖皮”手段,即充分运用已有的素数知识,帮计算机算出一部分素数。
void add_1_prime( Node * * pp )
{
if ( * pp == NULL )
{
add ( , pp ); //第一个素数
return ;
} switch ( ( * pp ) -> prime )
{
case : add ( , pp );
return ;
case : add ( , pp );
return ;
/* 这里可以依样写多个case,只要是按照素数从小到大的次序*/
default:
{
int next_p = ( * pp )->prime + ; //从最后一个素数之后开始找下一个素数 while ( !be_prime( next_p , pp ) )
next_p ++ ; add( next_p , pp ); //将下一个素数加入素数表
return ;
}
}
}
这里switch语句的结构非常有趣。
重构
/*
问题:
素数
在世博园某信息通信馆中,游客可利用手机等终端参与互动小游戏,与虚拟人物Kr. Kong 进行猜数比赛。
当屏幕出现一个整数X时,若你能比Kr. Kong更快的发出最接近它的素数答案,你将会获得一个意想不到的礼物。
例如:当屏幕出现22时,你的回答应是23;当屏幕出现8时,你的回答应是7;
若X本身是素数,则回答X;若最接近X的素数有两个时,则回答大于它的素数。
输入:第一行:N 要竞猜的整数个数
接下来有N行,每行有一个正整数X
输出:输出有N行,每行是对应X的最接近它的素数
样例:输入
4
22
5
18
8
输出
23
5
19
7
作者:薛非
出处:http://www.cnblogs.com/pmer/ “C语言初学者代码中的常见错误与瑕疵”系列博文
版本:V 2.1
*/
#include <stdio.h>
#include <stdbool.h> typedef
struct prime_list
{
unsigned prime;
struct prime_list * next;
}
Node; int get_nearest( int );
bool be_prime( int , Node * * );
int get_remainder( int , Node * * ) ;
void add_1_prime( Node * * );
bool sqr_less ( int , int );
void add ( int , Node * * );
void my_malloc( Node * * );
void my_free( Node * ); int main( void )
{
unsigned n ; puts( "数据组数=?" );
scanf( "%u" , &n ); while ( n -- > )
{
int x ; puts( "整数X=?" );
scanf( "%d", & x ); printf("%d\n" , get_nearest( x ) ); //求最接近x的素数
} return ;
} int get_nearest( int x )
{
int step = ; //步长增量
int sign = -; //符号
Node * head = NULL ; //素数链表 while ( ! be_prime( x , & head ) )
x += ( sign = - sign ) * ++ step ; my_free(head) ;
return x ;
} bool be_prime( int x , Node * * pp ) //根据素数表pp判断x是否为素数
{
if ( x <= )
return false ; if ( x == )
return true ; if ( get_remainder( x , pp ) == ) // x对素数表pp中素数逐个求余有0值
return false ; return true ;
} int get_remainder( int x , Node * * pp )//x对素数表pp中素数逐个求余
{
while ( * pp == NULL || sqr_less( (*pp) -> prime , x ) )//表中素数个数不足
add_1_prime ( pp ) ; //表中增加一个素数 Node * p = * pp ; while ( p != NULL )
{
if ( x % p -> prime == )
return ;
p = p -> next ;
} return ! ;
} bool sqr_less ( int n , int x )
{
return n * n < x ;
} //“偷奸耍滑”的add_1_prime()
void add_1_prime( Node * * pp )
{
if ( * pp == NULL )
{
add ( , pp ); //第一个素数
return ;
} switch ( ( * pp ) -> prime )
{
case : add ( , pp );
return ;
case : add ( , pp );
return ;
/* 这里可以依样写多个case,只要是按照素数从小到大的次序*/
default:
{
int next_p = ( * pp )->prime + ; //从最后一个素数之后开始找下一个素数 while ( !be_prime( next_p , pp ) )
next_p ++ ; add( next_p , pp ); //将下一个素数加入素数表
return ;
}
}
} //老老实实的add_1_prime()
//void add_1_prime( Node * * pp )
//{
// if ( * pp == NULL )
// {
// add ( 2 , pp ); //第一个素数
// return ;
// }
//
// int next_p = ( * pp )->prime + 1 ; //从最后一个素数之后开始找下一个素数
//
// while ( !be_prime( next_p , pp ) )
// next_p ++ ;
//
// add( next_p , pp ); //将下一个素数加入素数表
//} void add ( int prime , Node * * pp )
{
Node * temp ; my_malloc( & temp );
temp -> prime = prime ;
temp -> next = * pp ;
* pp = temp ;
} void my_malloc( Node * * p_p )
{
if ( ( * p_p = malloc( sizeof (* * p_p) ) ) == NULL )
exit();
} void my_free( Node * p )
{
Node * temp ;
while ( ( temp = p ) != NULL )
{
p = p->next;
free( temp );
}
}
相关博客:
偶然发现Jingle Guo网友后来研究同一问题的一篇博文,我感觉对阅读此文的网友可能有一定的参考价值,故在此给出相关链接:从关于素数的算法题来学习如何提高代码效率。
一个超复杂的间接递归——C语言初学者代码中的常见错误与瑕疵(6)的更多相关文章
- C语言初学者代码中的常见错误与瑕疵(5)
问题: 素数 在世博园某信息通信馆中,游客可利用手机等终端参与互动小游戏,与虚拟人物Kr. Kong 进行猜数比赛. 当屏幕出现一个整数X时,若你能比Kr. Kong更快的发出最接近它的素数答案,你将 ...
- 分数的加减法——C语言初学者代码中的常见错误与瑕疵(12)
前文链接:分数的加减法——C语言初学者代码中的常见错误与瑕疵(11) 重构 题目的修正 我抛弃了原题中“其中a, b, c, d是一个0-9的整数”这样的前提条件,因为这种限制毫无必要.只假设a, b ...
- C语言初学者代码中的常见错误与瑕疵(9)
题目 字母的个数 现在给你一个由小写字母组成字符串,要你找出字符串中出现次数最多的字母,如果出现次数最多字母有多个那么输出最小的那个. 输入:第一行输入一个正整数T(0<T<25) 随后T ...
- 要心中有“数”——C语言初学者代码中的常见错误与瑕疵(8)
在 C语言初学者代码中的常见错误与瑕疵(7) 中,我给出的重构代码中存在BUG.这个BUG是在飞鸟_Asuka网友指出“是不是时间复杂度比较大”,并说他“第一眼看到我就想把它当成一个数学问题来做”之后 ...
- C语言初学者代码中的常见错误与瑕疵(7)
问题: 矩形的个数 在一个3*2的矩形中,可以找到6个1*1的矩形,4个2*1的矩形3个1*2的矩形,2个2*2的矩形,2个3*1的矩形和1个3*2的矩形,总共18个矩形.给出A,B,计算可以从中找到 ...
- C语言初学者代码中的常见错误与瑕疵(23)
见:C语言初学者代码中的常见错误与瑕疵(23)
- C语言初学者代码中的常见错误与瑕疵(19)
见:C语言初学者代码中的常见错误与瑕疵(19)
- C语言初学者代码中的常见错误与瑕疵(14)
见:C语言初学者代码中的常见错误与瑕疵(14) 相关链接:http://www.anycodex.com/blog/?p=87
- C语言初学者代码中的常见错误与瑕疵(1)
曾在豆瓣上看到过一个小朋友贴出他自己的代码(http://www.douban.com/group/topic/40293109/),当时随口指点了几句.难得这位小朋友虚心修正.从善如流,不断地改,又 ...
随机推荐
- asp.net url重写相关技术问题整理
1.IIS7配置URL重写需要注意系统是32位还是64位的 在IIS7配置URL重写的时候,需要添加“脚本映射”,如果是64位系统,会有两个地方存放.net framework分别是32位系统和64位 ...
- 0422 发现( 数学口袋精灵)bug
团队博客地址: 甘佳萍:http://www.cnblogs.com/gjpg/ 李鹏飞:http://www.cnblogs.com/l549023320/ 赵创佳:http://www.cnblo ...
- R语言简单实现聚类分析计算与分析(基于系统聚类法)
聚类分析计算与分析(基于系统聚类法) 下面以一个具体的例子来实现实证分析.2008年我国其中31个省.市和自治区的农村居民家庭平均每人全年消费性支出. 根据原始数据对我国省份进行归类统计. 原始数据如 ...
- Build 2016概览
很快Microsoft Build 2016马上就要开始,在直播放出来之前,微软已经提前把本次大会期间的所有课程列表放了出来,你可以在这里看到: https://channel9.msdn.com/E ...
- thread_Semaphore信号量
Semaphore也是一个线程同步的辅助类,可以维护当前访问自身的线程个数,并提供了同步机制. 使用Semaphore可以控制同时访问资源的线程个数,例如,实现一个文件允许的并发访问数. 一个计数信 ...
- Requested registry access is not allowed(不允许所请求的注册表访问权)
尝试创建自定义事件日志时,将会收到“Requested registry access is not allowed(不允许所请求的注册表访问权)”错误消息 EventLog.CreateEventS ...
- 与众不同 windows phone (34) - 8.0 新的控件: LongListSelector
[源码下载] 与众不同 windows phone (34) - 8.0 新的控件: LongListSelector 作者:webabcd 介绍与众不同 windows phone 8.0 之 新的 ...
- PHPWind 8.7中代码结构与程序执行顺序
pw9在此不谈,他是完全重构的作品,是完全MVC下的体系.当然,其中很多东西在PW8.7下已经可见端倪. 主要代码结构 1. 以现代的观点,PW是多入口应用模式,程序根目录下的文件几乎都是入口: 2. ...
- vim编辑器,管道,输入输出重定向
1.vim的认识及其一些常用指令 a, 认识vim的命令行模式和插入模式: 当vim运行后默认进入该模式,他可以控制屏幕光标的移动,字符.字或行的删除,移动复制某区段及进入Insert mode下,或 ...
- http get post
使用java代码模拟http请求 package ftp; import java.io.BufferedReader; import java.io.IOException; import java ...