подсчет пар в списке
Я недавно начал использовать HackerRank и пытаюсь "Продажи по совпадениям". Я пришел к решению, которым я доволен с точки зрения использования возможностей программирования функций Kotlin. Однако я не получаю ожидаемого ответа...
Краткое описание проблемы:
Для массива: -> найти и вернуть общее количество пар.
то есть:
ввод -> [10, 20, 20, 10, 10, 30, 50, 10, 20]
количество пар -> 3
вот мой код и несколько поясняющих его комментариев:
fun sockMerchant(n: Int, pile: Array<Int>): Int{
var count = 0
mutableMapOf<Int, Int>().withDefault { 0 }.apply {
// the [attempted] logic behind this piece of code here is
// that as we iterate through the list, the 'getOrPut()'
// function will return either the value for the given [key]
// or throw an exception if there is no such key
// in the map. However, since our map was created by
// [withDefault], the function resorts to its `defaultValue` <-> 0
// instead of throwing an exception.
for (e in values) {
// this simplifies our code and inserts a zero [+1] where needed.
// if (key exists)
// // return its associated value and MOD it:
// case: even -> increment counter
// else -> do nothing
// else if (key dne)
// // insert default value <-> [0] + 1
// ....
// ....
// ....
if (getOrPut(e, { getValue(e) + 1 } ) % 2 == 0) count++
}
}
return count
}
fun main(args: Array<String>) {
val scan = Scanner(System.`in`)
val n = scan.nextLine().trim().toInt()
val ar = scan.nextLine().split(" ").map{ it.trim().toInt() }.toTypedArray()
val result = sockMerchant(n, ar)
println(result)
}
- Любая помощь или советы будут иметь большое значение здесь:)
2 ответа
Я немного изменил его, чтобы его было легче тестировать, но вот исправления:
import java.util.*
fun sockMerchant(n: Int, pile: Array<Int>): Int{
var count = 0
mutableMapOf<Int, Int>().withDefault { 0 }.apply {
// the [attempted] logic behind this piece of code here is
// that as we iterate through the list, the 'getOrPut()'
// function will return either the value for the given [key]
// or throw an exception if there is no such key
// in the map. However, since our map was created by
// [withDefault], the function resorts to its `defaultValue` <-> 0
// instead of throwing an exception.
for (e in pile) {
// this simplifies our code and inserts a zero [+1] where needed.
// if (key exists)
// // return its associated value and MOD it:
// case: even -> increment counter
// else -> do nothing
// else if (key dne)
// // insert default value <-> [0] + 1
// ....
// ....
// ....
println(e)
put(e, getValue(e) + 1)
if (getValue(e) % 2 == 0) count++
println(entries)
}
}
return count
}
val n = 5
val ar = "10 10 10 10 20 20 30 40".split(" ").map{ it.trim().toInt() }.toTypedArray()
val result = sockMerchant(n, ar)
println(result)
Вывод:
10
[10=1]
10
[10=2]
10
[10=3]
10
[10=4]
20
[10=4, 20=1]
20
[10=4, 20=2]
30
[10=4, 20=2, 30=1]
40
[10=4, 20=2, 30=1, 40=1]
3
Pair.kts:3:18: warning: parameter 'n' is never used
fun sockMerchant(n: Int, pile: Array<Int>): Int{
^
Process finished with exit code 0
Пояснение:
- Вы перебирали "значения", которые в начале пустые, поэтому вы никогда ничего не делали со своим кодом.
- Даже при зацикливании
pile
, ваша логика увеличения не превышает 1, поэтому условие никогда не было выполнено, и счетчик не увеличивался.
Но основная причина была правильной.
Я смог сделать это, сгруппировав числа вместе, взяв полученные списки и просуммировав, сколько пар содержит каждый из них:
fun sockMerchant(n: Int, pile: Array<Int>): Int =
pile.groupBy { it }.values.sumBy { it.size / 2 }
После того, как мы сделаем
pile.groupBy { it }
, у нас есть такая структура:
{10=[10, 10, 10, 10], 20=[20, 20, 20], 30=[30], 50=[50]}
Мы берем значения и суммируем их по размеру, разделив их на 2. Это округлит полупары до 0 и полные пары до 1 каждая.
Примечание: мне не совсем понятно, какова цель
n
есть в этом случае.