Home >Backend Development >PHP Tutorial >求教导怎么优化算法
只是第1,2次选择。还要再做2次。后面基本一样重复第二次志愿的。100个学生运行数据大概花了6-7秒。求指导怎么可以优化下~~谢谢
if($_POST['cal'])
{
//--------------------------------------------第一志愿-------------------------------------------
$sql = "select * from hos;";
$res = mysql_query($sql)or die(mysql_error());
$info = mysql_fetch_array($res)or die(mysql_error());
//echo $info['total']; //得到总医院数
do{ //循环1
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++//分界线
if($info['same']==0) //如果不分男女时
{
$hos_id = $info['id']; //医院ID号
//echo $hos_id.",";
$hos_total = $info['total']; //医院总需求人数
//echo $hos_total.".";
$sql_total = "select count(*) as total from stu where first='".$hos_id."' and get=0 ;"; //添该志愿的学生数
$res_total = mysql_query($sql_total);
$info_total = mysql_fetch_array($res_total); //得到填报该志愿的学生数
if($hos_total > $info_total['total'])
{$sql1 = "select id from stu where first='".$hos_id."' and get=0 ORDER BY score DESC ;";}
else {$sql1 = "select id from stu where first='".$hos_id."' and get=0 ORDER BY score DESC limit $hos_total;";}
$res1 = mysql_query($sql1);
$info1 = mysql_fetch_array($res1);
do //循环2
{
$stu_id = $info1['id'];
$sql2 = "UPDATE stu SET result='".$hos_id."',get=1 where id='".$stu_id."' ;";
$res2 = mysql_query($sql2);
}while($info1 = mysql_fetch_array($res1)); //循环2结束
}
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++//分界线
else //如果区分男女-----$info['same']==1;
{
$hos_id = $info['id']; //医院ID号
//男生SQL
$hos_total_men = $info['men']; //医院总需求男生数
$sql_total_men = "select count(*) as total_men from stu where first='".$hos_id."' and sex='f' and get=0 ;"; //添该志愿的男学生数
$res_total_men = mysql_query($sql_total_men);
$info_total_men = mysql_fetch_array($res_total_men); //得到填报该志愿的男学生数
if($hos_total_men > $info_total_men['total_men'])
{$sql1_men = "select id from stu where first='".$hos_id."' and sex='m' and get=0 ORDER BY score DESC ;";}
else {$sql1_men = "select id from stu where first='".$hos_id."'and sex='m' and get=0 ORDER BY score DESC limit $hos_total_men;";}
$res1_men = mysql_query($sql1_men);
$info1_men = mysql_fetch_array($res1_men);
do //循环2
{
//男生更新
$stu_id_men = $info1_men['id'];
$sql2_men = "UPDATE stu SET result='".$hos_id."',get=1 where id='".$stu_id_men."' ;";
$res2_men = mysql_query($sql2_men);
}while($info1_men = mysql_fetch_array($res1_men));
//女生SQL
$hos_total_wom = $info['wom']; //医院总需求女生数
$sql_total_wom = "select count(*) as total_wom from stu where first='".$hos_id."' and sex='f' and get=0 ;"; //添该志愿的男学生数
$res_total_wom = mysql_query($sql_total_wom);
$info_total_wom = mysql_fetch_array($res_total_wom); //得到填报该志愿的女学生数
if($hos_total_wom > $info_total_wom['total_wom'])
{$sql1_wom = "select id from stu where first='".$hos_id."' and sex='f' and get=0 ORDER BY score DESC ;";}
else {$sql1_wom = "select id from stu where first='".$hos_id."' and sex='f' and get=0 ORDER BY score DESC limit $hos_total_wom;";}
$res1_wom = mysql_query($sql1_wom);
$info1_wom = mysql_fetch_array($res1_wom);
do //循环2
{
//女生更新
$stu_id_men = $info1_wom['id'];
$sql2_wom = "UPDATE stu SET result='".$hos_id."',get=1 where id='".$stu_id_wom."';";
$res2_wom = mysql_query($sql2_wom);
}while($info1_wom = mysql_fetch_array($res1_wom));
//循环2结束
}
}while($info = mysql_fetch_array($res)); //循环1结束
//--------------------------------------------第一志愿结束-------------------------------------------
//----------------------------------------------第二志愿---------------------------------------------
$s_sql = "select * from hos;";
$s_res = mysql_query($s_sql)or die(mysql_error());
$s_info = mysql_fetch_array($s_res)or die(mysql_error());
do{ //循环1
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++//分界线
if($s_info['same']==0) //如果不分男女时
{
$s_hos_id = $s_info['id']; //医院ID号
$s_hos_total = $s_info['total']; //医院总需求人数
$s_sql_total = "select count(*) as total from stu where second='".$s_hos_id."' and get=0 ;"; //添该志愿的学生数
$s_res_total = mysql_query($s_sql_total);
$s_info_total = mysql_fetch_array($s_res_total); //得到第二志愿填报该志愿的学生数
$s_get_sql = "select count(*) as total from stu where result='".$s_hos_id."' ;"; //得到第一志愿已为HOS_ID的学生
$s_get_res = mysql_query($s_get_sql);
$s_get_info = mysql_fetch_array($s_get_res);
if($s_get_info['total'] == $s_hos_total){}
else
{
$xuqiu2 = $s_hos_toal-$s_get_info['total']; //还需要多少学生数
if( $xuqiu2 > $s_info_total['total']){$s_sql1 = "select id from stu where second='".$s_hos_id."' and get=0 ORDER BY score DESC ;";}
else {$s_sql1 = "select id from stu where second='".$s_hos_id."' and get=0 ORDER BY score DESC limit $xuqiu2;";}
$s_res1 = mysql_query($s_sql1);
$s_info1 = mysql_fetch_array($s_res1);
do //循环2
{
$s_stu_id = $s_info1['id'];
$s_sql2 = "UPDATE stu SET result='".$s_hos_id."',get=1 where id='".$s_stu_id."' ;";
$s_res2 = mysql_query($s_sql2);
}while($s_info1 = mysql_fetch_array($s_res1));
}
}//循环2结束
else //如果区分男女-----$info['same']==1;
{
$s_hos_id = $s_info['id']; //医院ID号
//男生SQL
$s_hos_total_men = $s_info['men']; //医院总需求男生数
$s_sql_total_men = "select count(*) as total_men from stu where second='".$s_hos_id."' and sex='m' and get=0 ;"; //添该志愿的男学生数
$s_res_total_men = mysql_query($s_sql_total_men);
$s_info_total_men = mysql_fetch_array($s_res_total_men); //得到填报该志愿的男学生数
$s_get_sql_men = "select count(*) as total from stu where result='".$s_hos_id."' and sex='m' ;"; //得到第一志愿已为HOS_ID的男学生
$s_get_res_men = mysql_query($s_get_sql_men);
$s_get_info_men = mysql_fetch_array($s_get_res_men);
if($s_get_info_men['total'] == $s_hos_total_men){}
else
{
$xuqiu2_men = $s_hos_toal_men-$s_get_info_men['total']; //还需要多少男学生数
if($xuqiu2_men > $s_info_total_men['total_men'])
{$s_sql1_men = "select id from stu where second='".$s_hos_id."' and sex='m' and get=0 ORDER BY score DESC ;";}
else {$s_sql1_men = "select id from stu where second='".$s_hos_id."'and sex='m' and get=0 ORDER BY score DESC limit $xuqiu2_men;";}
$s_res1_men = mysql_query($s_sql1_men);
$s_info1_men = mysql_fetch_array($s_res1_men);
do //循环2
{
//男生更新
$s_stu_id_men = $s_info1_men['id'];
$s_sql2_men = "UPDATE stu SET result='".$s_hos_id."',get=1 where id='".$s_stu_id_men."' ;";
$s_res2_men = mysql_query($s_sql2_men);
}while($s_info1_men = mysql_fetch_array($s_res1_men));
}
//女生SQL
$s_hos_total_wom = $s_info['wom']; //医院总需求女生数
$s_sql_total_wom = "select count(*) as total_wom from stu where second='".$s_hos_id."' and sex='f' and get=0 ;"; //添该志愿的女学生数
$s_res_total_wom = mysql_query($s_sql_total_wom);
$s_info_total_wom = mysql_fetch_array($s_res_total_wom); //得到填报该志愿的女学生数
$s_get_sql_wom = "select count(*) as total from stu where result='".$s_hos_id."' and sex='f' ;"; //得到第一志愿已为HOS_ID的女学生
$s_get_res_wom = mysql_query($s_get_sql_wom);
$s_get_info_wom = mysql_fetch_array($s_get_res_wom);
if($s_get_info_wom['total'] == $s_hos_total_wom){}
else
{
$xuqiu2_wom = $s_hos_toal_wom-$s_get_info_wom['total']; //还需要多少女学生数
if($xuqiu2_wom > $s_info_total_wom['total_wom'])
{$s_sql1_wom = "select id from stu where second='".$s_hos_id."' and sex='f' and get=0 ORDER BY score DESC ;";}
else {$s_sql1_wom = "select id from stu where second='".$s_hos_id."'and sex='f' and get=0 ORDER BY score DESC limit $xuqiu2_wom;";}
$s_res1_wom = mysql_query($s_sql1_wom);
$s_info1_wom = mysql_fetch_array($s_res1_wom);
do //循环2
{
//女生更新
$s_stu_id_wom = $s_info1_wom['id'];
$s_sql2_wom = "UPDATE stu SET result='".$s_hos_id."',get=1 where id='".$s_stu_id_wom."' ;";
$s_res2_wom = mysql_query($s_sql2_wom);
}while($s_info1_wom = mysql_fetch_array($s_res1_wom));
}
}
}while($info = mysql_fetch_array($res));
太长 没怎么看。。。但目测 应该得 联表
太长 没怎么看。。。但目测 应该得 联表
链表是双表查询么??速度能快多少呢
噢 我稍微看了下 。。。。你 查询后更新。。。是吧
你应该得 改成 update 下的 子 查询 拿你 第一段的 $sql2 = "UPDATE stu SET result='".$hos_id."',get=1 where id='".$stu_id."' ;";
和之前的 那些 改下 成 "update stu set result='$hos_id',get=1 where id in(select id) "
大致这样吧。。。当然 要按你要求去改。。。
我不知道 我说到你的点子上去么 毕竟不晓得你的意图
噢 我稍微看了下 。。。。你 查询后更新。。。是吧
你应该得 改成 update 下的 子 查询 拿你 第一段的 $sql2 = "UPDATE stu SET result='".$hos_id."',get=1 where id='".$stu_id."' ;";
和之前的 那些 改下 成 "update stu set result='$hos_……
意思是吧select语句放在update里面么?
恩。。。按你的要求的条件去改 不要做那些循环。。操作了那么多次SQL
改了后 绝对是质的飞跃
恩。。。按你的要求的条件去改 不要做那些循环。。操作了那么多次SQL
改了后 绝对是质的飞跃
但我要循环整个表,把每个学生的志愿都有安排出来。如果不循环怎么搞?
update stu set result='$hos_id',get=1 where in (select id) and get=0
你 吧 $hos_id 改为你要那参数,然后去 SQL 直接执行下
update stu set result='$hos_id',get=1 where in (select id) and get=0
你 吧 $hos_id 改为你要那参数,然后去 SQL 直接执行下
不太懂~~可以给些相关的代码参考下么,初学没多久。不太会
1、有比较才能有鉴别。因为你需要的是优化,所以你应该给出表结构和测试数据
2、你的代码带乱,应该格式化一下。两大部分代码应合并在一起,这是代码的优化
3、一般思路是:对每一个学生检查志愿是否可达。而你的是对每一个医院检查是否有意愿学生
嵌套的循环中,循环次数多的应尽可能的放在外层。这是算法的优化
其实代码可以写得好看点,这样有助于帮你解决问题。
代码尝试简化了一下~~继续求高手指导优化。现在运算比原来快了4倍了。但是还是要1-2秒~~继续求优化
for($i=1;$i {
//开始取医院信息
$sql = "select * from hos;";
$res = mysql_query($sql)or die(mysql_error());
$info = mysql_fetch_array($res)or die(mysql_error());
while($info = mysql_fetch_array($res))
{
$id=$info['id']; //得到医院ID
$total=$info['total']; //得到医院需求总人数
if($i==1){$key='first';}
else if($i==2){$key='second';}
else {$key='third';}
if($info['same']==0) //不区分男女时
{
$sql = "update stu set result=$id,get=1 where id in (select id ORDER BY score DESC) and $key=$id and get=0 ";
$num_sql = "select count(id) as total from stu where result=$id ;"; //得到第i志愿已为医院_ID的学生
$num_res = mysql_query($num_sql);
$num_info = mysql_fetch_array($num_res);
if($num_info['total']>$total)
{
$sql.="limit $total ";
mysql_query($sql);
}
else if($num_info['total']==$total){ $sql.="limit 0 "; mysql_query($sql);}
else
{
$need=$total-$num_info['total'];
$sql.="limit $need";
mysql_query($sql);
}
}
else //区分男女
{
$m_total=$info['men']; //得到医院需求总男生数
$f_total=$info['wom']; //得到医院需求总女生数
$sql = "update stu set result=$id,get=1 where id in (select id ORDER BY score DESC) and $key=$id and get=0 ";
$num_msql = "select count(id) as total from stu where result=$id and sex='m'";
$num_mres = mysql_query($num_msql);
$num_minfo = mysql_fetch_array($num_mres);
if($num_minfo['total']>$info['men']){$sql.="limit $m_total "; mysql_query($sql);}
else if($num_minfo['total']==$info['men']){$sql.="limit 0 "; mysql_query($sql);}
else
{
$m_need=$info['men']-$num_minfo['total'];
$sql.="and sex='m' limit $m_need ";
mysql_query($sql);
}
$sql1 = "update stu set result=$id,get=1 where id in (select id ORDER BY score DESC) and $key=$id and get=0 ";
$num_wsql = "select count(id) as total from stu where result=$id and sex='f'";
$num_wres = mysql_query($num_wsql);
$num_winfo = mysql_fetch_array($num_wres);
if($num_winfo['total']>$info['wom']){$sql1.="limit $f_total " ;mysql_query($sql);}
else if($num_winfo['total']==$info['wom']){$sql1.="limit 0 "; mysql_query($sql);}
else
{
$w_need=$info['wom']-$num_minfo['total'];
$sql1.="and sex='f' limit '$w_need' ";
mysql_query($sql1);
}
}
}
}
}
//----------------------------------------------最后随即分配---------------------------------------------
$l_sql = "select * from hos;";
$l_res = mysql_query($l_sql)or die(mysql_error());
$l_info = mysql_fetch_array($l_res)or die(mysql_error());
do{ //循环1
$l_id = $l_info['id']; //医院ID号
$l_total = $l_info['total']; //医院总需求人数
$l_get_sql = "select count(*) as total from stu where result='".$l_id."';";
$l_get_res = mysql_query($l_get_sql);
$l_get_info = mysql_fetch_array($l_get_res); //得到结果已为医院ID的学生数
if($l_get_info['total'] == $l_total){}
else
{
$xuqiu4 = $l_total-$l_get_info['total']; //还需要多少学生数
$sql = "update stu set result='$l_id',get=1 where id in(select id ORDER BY score DESC ) and get=0 limit $xuqiu4 ";
mysql_query($sql);
}
}while($l_info = mysql_fetch_array($l_res));
有点失眠,敲几行代码
楼主的需求应该是:有一批医院让学生报考,学生们有三个志愿,都可以填这些医院,现在要将那些报考了的学生按成绩由高到低进行录取,有的医院有男女数量限制,有的没有。
// 录取所有学生function fetchAllStu(){ // 志愿批次列表 $batch_list = array("first", "second", "third"); // 涉及多个更新操作,需要事务处理 mysql_query("start transaction"); $sql = "select * from hos"; $res = mysql_query($sql) or die(mysql_error()); while($info = mysql_fetch_array($res)) { foreach ($batch_list as $val) { if(empty($info['same'])) { // 不分男女录取 $need_cnt = $info['total'] - getStuCount($info['id']); if(!fetchHosStu($val, $info['id'], $need_cnt)) { mysql_query("rollback"); return false; } } else { // 录取男生 $m_need_cnt = $info['men'] - getStuCount($info['id'], 'm'); if(!fetchHosStu($val, $info['id'], $m_need_cnt, 'm')) { mysql_query("rollback"); return false; } // 录取女生 $f_need_cnt = $info['wom'] - getStuCount($info['id'], 'f'); if(!fetchHosStu($val, $info['id'], $f_need_cnt, 'f')) { mysql_query("rollback"); return false; } } } } mysql_query("commit"); return true;}// 录取指定医院的学生function fetchHosStu($index, $hos_id, $num_limit, $sex=null) { global $field_list; $sub_condition = ''; if (isset($sex)) { if(!isValidSex($sex)) { return false; } $sub_condition = "where sex='{$sex}'"; } // 只有人没录取满才能继续录取 if($num_limit > 0) { $cond = array(); $cond[] = "id in (select id from stu {$sub_condition} order by score desc limit {$num_limit}) and get=0"; $cond[] = "{$index}={$hos_id}"; // 拼接SQL $condition = "where ". implode(' and ', $cond); $sql = "update stu set result={$hos_id},get=1 where {$condition}"; //echo $sql; return mysql_query($sql); } return false;}// 判断是否是合法的性别function isValidSex($sex) { return in_array(strtolower($sex), array('f', 'm'));}// 取得医院已录取学生数function getStuCount($hos_id, $sex=null) { $cond = array(); if (isset($sex)) { if(!isValidSex($sex)) { return false; } $cond[] = "sex='{$sex}'"; } $cond[] = "result={$hos_id}"; // 拼接SQL $condition = "where ". implode(' and ', $cond); $sql = "select count(1) as total from stu {$condition}"; //echo $sql; $res = mysql_query($sql); $info = mysql_fetch_array($res); return $info['total'];}fetchAllStu();
没有编辑权限,只好补充说明一下:
1.录取操作应该是不能中断的,一旦中途失败后果很严重(要是我们高考的录取系统会出这个问题那就好玩了)。所以需要用INNODB的表来存放*,并且要用事务来保证更新操作的完整性。
2.这里假设楼主是需要按批次来进行更新的,如果没有这个限制,还可以进一步优化。
3.53行没用了,请删除之
4.对楼主提点建议。请将代码以代码格式发布,文本格式真的让人很抓狂。雷同的代码尽量封装成函数,咱们是程序员,不是编辑。函数体的代码尽量控制在100行以内,编写和调试都很简单,百万字的长篇小说都会有段落的。
没有编辑权限,只好补充说明一下:
1.录取操作应该是不能中断的,一旦中途失败后果很严重(要是我们高考的录取系统会出这个问题那就好玩了)。所以需要用INNODB的表来存放*,并且要用事务来保证更新操作的完整性。
2.这里假设楼主是需要按批次来进行更新的,如果没有这个限制,还可以进一步优化。
3.53行没用了,请删除之
4.对楼主提点建议。请将代码以代码格式……
我想请问下是在代码里直接调用函数就可以么?我第一次发帖,所以也不是很会,见谅了。
附上数据库表结构~~
调用了函数没有结果。不知道是不是我操作问题~~~
74行有点问题,多了个where。
你可以把里面注掉的sql打开检查一下。
函数调用放在
if($_POST['cal']){ // 调用放这里}
74行有点问题,多了个where。
你可以把里面注掉的sql打开检查一下。
函数调用放在
PHP code?1234if($_POST['cal']){ // 调用放这里}
输出的sql是
select count(1) as total from stu where sex='m' and result=1
update stu set result=1,get=1 where id in (select id from stu where sex='m' order by score desc limit 4) and get=0 and first=1
还是没结果~不进行任何更新呢~
to a403983161
你是楼主吗?还是马甲?
自己有问题就自己发帖,在人家的帖子里捣什么乱
to a403983161
你是楼主吗?还是马甲?
自己有问题就自己发帖,在人家的帖子里捣什么乱
楼主~~我换了个账号。另外那个要用QQ登陆~有点麻烦
那么人家(比如ustb)给了你帮助,你如何给分答谢他?
那么人家(比如ustb)给了你帮助,你如何给分答谢他?
到时候用另外一个号给呢。这个才是我经常用的账号,到时候给分上那个就好了
楼主,如果你执行之后没有效果,有可能是事务回滚了,你可以将事务的部分注释掉,再测试一下。
楼主,如果你执行之后没有效果,有可能是事务回滚了,你可以将事务的部分注释掉,再测试一下。
噢噢~我试下
楼主,如果你执行之后没有效果,有可能是事务回滚了,你可以将事务的部分注释掉,再测试一下。
还是不行,我把每个都分别注释了下都不可以。要不你方便的话留个QQ,我再详细询问请教下你~
楼主,如果你执行之后没有效果,有可能是事务回滚了,你可以将事务的部分注释掉,再测试一下。
大神,问题还没解决,求指教。
非得要把代码一行不漏的贴出来才行么?
$conn = mysql_connect("localhost", "test", "test");mysql_select_db("test", $conn);mysql_set_charset("utf8", $conn);// 录取所有学生function fetchAllStu(){ // 志愿批次列表 $batch_list = array("first", "second", "third"); // 涉及多个更新操作,需要事务处理 mysql_query("start transaction"); $sql = "select * from hos"; $res = mysql_query($sql) or die(mysql_error()); while($info = mysql_fetch_array($res)) { foreach ($batch_list as $val) { if(empty($info['same'])) { // 不分男女录取 $need_cnt = $info['total'] - getStuCount($info['id']); if(!fetchHosStu($val, $info['id'], $need_cnt)) { mysql_query("rollback"); return false; } } else { // 录取男生 $m_need_cnt = $info['men'] - getStuCount($info['id'], 'm'); if(!fetchHosStu($val, $info['id'], $m_need_cnt, 'm')) { mysql_query("rollback"); return false; } // 录取女生 $f_need_cnt = $info['wom'] - getStuCount($info['id'], 'f'); if(!fetchHosStu($val, $info['id'], $f_need_cnt, 'f')) { mysql_query("rollback"); return false; } } } } mysql_query("commit"); return true;}// 录取指定医院的学生function fetchHosStu($index, $hos_id, $num_limit, $sex=null){ $sub_condition = ''; if (isset($sex)) { if(!isValidSex($sex)) { return false; } $sub_condition = "where sex='{$sex}'"; } // 只有人没录取满才能继续录取 if($num_limit > 0) { $cond = array(); $cond[] = "{$index}={$hos_id}"; // 拼接SQL $condition = implode(' and ', $cond); $sql = "update stu set result=1,get=1 where get=0 and {$condition} order by score desc limit {$num_limit}"; //echo $sql; return mysql_query($sql); } return false;}// 判断是否是合法的性别function isValidSex($sex){ return in_array(strtolower($sex), array('f', 'm'));}// 取得医院已录取学生数function getStuCount($hos_id, $sex=null){ $cond = array(); if (isset($sex)) { if(!isValidSex($sex)) { return false; } $cond[] = "sex='{$sex}'"; } $cond[] = "result={$hos_id}"; // 拼接SQL $condition = "where ". implode(' and ', $cond); $sql = "select count(1) as total from stu {$condition} for update"; //echo $sql; $res = mysql_query($sql); $info = mysql_fetch_array($res); return $info['total'];}fetchAllStu();
说明一下:
1.有一个bug修复了,在getStuCount函数中,select语句需要锁记录,以免发生数量变更。
2.更新语句去掉子查询,直接用score排序即可。
以上程序在我本地调试通过,10条记录耗时约0.02秒,应该比楼主之前的效率要高很多了。
说明一下:
1.有一个bug修复了,在getStuCount函数中,select语句需要锁记录,以免发生数量变更。
2.更新语句去掉子查询,直接用score排序即可。
以上程序在我本地调试通过,10条记录耗时约0.02秒,应该比楼主之前的效率要高很多了。
大神,麻烦到你真不好意思的。毕竟水平比较低,刚初学没多久。代码我在本地测试了一下,我大概是100条记录。时间是0.2秒左右,但是结果还是没出来,不知道是什么原因呢?
额滴神呀,上面的代码只是用于更新,取数据并显示出来你自己就能写,你好歹写两行吧,不然我就全写完了。。。
额滴神呀,上面的代码只是用于更新,取数据并显示出来你自己就能写,你好歹写两行吧,不然我就全写完了。。。
更新不了呢~~数据库里的表信息还是原来的。取数据显示出来我原来就写了的。