|
|
|
// Copyright 2019-present the Flutter authors. All Rights Reserved.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
import 'dart:isolate';
|
|
|
|
import 'dart:math';
|
|
|
|
import 'dart:typed_data';
|
|
|
|
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:provider/provider.dart';
|
|
|
|
|
|
|
|
class DataTransferPageStarter extends StatelessWidget {
|
|
|
|
const DataTransferPageStarter({super.key});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return ChangeNotifierProvider(
|
|
|
|
create: (context) => DataTransferIsolateController(),
|
|
|
|
child: const DataTransferPage(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class DataTransferPage extends StatelessWidget {
|
|
|
|
const DataTransferPage({super.key});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(context) {
|
|
|
|
final controller = Provider.of<DataTransferIsolateController>(context);
|
|
|
|
|
|
|
|
return SafeArea(
|
|
|
|
child: Column(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: [
|
|
|
|
Container(
|
|
|
|
padding: const EdgeInsets.all(8),
|
|
|
|
child: Text(
|
|
|
|
'Number Generator Progress',
|
|
|
|
style: Theme.of(context).textTheme.titleLarge,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
LinearProgressIndicator(
|
|
|
|
value: controller.progressPercent,
|
|
|
|
backgroundColor: Colors.grey[200],
|
|
|
|
),
|
|
|
|
const Expanded(
|
|
|
|
child: RunningList(),
|
|
|
|
),
|
|
|
|
Column(
|
|
|
|
children: [
|
|
|
|
ElevatedButton(
|
|
|
|
style: ElevatedButton.styleFrom(
|
|
|
|
foregroundColor: (controller.runningTest == 1)
|
|
|
|
? Colors.blueAccent
|
|
|
|
: Colors.grey[300]),
|
|
|
|
onPressed: () => controller.generateRandomNumbers(false),
|
|
|
|
child: const Text('Transfer Data to 2nd Isolate'),
|
|
|
|
),
|
|
|
|
ElevatedButton(
|
|
|
|
style: ElevatedButton.styleFrom(
|
|
|
|
foregroundColor: (controller.runningTest == 2)
|
|
|
|
? Colors.blueAccent
|
|
|
|
: Colors.grey[300]),
|
|
|
|
onPressed: () => controller.generateRandomNumbers(true),
|
|
|
|
child: const Text('Transfer Data with TransferableTypedData'),
|
|
|
|
),
|
|
|
|
ElevatedButton(
|
|
|
|
style: ElevatedButton.styleFrom(
|
|
|
|
foregroundColor: (controller.runningTest == 3)
|
|
|
|
? Colors.blueAccent
|
|
|
|
: Colors.grey[300]),
|
|
|
|
onPressed: controller.generateOnSecondaryIsolate,
|
|
|
|
child: const Text('Generate on 2nd Isolate'),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class DataTransferIsolateController extends ChangeNotifier {
|
|
|
|
Isolate? _isolate;
|
|
|
|
late ReceivePort _incomingReceivePort;
|
|
|
|
late SendPort _outgoingSendPort;
|
|
|
|
|
|
|
|
final currentProgress = <String>[];
|
|
|
|
int runningTest = 0;
|
|
|
|
Stopwatch _timer = Stopwatch();
|
|
|
|
double progressPercent = 0;
|
|
|
|
|
|
|
|
bool get running => runningTest != 0;
|
|
|
|
|
|
|
|
DataTransferIsolateController() {
|
|
|
|
createIsolate();
|
|
|
|
listen();
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> createIsolate() async {
|
|
|
|
_incomingReceivePort = ReceivePort();
|
|
|
|
_isolate = await Isolate.spawn(
|
|
|
|
_secondIsolateEntryPoint, _incomingReceivePort.sendPort);
|
|
|
|
}
|
|
|
|
|
|
|
|
void listen() {
|
|
|
|
_incomingReceivePort.listen((dynamic message) {
|
|
|
|
if (message is SendPort) {
|
|
|
|
_outgoingSendPort = message;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (message is int) {
|
|
|
|
currentProgress.insert(
|
|
|
|
0, '$message% - ${_timer.elapsedMilliseconds / 1000} seconds');
|
|
|
|
progressPercent = message / 100;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (message is String && message == 'done') {
|
|
|
|
runningTest = 0;
|
|
|
|
_timer.stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
notifyListeners();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void generateOnSecondaryIsolate() {
|
|
|
|
if (running) return;
|
|
|
|
runningTest = 3;
|
|
|
|
currentProgress.clear();
|
|
|
|
|
|
|
|
_timer = Stopwatch();
|
|
|
|
_timer.start();
|
|
|
|
_outgoingSendPort.send('start');
|
|
|
|
|
|
|
|
notifyListeners();
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> generateRandomNumbers(bool transferableTyped) async {
|
|
|
|
if (running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (transferableTyped) {
|
|
|
|
runningTest = 2;
|
|
|
|
} else {
|
|
|
|
runningTest = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
var random = Random();
|
|
|
|
|
|
|
|
currentProgress.clear();
|
|
|
|
|
|
|
|
_timer.reset();
|
|
|
|
_timer.start();
|
|
|
|
|
|
|
|
var randNums = <int>[];
|
|
|
|
for (var i = 0; i < 100; i++) {
|
|
|
|
randNums.clear();
|
|
|
|
|
|
|
|
for (var j = 0; j < 1000000; j++) {
|
|
|
|
randNums.add(random.nextInt(100));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (transferableTyped) {
|
|
|
|
final transferable =
|
|
|
|
TransferableTypedData.fromList([Int32List.fromList(randNums)]);
|
|
|
|
await sendNumbers(transferable);
|
|
|
|
} else {
|
|
|
|
await sendNumbers(randNums);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> sendNumbers(dynamic numList) async {
|
|
|
|
return Future<void>(() {
|
|
|
|
_outgoingSendPort.send(numList);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
|
|
|
super.dispose();
|
|
|
|
_isolate?.kill(priority: Isolate.immediate);
|
|
|
|
_isolate = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class RunningList extends StatelessWidget {
|
|
|
|
const RunningList({super.key});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
final progress =
|
|
|
|
Provider.of<DataTransferIsolateController>(context).currentProgress;
|
|
|
|
|
|
|
|
return DecoratedBox(
|
|
|
|
decoration: BoxDecoration(
|
|
|
|
color: Colors.grey[200],
|
|
|
|
),
|
|
|
|
child: ListView.builder(
|
|
|
|
itemCount: progress.length,
|
|
|
|
itemBuilder: (context, index) {
|
|
|
|
return Column(
|
|
|
|
children: [
|
|
|
|
Card(
|
|
|
|
color: Colors.lightGreenAccent,
|
|
|
|
child: ListTile(
|
|
|
|
title: Text(progress[index]),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
const Divider(
|
|
|
|
color: Colors.blue,
|
|
|
|
height: 3,
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _secondIsolateEntryPoint(SendPort sendPort) async {
|
|
|
|
var receivePort = ReceivePort();
|
|
|
|
sendPort.send(receivePort.sendPort);
|
|
|
|
var length = 1;
|
|
|
|
|
|
|
|
receivePort.listen(
|
|
|
|
(dynamic message) async {
|
|
|
|
if (message is String && message == 'start') {
|
|
|
|
await generateAndSum(sendPort, createNums(), length);
|
|
|
|
|
|
|
|
sendPort.send('done');
|
|
|
|
} else if (message is TransferableTypedData) {
|
|
|
|
await generateAndSum(
|
|
|
|
sendPort, message.materialize().asInt32List(), length);
|
|
|
|
length++;
|
|
|
|
} else if (message is List<int>) {
|
|
|
|
await generateAndSum(sendPort, message, length);
|
|
|
|
length++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (length == 101) {
|
|
|
|
sendPort.send('done');
|
|
|
|
length = 1;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Iterable<int> createNums() sync* {
|
|
|
|
var random = Random();
|
|
|
|
for (var i = 0; i < 100000000; i++) {
|
|
|
|
yield random.nextInt(100);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<int> generateAndSum(
|
|
|
|
SendPort callerSP,
|
|
|
|
Iterable<int> iter,
|
|
|
|
int length,
|
|
|
|
) async {
|
|
|
|
var sum = 0;
|
|
|
|
var count = 1;
|
|
|
|
|
|
|
|
for (var x in iter) {
|
|
|
|
sum += x;
|
|
|
|
if (count % 1000000 == 0) {
|
|
|
|
callerSP.send((count ~/ 1000000) * length);
|
|
|
|
}
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return sum;
|
|
|
|
}
|