0

0

php解决装箱问题

php中文网

php中文网

发布时间:2016-07-25 08:51:00

|

1470人浏览过

|

来源于php中文网

原创

我对装箱问题,石头过河问题的解法
见http://www.oschina.net/question/117304_112681

实现思路主要为:
1. 大块石头必须优先装箱(早装和留到后面装都要装,先解决之)
2. 优先装重量接近w的
3. 同样重量优先装多块,如装9,6和装9,5,1比,则优先951装箱
4. 使用php的函数以简化代码,并使用根据k值生成函数的技巧
5. 此类问题由于本身性质,计算量较大,请酌情设置参数测试。

示例输出:(当rocks为1~9,w为15,k为3)
寻找由 3 个元素组成的最大解:
Array
(
[0] => 9
[1] => 5
[2] => 1
)

寻找由 2 个元素组成的最大解:
Array
(
[0] => 9
[1] => 6
)

寻找由 1 个元素组成的最大解:
Array
(
[0] => 9
)

寻找由 3 个元素组成的最大解:
Array
(
[0] => 8
[1] => 4
[2] => 3
)

寻找由 2 个元素组成的最大解:
Array
(
[0] => 8
[1] => 7
)

寻找由 1 个元素组成的最大解:
Array
(
[0] => 8
)

寻找由 3 个元素组成的最大解:
Array
(
[0] => 7
[1] => 6
[2] => 2
)

寻找由 2 个元素组成的最大解:
Array
(
[0] => 7
[1] => 6
)

寻找由 1 个元素组成的最大解:
Array
(
[0] => 7
)

最小次数:3
装船过程:Array
(
[0] => Array
(
[0] => 9
[1] => 5
[2] => 1
)

[1] => Array
(
[0] => 8
[1] => 4
[2] => 3
)

[2] => Array
(
[0] => 7
[1] => 6
[2] => 2
)

)
  1. // php 练习 之装箱问题
  2. // author: mx (http://my.oschina.net/meikaiyuan)
  3. // 2013/5/30
  4. // 问题:
  5. // http://www.oschina.net/question/117304_112681
  6. /*
  7. 题目:
  8. 以前问过类似问题,没有很好解答。所以想再问一次。
  9. 有大大小小的一堆石头要用船拉到河对岸
  10. --石头有m块,每块的重量已知
  11. --船每次只能装k块石头,并且装载重量不可以超过w
  12. --想求出最少几趟能把全部石头运过河。
  13. ------------------------------------
  14. 例1
  15. 石头有9块,重量分别是1,2,3,4,5,6,7,8,9
  16. k=3
  17. w=15
  18. 那么结果是,最少3次就可以运完。
  19. ------------------------------------
  20. 例2
  21. 石头有9块,重量分别是1,1,1,5,6,6,7,9,9
  22. k=3
  23. w=15
  24. 那么结果是,最少 4 次才可以运完。
  25. */
  26. //代码开始
  27. //石头
  28. global $rocks;
  29. // 船每次最多装几块
  30. global $k;
  31. // 船最大载重量
  32. global $w;
  33. $k=3;
  34. $rocks=array(1,2,3,4,5,6,7,8,9);
  35. // $rocks=array(1,1,1,5,6,6,7,9,9); //换成这组数据试试结果?
  36. $w=15;
  37. // 当前运了几次
  38. $count=0;
  39. // 运输过程,二维数组,形如 1=>array(9,5,1),表示第几次运了哪一些
  40. $process=array();
  41. // 求数组$rocks中一组合,使得最多$k个元素且这些元素的和尽可能大但小于等于指定值$w, 数组已经按从大到小排序过
  42. function getMaxCombination( ) {
  43. //石头
  44. global $rocks;
  45. // 船每次最多装几块
  46. global $k;
  47. // 船最大载重量
  48. global $w;
  49. // 保存各种$k下满足所有元素之和小于等于w且最大的集合
  50. $k_w_result=array();
  51. // 最大组合值
  52. $max_sum=0;
  53. // 哪项最大
  54. $max_one=0;
  55. for ($start=$k;$start>0;$start--){
  56. // 找到由固定$start个元素组成的最大解
  57. $start_w_arr = getMaxCombination2($start);
  58. echo "寻找由 $start 个元素组成的最大解: \n";
  59. print_r($start_w_arr);
  60. echo "\n";
  61. $sum=array_sum( $start_w_arr );
  62. // 注意:因为是降序排列的,$k--, 越早找到的同sum的组合$k越大,也就是解越好,所以是小于不是小于等于
  63. if($sum>$max_sum){
  64. $max_sum=$sum;
  65. $max_one=$k-$start;
  66. }
  67. $k_w_result[]= $start_w_arr ;
  68. }
  69. return $k_w_result[$max_one];
  70. }
  71. // 求数组$rocks中一由给定$start个元素构成的组合,这些元素的和尽可能大但小于等于指定值$w, 数组已经按从大到小排序过
  72. function getMaxCombination2($start ) {
  73. //石头们
  74. global $rocks;
  75. // 船每次最多装几块
  76. global $k;
  77. // 船最大载重量
  78. global $w;
  79. if(count($rocks) return array(0);
  80. }
  81. $c=count($rocks);
  82. // 根据$start生成一函数,内含$start层for循环代码, 然后包含进来再调用此函数
  83. if(!file_exists( "$start.php")){
  84. $output_1="";
  85. $output_2='$sum=';
  86. $output_3='if($sum $output_4='';
  87. for($i=0;$i $output_1.='for($p'.$i.'='.$i.';$p'.$i.' if($i>0){
  88. $output_2.='+';
  89. }
  90. $output_2.='$rocks[$p'.$i.']';
  91. $output_3.='$arr[]=$rocks[$p'.$i.'];';
  92. $output_4.='}';
  93. }
  94. $output_2.=';';
  95. $output_3.=' return $arr; }' ;
  96. $output='';
  97. file_put_contents("$start.php",$output);
  98. include_once "$start.php";
  99. }
  100. else{
  101. include_once "$start.php";
  102. }
  103. return call_user_func('myfor'.$start ,$rocks,$c,$w);
  104. }
  105. //开始计算
  106. // 数组先从大到小排序, 此操作是后续算法省时省力的关键
  107. rsort($rocks);
  108. // 为了防止石头过大船过小造成下面算法死循环
  109. foreach ($rocks as $v){
  110. if($v>$w){
  111. die("有石头不可能装船,换大船来再战!");
  112. }
  113. }
  114. // 算法开始
  115. while(!empty($rocks)){
  116. // 开始装一船
  117. $process[$count]=array();
  118. // 装船
  119. $process[$count]= getMaxCombination( ) ;
  120. // 从石头中移除已经装船的
  121. foreach($process[$count] as $v){
  122. $key=array_search($v, $rocks);
  123. unset( $rocks[$key]);
  124. }
  125. $rocks=array_values($rocks);
  126. // 装船数+1
  127. $count++;
  128. }
  129. // 输出结果
  130. echo '最小次数:'.$count."\n";
  131. echo '装船过程:';
  132. print_r($process);
  133. ?>
