- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我经常教授大型入门编程类(class)(400 - 600 名学生),当考试时间临近时,我们通常不得不将类(class)分成不同的房间,以确保每个人都有座位参加考试。
为了在逻辑上保持简单,我通常按姓氏将类(class)分开。例如,我可能会将姓氏 A - H 的学生送到一个房间,将姓氏 I - L 送到第二个房间,将姓氏 M - S 送到第三个房间,将姓氏 T - Z 送到第四个房间。
这样做的挑战在于,房间的容纳人数通常大相径庭,而且很难找到一种让每个人都能适应的方式来划分类(class)。例如,假设姓氏的分布(为简单起见)如下:
假设我有容量为 350、50 和 50 的房间。用于查找房间分配的贪婪算法可能是将房间按容量降序排列,然后尝试按该顺序填充房间。不幸的是,这并不总是有效。例如,在这种情况下,正确的选择是将姓氏 A 放入一个大小为 50 的房间,将姓氏 B - C 放入大小为 350 的房间,将姓氏 D 放入另一个大小为 50 的房间。贪心算法将将姓氏 A 和 B 放入 350 人的房间,然后无法为其他人找到座位。
只需尝试房间排序的所有可能排列,然后对每个排序运行贪心算法,即可轻松解决此问题。这将找到有效的分配或报告不存在。但是,考虑到房间数量可能在 10 到 20 之间并且检查所有排列可能不可行,我想知道是否有更有效的方法来执行此操作。
总而言之,正式的问题陈述如下:
You are given a frequency histogram of the last names of the students in a class, along with a list of rooms and their capacities. Your goal is to divvy up the students by the first letter of their last name so that each room is assigned a contiguous block of letters and does not exceed its capacity.
是否有一个有效的算法,或者至少有一个对合理的房间大小有效的算法?
编辑:很多人都问过连续条件。规则是
例如,您不能将 A - E、H - N 和 P - Z 放入同一个房间。您也不能将 A - C 放在一个房间,而将 B - D 放在另一个房间。
谢谢!
最佳答案
可以在 [m, 2^n]
空间上使用某种 DP 解决方案来解决,其中 m
是字母数(英语为 26)并且 n
是房间数。使用 m == 26
和 n == 20
它将占用大约 100 MB 的空间和大约 1 秒的时间。下面是我刚刚用 C# 实现的解决方案(它也可以在 C++ 和 Java 上成功编译,只需要做一些小的改动):
int[] GetAssignments(int[] studentsPerLetter, int[] rooms)
{
int numberOfRooms = rooms.Length;
int numberOfLetters = studentsPerLetter.Length;
int roomSets = 1 << numberOfRooms; // 2 ^ (number of rooms)
int[,] map = new int[numberOfLetters + 1, roomSets];
for (int i = 0; i <= numberOfLetters; i++)
for (int j = 0; j < roomSets; j++)
map[i, j] = -2;
map[0, 0] = -1; // starting condition
for (int i = 0; i < numberOfLetters; i++)
for (int j = 0; j < roomSets; j++)
if (map[i, j] > -2)
{
for (int k = 0; k < numberOfRooms; k++)
if ((j & (1 << k)) == 0)
{
// this room is empty yet.
int roomCapacity = rooms[k];
int t = i;
for (; t < numberOfLetters && roomCapacity >= studentsPerLetter[t]; t++)
roomCapacity -= studentsPerLetter[t];
// marking next state as good, also specifying index of just occupied room
// - it will help to construct solution backwards.
map[t, j | (1 << k)] = k;
}
}
// Constructing solution.
int[] res = new int[numberOfLetters];
int lastIndex = numberOfLetters - 1;
for (int j = 0; j < roomSets; j++)
{
int roomMask = j;
while (map[lastIndex + 1, roomMask] > -1)
{
int lastRoom = map[lastIndex + 1, roomMask];
int roomCapacity = rooms[lastRoom];
for (; lastIndex >= 0 && roomCapacity >= studentsPerLetter[lastIndex]; lastIndex--)
{
res[lastIndex] = lastRoom;
roomCapacity -= studentsPerLetter[lastIndex];
}
roomMask ^= 1 << lastRoom; // Remove last room from set.
j = roomSets; // Over outer loop.
}
}
return lastIndex > -1 ? null : res;
}
来自 OP 问题的示例:
int[] studentsPerLetter = { 25, 150, 200, 50 };
int[] rooms = { 350, 50, 50 };
int[] ans = GetAssignments(studentsPerLetter, rooms);
答案是:
2
0
0
1
其中表示每个学生姓氏字母的空间索引。如果无法分配,我的解决方案将返回 null
。
[编辑]
经过数千次自动生成的测试后,我的 friend 在代码中发现了一个向后构造解决方案的错误。它不影响主要算法,因此修复此错误将成为读者的练习。
揭示错误的测试用例是 students = [13,75,21,49,3,12,27,7]
和 rooms = [6,82,89, 6,56]
。我的解决方案没有返回任何答案,但实际上有一个答案。请注意,解决方案的第一部分工作正常,但答案构建部分失败。
关于algorithm - 按姓氏将人们分配到房间?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16377079/
我在使用 Room 或 idk 时遇到问题,实际上问题出在哪里,我需要帮助找出问题出在哪里,我正在使用 Hilt DI,创建数据库实例的那一刻它崩溃了这是我的代码 错误 E/AndroidRuntim
我有一个关于数据结构和类设计的问题(抱歉太长了)。为了简单起见,假设这是一个游戏,我想在房间之间导航(想象一系列 2D 非滚动屏幕,例如早期的银河战士/恶魔城)。每个房间可以有许多导出(例如上、下、左
我使用当前日期(1-25)作为父ID并使用房间(08-00_11-00_karpet1-)作为 child ID。该数据库中包含在该日期(父 ID)订购该房间(子 id)的用户信息。 问题1 使用此布
在我的 Android 项目中,我使用 Room 库来处理 SQLite 数据库。我使用我的数据库来保存国家电话代码。我的数据库预装了两个国家(观看 populateDatabaseWithCount
我正在尝试将 Room 持久性库添加到 Android 应用程序项目中。在 build.gradle 文件中,我添加了以下依赖项: implementation 'android.arch.persi
人们可以提前从 25 场讲座中选择最多 5 场。所有这些讲座都在五个房间的五个时间段内在一天内进行。听众可以参加的每个(首选)讲座都让她更快乐,他选择但不能参加的每个讲座(因为另一个首选讲座在同一时间
我在 Android 上使用 OrmLite 而不是 SQLite 和 SQLCipher 来加密数据库。有没有办法加密 Room 数据库? 最佳答案 默认情况下,Room 将数据存储在应用程序的内部
使用 Room ORM,我使用 @Entity 注释声明了一个实体 EQPreset。该实体包含一个数组 int[]。它给出以下错误: 错误:无法确定如何将此字段 (int[] arr) 保存到数据库
我正在尝试构建一个管理 child 托儿所的应用程序,特别是管理哪个 child 在哪个时间点在托儿所的哪个房间里。 Nursery 链式店有多个分支机构。每个分店有几个房间,每个房间对应一个年龄段,
我在生产环境中遇到了“android.database.sqlite.SQLiteDatabaseLockedException”异常。错误分析时出现异常。我的项目数据库有空间。项目中没有使用多进程。
我想实现 Android Room 持久性。 这是我的 DAO 界面。 @Dao interface FoodDao { /** * Returns all data in tabl
我正在尝试使用 Room 数据库和 LiveData。我有 ViewModels,它保存从 dao 获得的 LiveData。如果我更新Transaction ,然后LiveData>观察正常,但是
在 Firebase ,创建“房间”(例如用于聊天)很容易,正如其各种示例中所记录的那样。 对于聊天的数据结构,我会使用这样的东西: rooms room1 member_co
我试图从 Activity 中将一行插入 SQLITE 数据库,然后返回要存储在 Activity 中的变量中的 rowId。请参阅下面我使用的方法和逻辑。 private void insert
我正在使用 XMPPFramework 开发聊天应用程序 加入现有房间后如何接收消息历史记录? 现在我像这样加入房间: XMPPJID *roomJid = [XMPPJID jidWithStrin
我在我的应用程序中使用 Room 并将数据插入到我的数据库中时 ConcurrentModificationException有时会被抛出。为什么会这样? 我使用分页 api,在每次 api 调用后,
我想为 pb 添加值,由于将 pb_value 包含到实体中,应用程序崩溃了。我是学习室的新手,我不确定将额外项目合并到数据库中的正确方法。 E/AndroidRuntime: FATAL EXCEP
我想在当前 pb 中添加值、日期和详细信息。我在 pbInfo 的数据库中收到错误“冲突声明”。我应该如何修复此错误? @Entity(tableName = "pb_table") data cla
我正在尝试制作一个聊天应用程序,用户可以在其中聊天。我想将两个用户 uid 字符串插入函数并返回一个连接的字符串。但我希望以某种方式组织 uid,以便返回的值始终相同。 func (id1, id2)
我不断收到以下错误: Cannot figure out how to save this field into database. You can consider adding a type co
我是一名优秀的程序员,十分优秀!