jjzjj

php - 受限笛卡尔积计算 - PHP

coder 2024-04-06 原文

编辑 1 -自从发布后我了解到基本问题是关于如何找到笛卡尔积(现在去谷歌),但不仅因为我不想要每个烫发,我想找到使用相同子数组的笛卡尔积每次排列的关键永远不会超过一次,我的“额外”问题更多地是关于如何最大限度地减少笛卡尔积所需的工作量(我不得不说,接受小的错误率)-

想象一下......我有四个厨师和四个食谱,每个厨师对每个食谱都有一个分数,今天我希望每个厨师做一道菜(但没有一道菜应该做两次)并且决定应该基于最好的(最高总分)所有四个的排列(所以也许一个厨师不会做出他个人的最好成绩)。

我已将数据放入多维数组中

 array(
   array (1,2,3,4),
   array (35,0,0,0),
   array (36,33,1,1),
   array (20,20,5,3)
 )
  • 它在每个子数组中的值对数量与子数组的数量相同(如果有帮助的话)
  • 实际上,子阵列的数量将达到最大值 8(因此最大 perms = 8!,大约 40,000 不是 8^8,因为不允许许多组合)
  • 如果有帮助的话,选择这种格式的数据是灵活的

  • 我正在尝试创建第二个数组,该数组将根据 KEY 输出子数组的最佳(即最高值)可能的组合,其中每个子数组中只能使用一个

    -- 所以这里每个子数组 [0][1][2][3] 将在每个排列中使用一次
    并且每个 subarrayKey [0][1][2][3] 将在每个排列中使用一次,在我的实际问题中,我使用的是关联数组,但这对于这个问题来说是额外的。--

    所以这个例子会创建一个数组
    newArray (35,33,5,4)//注意 [2][0] 没有被使用

    理想情况下,我宁愿不生产所有烫发,而是以某种方式丢弃许多显然不是最合适的组合。

    关于如何开始的任何想法?我会接受伪代码。

    有关笛卡尔积的 SO 示例,请参阅 PHP 2D Array output all combinations

    编辑 2
    有关使笛卡尔积更有效的更多信息,以及如果您想查看是否可以偷工减料(有风险)Efficient Cartesian Product algorithm 为什么必须针对具体情况

    最佳答案

    抱歉,但这将更多地是逻辑布局而不是代码......

    我不太清楚数组(1,2,3,4)是第一道菜还是第一道菜的分数,但我可能会使用这样的数组

    $array[$cook_id][$dish_number] = $score;
    

    asort() 每个数组,这样 $array[$cook_id] = array($lowest_scored_dish,...,$highest);

    考虑一个特定厨师做一道菜的加权偏好是最好的菜和另一道菜的分数之间的差异。

    作为一个非常简单的例子, cooking a,b,c 和菜肴 0,1,2
    $array['a'] = array(0=>100, 1=>50, 2=>0); // cook a prefers 0 over 1 with weight 50, over 2 with weight 100
    $array['b'] = array(0=>100, 1=>100, 2=>50); // cook b prefers 0,1 over 2 with weight 50
    $array['c'] = array(0=>50, 1=>50, 2=>100); // cook c prefers 2 with weight 50
    

    在 asort() 之后:
    $array['a'] = array(0=>100, 1=>50, 2=>0);
    $array['b'] = array(0=>100, 1=>100, 2=>50);
    $array['c'] = array(2=>100, 0=>50, 1=>50);

    从厨师 'a' 开始,他更喜欢第 0 道菜而不是他的下一个最好的菜 50 分(重量)。 Cook 'b' 也更喜欢 dih 0,但下一道菜的权重为 0。因此很可能(虽然还不确定厨师 'a' 应该制作第 0 道菜。

    考虑保留第 0 道菜,然后继续 cooking 'b'。除了菜 0,厨师 'b' 喜欢菜 1。没有其他厨师喜欢菜 1,所以厨师 'b' 被分配到菜 1。

    Cook 'c' 默认获得第 2 道菜。

    这是一个非常方便的例子,每个厨师都可以做一些个人最大的事情,但我希望它可以说明一些可行的逻辑。

    让我们让它不那么方便:
    $array['a'] = array(0=>75, 1=>50, 2=>0);
    $array['b'] = array(0=>100, 1=>50, 2=>50);
    $array['c'] = array(0=>100, 1=>25, 2=>25);
    

    再次开始 cooking 'a',看到 0 是首选,但这次重量为 25。厨师 'b' 喜欢重量为 50,厨师 'c' 喜欢重量为 75。厨师 'c' 赢得第 0 道菜.

    回到可用厨师列表,“a”更喜欢权重为 50 的 1,但“b”更喜欢权重为 0。“a”得到第 1 道菜,“b”得到第 2 道菜。

    这仍然不能解决所有的复杂问题,但这是朝着正确方向迈出的一步。有时,对第一个厨师/菜肴组合所做的假设是错误的。

    不太方便:
    $array['a'] = array(0=>200, 1=>148, 2=>148, 3=>0);
    $array['b'] = array(0=>200, 1=>149, 2=>0, 3=>0);
    $array['c'] = array(0=>200, 1=>150, 2=>147, 3=>147);
    $array['d'] = array(0=>69, 1=>18, 2=>16, 3=>15);
    

    'a' 得到 0,因为这是最大值并且没有其他喜欢 0 的人有更高的权重
    'b' 以 149 的权重赢得 1
    'd' 获胜 2,因为 'c' 对可用选项没有偏好
    'c' 得到 3

    得分:200+149+147+16 = 512

    虽然这是在没有检查每个排列的情况下收集的一个很好的猜测,但它可能是错误的。从这里开始,问:“如果一位厨师与其他任何一位厨师进行交易,总数会增加吗?”

    答案是肯定的,a(0)+d(2) = 200+16 = 216,但是 a(2)+d(0) = 148+69 = 217。

    我将让您使用加权方法为“最佳猜测”编写代码,但在此之后,这对您来说是一个好的开始:
    // a totally uneducated guess...
    $picks = array(0=>'a', 1=>'b', 2=>'c', 3=>'d');
    
    do {
        $best_change = false;
        $best_change_weight = 0;
        foreach ($picks as $dish1 => $cook1) {
            foreach ($picks as $dish2 => $cook2) {
                if (($array[$cook1][$dish1] + $array[$cook2][$dish2]) <
                    ($array[$cook1][$dish2] + $array[$cook2][$dish1]))
                {
                    $old_score = $array[$cook1][$dish1] + $array[$cook2][$dish2];
                    $new_score = $array[$cook1][$dish2] + $array[$cook2][$dish1];
                    if (($new_score - $old_score) > $best_change_weight) {
                        $best_change_weight = $new_score - $old_score;
                        $best_change = $dish2;
                    }
                }
            }
            if ($best_change !== false) {
                $cook2 = $picks[$best_change];
                $picks[$dish1] = $cook2;
                $picks[$dish2] = $cook1;
                break;
            }
        }
    } while ($best_change !== false);
    

    我找不到反例来证明这不起作用,但我怀疑这种情况
    ($array[$cook1][$dish1] + $array[$cook2][$dish2])
    ==
    ($array[$cook1][$dish2] + $array[$cook2][$dish1])

    也许其他人会跟进这个“如果?”的答案。

    给定这个矩阵,括号中的项目是“选择”
    [a1]   a2   a3
     b1   [b2]  b3
     c1    c2  [c3]
    

    如果 a1 + b2 == a2 + b1,则 'a' 和 'b' 不会切换菜品。我不是 100% 确定的情况是,是否存在一个矩阵使得这是一个更好的选择:
     a1   [a2]   a3
     b1    b2   [b3]
    [c1]   c2    c3
    

    从第一个状态到第二个状态需要两个开关,第一个似乎是任意的,因为它不会改变总数。但是,只有通过这种任意更改才能进行最后一次切换。

    我试图找到一个 3x3 的例子,这样基于我在上面写的“加权偏好”模型,第一个将被选择,但真正的最佳选择由第二个给出。我找不到示例,但这并不意味着它不存在。我现在不想做更多的矩阵代数,但也许有人会从我离开的地方开始。哎呀,也许这种情况不存在,但我想我应该指出这一点。

    如果它确实有效并且您从正确的选择开始,则上面的代码将只循环 64 次(8x8),用于 8 个厨师/菜肴。如果选择不正确,第一个厨师有变化,那么它会上升到 72。如果第 8 个厨师有变化,它会上升到 128。 do-while 可能会循环几次,但我怀疑它将接近对所有 40k 组合求和所需的 CPU 周期。

    关于php - 受限笛卡尔积计算 - PHP,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16231244/

    有关php - 受限笛卡尔积计算 - PHP的更多相关文章

    1. ruby-on-rails - 使用一系列等级计算字母等级 - 2

      这里是Ruby新手。完成一些练习后碰壁了。练习:计算一系列成绩的字母等级创建一个方法get_grade来接受测试分数数组。数组中的每个分数应介于0和100之间,其中100是最大分数。计算平均分并将字母等级作为字符串返回,即“A”、“B”、“C”、“D”、“E”或“F”。我一直返回错误:avg.rb:1:syntaxerror,unexpectedtLBRACK,expecting')'defget_grade([100,90,80])^avg.rb:1:syntaxerror,unexpected')',expecting$end这是我目前所拥有的。我想坚持使用下面的方法或.join,

    2. 计算机毕业设计ssm+vue基本微信小程序的小学生兴趣延时班预约小程序 - 2

      项目介绍随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱小学生兴趣延时班预约小程序的设计与开发被用户普遍使用,为方便用户能够可以随时进行小学生兴趣延时班预约小程序的设计与开发的数据信息管理,特开发了小程序的设计与开发的管理系统。小学生兴趣延时班预约小程序的设计与开发的开发利用现有的成熟技术参考,以源代码为模板,分析功能调整与小学生兴趣延时班预约小程序的设计与开发的实际需求相结合,讨论了小学生兴趣延时班预约小程序的设计与开发的使用。开发环境开发说明:前端使用微信微信小程序开发工具:后端使用ssm:VU

    3. ruby - 如何计算 Liquid 中的变量 +1 - 2

      我对如何计算通过{%assignvar=0%}赋值的变量加一完全感到困惑。这应该是最简单的任务。到目前为止,这是我尝试过的:{%assignamount=0%}{%forvariantinproduct.variants%}{%assignamount=amount+1%}{%endfor%}Amount:{{amount}}结果总是0。也许我忽略了一些明显的东西。也许有更好的方法。我想要存档的只是获取运行的迭代次数。 最佳答案 因为{{incrementamount}}将输出您的变量值并且不会影响{%assign%}定义的变量,我

    4. ruby - 使用 Ruby,计算 n x m 数组的每一列中有多少个 true 的简单方法是什么? - 2

      给定一个nxmbool数组:[[true,true,false],[false,true,true],[false,true,true]]有什么简单的方法可以返回“该列中有多少个true?”结果应该是[1,3,2] 最佳答案 使用转置得到一个数组,其中每个子数组代表一列,然后将每一列映射到其中的true数:arr.transpose.map{|subarr|subarr.count(true)}这是一个带有inject的版本,应该在1.8.6上运行,没有任何依赖:arr.transpose.map{|subarr|subarr.in

    5. arrays - 计算数组中的匹配元素 - 2

      给定两个大小相等的数组,如何找到不考虑位置的匹配元素的数量?例如:[0,0,5]和[0,5,5]将返回2的匹配项,因为有一个0和一个5共同;[1,0,0,3]和[0,0,1,4]将返回3的匹配项,因为0有两场,1有一场;[1,2,2,3]和[1,2,3,4]将返回3的匹配项。我尝试了很多想法,但它们都变得相当粗糙和令人费解。我猜想有一些不错的Ruby习惯用法,或者可能是一个正则表达式,可以很好地回答这个解决方案。 最佳答案 您可以使用count完成它:a.count{|e|index=b.index(e)andb.delete_at

    6. ruby-on-rails - 如何计算 Ruby/Rails 中 JSON 对象的数量 - 2

      Ruby中如何“一般地”计算以下格式(有根、无根)的JSON对象的数量?一般来说,我的意思是元素可能不同(例如“标题”被称为其他东西)。没有根:{[{"title":"Post1","body":"Hello!"},{"title":"Post2","body":"Goodbye!"}]}根包裹:{"posts":[{"title":"Post1","body":"Hello!"},{"title":"Post2","body":"Goodbye!"}]} 最佳答案 首先,withoutroot代码不是有效的json格式。它将没有包

    7. ruby - 如何计算自 Ruby 中给定日期以来的周数? - 2

      目标我正在尝试计算自给定日期以来周的距离,而无需跳过任何步骤。我更喜欢用普通的Ruby来做,但ActiveSupport无疑是一个可以接受的选择。我的代码我写了以下内容,这似乎可行,但对我来说似乎还有很长的路要走。require'date'DAYS_IN_WEEK=7.0defweeks_sincedate_stringdate=Date.parsedate_stringdays=Date.today-dateweeks=days/DAYS_IN_WEEKweeks.round2endweeks_since'2015-06-15'#=>32.57ActiveSupport的#weeks

    8. 最新版人脸识别小程序 图片识别 生成二维码签到 地图上选点进行位置签到 计算签到距离 课程会议活动打卡日常考勤 上课签到打卡考勤口令签到 - 2

      技术选型1,前端小程序原生MINA框架cssJavaScriptWxml2,管理后台云开发Cms内容管理系统web网页3,数据后台小程序云开发云函数云开发数据库(基于MongoDB)云存储4,人脸识别算法基于百度智能云实现人脸识别一,用户端效果图预览老规矩我们先来看效果图,如果效果图符合你的需求,就继续往下看,如果不符合你的需求,可以跳过。1-1,登录注册页可以看到登录页有注册入口,注册页如下我们的注册,需要管理员审核,审核通过后才可以正常登录使用小程序1-2,个人中心页登录成功以后,我们会进入个人中心页我们在个人中心页可以注册人脸,因为我们做人脸识别签到,需要先注册人脸才可以进行人脸比对,进

    9. ruby-on-rails - 这个 C 和 PHP 程序员如何学习 Ruby 和 Rails? - 2

      按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visitthehelpcenter指导。关闭9年前。我来自C、php和bash背景,很容易学习,因为它们都有相同的C结构,我可以将其与我已经知道的联系起来。然后2年前我学了Python并且学得很好,Python对我来说比Ruby更容易学。然后从去年开始,我一直在尝试学习Ruby,然后是Rails,我承认,直到现在我还是学不会,讽刺的是那些打着简单易学的烙印,但是对于我这样一个老练的程序员来说,我只是无法将它

    10. ruby - 如何计算两个字符串共有的字符数? - 2

      如何计算两个字符串之间的字符交集?例如(假设我们有一个名为String.intersection的方法):"abc".intersection("ab")=2"hello".intersection("hallo")=4好的,男孩女孩们,感谢你们的大量反馈。更多示例:"aaa".intersection("a")=1"foo".intersection("bar")=0"abc".intersection("bc")=2"abc".intersection("ac")=2"abba".intersection("aa")=2一些补充说明:维基百科定义intersection如下:Int

    随机推荐