复制代码


相关文章

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

14

2026.01.30

c++ 字符串格式化
c++ 字符串格式化

本专题整合了c++字符串格式化用法、输出技巧、实践等等内容,阅读专题下面的文章了解更多详细内容。

9

2026.01.30

java 字符串格式化
java 字符串格式化

本专题整合了java如何进行字符串格式化相关教程、使用解析、方法详解等等内容。阅读专题下面的文章了解更多详细教程。

12

2026.01.30

python 字符串格式化
python 字符串格式化

本专题整合了python字符串格式化教程、实践、方法、进阶等等相关内容,阅读专题下面的文章了解更多详细操作。

4

2026.01.30

java入门学习合集
java入门学习合集

本专题整合了java入门学习指南、初学者项目实战、入门到精通等等内容,阅读专题下面的文章了解更多详细学习方法。

20

2026.01.29

java配置环境变量教程合集
java配置环境变量教程合集

本专题整合了java配置环境变量设置、步骤、安装jdk、避免冲突等等相关内容,阅读专题下面的文章了解更多详细操作。

18

2026.01.29

java成品学习网站推荐大全
java成品学习网站推荐大全

本专题整合了java成品网站、在线成品网站源码、源码入口等等相关内容,阅读专题下面的文章了解更多详细推荐内容。

19

2026.01.29

Java字符串处理使用教程合集
Java字符串处理使用教程合集

本专题整合了Java字符串截取、处理、使用、实战等等教程内容,阅读专题下面的文章了解详细操作教程。

3

2026.01.29

Java空对象相关教程合集
Java空对象相关教程合集

本专题整合了Java空对象相关教程,阅读专题下面的文章了解更多详细内容。

6

2026.01.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
微信小程序开发之API篇
微信小程序开发之API篇

共15课时 | 1.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号