1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43
java解法
每条消息以 (sender, receiver) 的形式表示,分别为发送方和接收方的 ID。 给定一个目标用户 ID tid,需要判定该用户是否为垃圾发送者。 垃圾发送者的判定规则: tid 发送的接收方中,有超过 5 个未回复的接收方。 tid 总发送消息数量减去总接收消息数量超过 10。 对任意接收方,如果 tid 向其发送的消息数量比从其接收的消息数量多 5 条以上,则为垃圾发送者。 算法设计
使用 集合 和 映射 数据结构统计消息的发送和接收关系: sentTo:tid 发送消息的所有接收方。 receivedFrom:tid 接收消息的所有发送方。 sendCount:tid 对每个接收方发送的消息数量。 receiveCount:每个发送方对 tid 发送的消息数量。 根据统计数据,计算以下结果: 未回复的接收方数量 l。 总发送消息数量与接收消息数量的差值 m。 是否满足垃圾发送者判定规则。 实现细节
使用 SpamDetector 类封装判定逻辑,包括消息处理、垃圾发送者判定,以及结果计算方法。 在主函数中读取输入,调用 SpamDetector 执行判定。
import java. util. * ;
public class Main {
public static void main ( String [ ] args) {
Scanner scanner = new Scanner ( System . in) ;
int n = scanner. nextInt ( ) ;
List < int [ ] > messages = new ArrayList < > ( ) ;
for ( int i = 0 ; i < n; i++ ) {
int [ ] entry = new int [ 2 ] ;
entry[ 0 ] = scanner. nextInt ( ) ;
entry[ 1 ] = scanner. nextInt ( ) ;
messages. add ( entry) ;
}
int targetId = scanner. nextInt ( ) ;
System . out. println ( checkSpammer ( targetId, messages) ) ;
}
public static String checkSpammer ( int tid, List < int [ ] > messages) {
SpamDetector detector = new SpamDetector ( tid) ;
for ( int [ ] msg : messages) {
detector. processMessage ( msg[ 0 ] , msg[ 1 ] ) ;
}
return detector. isSpammer ( ) + " " + detector. getL ( ) + " " + detector. getM ( ) ;
}
}
class SpamDetector {
private int tid;
private Set < Integer > sentTo = new HashSet < > ( ) ;
private Set < Integer > receivedFrom = new HashSet < > ( ) ;
private Map < Integer , Integer > sendCount = new HashMap < > ( ) ;
private Map < Integer , Integer > receiveCount = new HashMap < > ( ) ;
public SpamDetector ( int tid) {
this . tid = tid;
}
public void processMessage ( int sender, int receiver) {
if ( sender == tid) {
sentTo. add ( receiver) ;
sendCount. put ( receiver, sendCount. getOrDefault ( receiver, 0 ) + 1 ) ;
} else if ( receiver == tid) {
receivedFrom. add ( sender) ;
receiveCount. put ( sender, receiveCount. getOrDefault ( sender, 0 ) + 1 ) ;
}
}
public boolean isSpammer ( ) {
int l = sentTo. size ( ) - ( int ) sentTo. stream ( ) . filter ( receivedFrom:: contains ) . count ( ) ;
int m = sendCount. values ( ) . stream ( ) . mapToInt ( Integer :: intValue ) . sum ( ) -
receiveCount. values ( ) . stream ( ) . mapToInt ( Integer :: intValue ) . sum ( ) ;
if ( l > 5 || m > 10 ) {
return true ;
}
for ( int id : sentTo) {
if ( sendCount. get ( id) - receiveCount. getOrDefault ( id, 0 ) > 5 ) {
return true ;
}
}
return false ;
}
public int getL ( ) {
return sentTo. size ( ) - ( int ) sentTo. stream ( ) . filter ( receivedFrom:: contains ) . count ( ) ;
}
public int getM ( ) {
return sendCount. values ( ) . stream ( ) . mapToInt ( Integer :: intValue ) . sum ( ) -
receiveCount. values ( ) . stream ( ) . mapToInt ( Integer :: intValue ) . sum ( ) ;
}
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127
C++解法
给定一个消息记录的列表,记录的形式为 (sender, receiver),表示某个发送者向某个接收者发送了一条消息。 给定一个目标用户 ID targetId,需要分析该用户是否为垃圾发送者,并输出以下三个结果: 是否为垃圾发送者(true 或 false)。 L:目标用户发送的接收者中未回复的接收者数量。 M:目标用户的总发送消息数量减去总接收消息数量的差值。 垃圾发送者判定规则
满足以下任意一条规则,即判定为垃圾发送者: L > 5,即未回复的接收者数量超过 5。 M > 10,即发送和接收的消息数量差值超过 10。 对任意接收者,如果发送消息数量比接收到的消息数量多 5 条以上,则判定为垃圾发送者。 算法设计
遍历消息记录,统计目标用户的发送和接收情况: 使用 unordered_set 分别记录目标用户发送消息的接收者和接收消息的发送者。 使用 unordered_map 记录每个接收者收到的消息数量和每个发送者发送给目标用户的消息数量。 累计目标用户的总发送消息数量和接收消息数量。 计算 L 和 M: L 为发送集合中未被接收集合覆盖的部分。 M 为总发送消息数量减去总接收消息数量。 判断是否满足垃圾发送者条件
# include
# include
# include
# include
using namespace std;
int main ( ) {
int entryCount;
cin >> entryCount;
vector< pair< int , int >> records ( entryCount) ;
for ( int i = 0 ; i < entryCount; i++ ) {
int sender, receiver;
cin >> sender >> receiver;
records[ i] = { sender, receiver} ;
}
int targetId;
cin >> targetId;
unordered_set< int > sentTo;
unordered_set< int > receivedFrom;
unordered_map< int , int > sentCount;
unordered_map< int , int > receivedCount;
int sentMessages = 0 ;
int receivedMessages = 0 ;
for ( auto & record : records) {
if ( record. first == targetId) {
sentTo. insert ( record. second) ;
sentMessages++ ;
sentCount[ record. second] ++ ;
} else if ( record. second == targetId) {
receivedMessages++ ;
receivedFrom. insert ( record. first) ;
receivedCount[ record. first] ++ ;
}
}
for ( auto & id : receivedFrom) {
sentTo. erase ( id) ;
}
int L = sentTo. size ( ) ;
int M = sentMessages - receivedMessages;
bool isSpam = ( L > 5 ) || ( M > 10 ) ;
for ( auto & [ receiver, count] : sentCount) {
if ( receivedCount. count ( receiver) && ( count - receivedCount[ receiver] > 5 ) ) {
isSpam = true ;
break ;
}
}
cout << boolalpha << isSpam << " " << L << " " << M << endl;
return 0 ;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67
C解法
更新中
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
JS解法
给定若干条消息日志,每条日志记录了消息的发送者和接收者 (sender, receiver)。 需要分析一个指定的用户 targetId 的消息行为,判断其是否为垃圾短信 发送者。 判断规则: 目标用户的独立接收者数量 L > 5。 目标用户发送的总消息数量减去接收到的总消息数量 M > 10。 目标用户对某个接收者发送的消息数量比从该接收者接收到的消息数量多出 5 条。 输入输出格式
输入: 第一行是消息日志的条目数 n。 接下来 n 行,每行包含一条消息日志,格式为 sender receiver。 最后一行是目标用户的 ID targetId。 输出: 格式为 “isSpammer L M”,其中: isSpammer 是 true 或 false,表示目标用户是否为垃圾短信发送者; L 是独立接收者数量; M 是总发送消息数量与接收消息数量的差值。 算法设计
使用两个哈希表统计消息记录: outgoingMessages:记录每个用户的发送消息详情,包括总发送量和独立接收者集合。 incomingMessages:记录每个用户接收消息的总数。 统计目标用户的 L 和 M 值: L 为目标用户的独立接收者数量。 M 为目标用户的发送消息数量减去接收消息数量。 检查垃圾发送者的规则,逐条判断是否满足
const readline = require ( "readline" ) ;
const rl = readline. createInterface ( {
input : process. stdin,
output : process. stdout,
} ) ;
let inputs = [ ] ;
let totalRecords;
rl. on ( "line" , ( line ) => {
inputs. push ( line) ;
if ( inputs. length === 1 ) {
totalRecords = parseInt ( inputs[ 0 ] , 10 ) ;
}
if ( inputs. length === totalRecords + 2 ) {
inputs. shift ( ) ;
const targetId = inputs. pop ( ) ;
const messageLogs = inputs. map ( ( line ) => line. split ( " " ) . map ( Number) ) ;
console. log ( analyzeMessages ( parseInt ( targetId, 10 ) , messageLogs) ) ;
inputs = [ ] ;
}
} ) ;
function analyzeMessages ( targetId, messageLogs ) {
const outgoingMessages = { } ;
const incomingMessages = { } ;
messageLogs. forEach ( ( [ sender, receiver] ) => {
if ( ! outgoingMessages[ sender] ) {
outgoingMessages[ sender] = { totalSent : 0 , uniqueReceivers : new Set ( ) } ;
}
if ( ! incomingMessages[ receiver] ) {
incomingMessages[ receiver] = 0 ;
}
outgoingMessages[ sender] . totalSent++ ;
outgoingMessages[ sender] . uniqueReceivers. add ( receiver) ;
incomingMessages[ receiver] ++ ;
} ) ;
const totalSent = outgoingMessages[ targetId] ?. totalSent || 0 ;
const totalReceived = incomingMessages[ targetId] || 0 ;
const uniqueReceivers = outgoingMessages[ targetId] ?. uniqueReceivers. size || 0 ;
const L = uniqueReceivers;
const M = totalSent - totalReceived;
let isSpammer = L > 5 || M > 10 ;
if ( ! isSpammer) {
for ( let receiver of outgoingMessages[ targetId] ?. uniqueReceivers || [ ] ) {
const sentToReceiver = outgoingMessages[ targetId] . totalSent;
const receivedFromReceiver = incomingMessages[ receiver] || 0 ;
if ( sentToReceiver - receivedFromReceiver > 5 ) {
isSpammer = true ;
break ;
}
}
}
return ` ${ isSpammer} ${ L } ${ M } ` ;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
注意:
如果发现代码有用例覆盖不到的情况,欢迎反馈!会在第一时间修正,更新。 解题不易,如对您有帮助,欢迎点赞/收藏
评论记录:
回复评论: