一个超复杂的间接递归——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/),当时随口指点了几句.难得这位小朋友虚心修正.从善如流,不断地改,又 ...
随机推荐
- 就这样获取文件的MD5和大小
纠结真蛋疼 判断一件事值不值得去做的唯一标准是这件事是不是令我纠结.如果纠结了,就不去做了!但是,人总要活着,又能怎样.谁说男人就没有那么几天...... 从极速妙传说起 在现在各大厂商都推出免费云盘 ...
- 三、Authentication & sessionid
客户在访问Django的某些敏感资料时,被要求需要先登录,客户通过/admin/login进行登录,客户登录成功后,Django给客户分配一个sessionid,后续的访问过程,客户端只需在http头 ...
- hibernate集成
hibernate是一个优秀的持久化框架负责简化将对象保存到数据库中,或从数据库中读取数据并封装到对象的工作.hibernate通过简单配置和编码即可替代jdbc繁琐的程序代码. 下面是集成hiber ...
- [Solution] NPOI操作Excel
NPOI 是 POI 项目的 .NET 版本.POI是一个开源的Java读写Excel.WORD等微软OLE2组件文档的项目.使用 NPOI 你就可以在没有安装 Office 或者相应环境的机器上对 ...
- Java读取Excel文件的几种方法
Java读取 Excel 文件的常用开源免费方法有以下几种: 1. JDBC-ODBC Excel Driver 2. jxl.jar 3. jcom.jar 4. poi.jar 简单介绍: 百度文 ...
- Node.js爬虫抓取数据 -- HTML 实体编码处理办法
cheerio DOM化并解析的时候 1.假如使用了 .text()方法,则一般不会有html实体编码的问题出现 2.如果使用了 .html()方法,则很多情况下(多数是非英文的时候)都会出现,这时, ...
- 使用saripaar对android输入控件进行快速验证
saripaar是个android的第三方快速校验,使用注解快速添加验证规则. public class LoginActivity extends Activity implements Valid ...
- 11条javascript知识
1.局部变量和全局变量 var操作符定义的变量将成为定义该变量作用域中的局部变量.这个局部变量会在函数退出后销毁.不同于其他语言,javaScript不存在块级作用域. 全局变量就是window对象的 ...
- 轻量级的移动 webapp 框架Jingle
一大早爬起来,发现这样的一个东东,国产,感觉实用性很强,试着用用. 1.28补记: 试着用jingle做了一个网站的移动版,感觉如果在布局上要求不高的话 - 目前支持的布局只有list,还是挺不错,做 ...
- java操作小技巧,遇到过的会一直更新,方便查找
1.<c:forEach>可以循环map array List 2.操纵数组,不知道类型的情况下,不需要判断数组类型,直接用反射,arrays.Class.isArrays() 获取数组长 ...