我正在尝试使用dart隔离库来提高我的应用程序性能。
请看以下代码:
import 'dart:isolate';
import 'package:dbcrypt/dbcrypt.dart';
main() {
var pwConPort = new ReceivePort();
pwConPort.listen((data) {
print(data);
pwConPort.close();
}, onError: (err) {
print(err);
});
Isolate.spawn(generatePasswordConcurrency, pwConPort.sendPort);
}
void generatePasswordConcurrency(SendPort sendPort) {
sendPort.send(_generateHashPassword('Passsowr1222!'));
}
String _generateHashPassword(String password) {
var regex = new RegExp(r'^.*(?=.{7,})(?=.*\d)(?=.*[a-z])(?=.*[A-Z])(?=.*[^a-zA-Z0-9]).*$');
if (!regex.hasMatch(password)) {
throw new StateError('Errors');
}
return new DBCrypt().hashpw(password, new DBCrypt().gensalt());
}
一切正常但我只能传递一个静态密码,或者更好地说,我不知道,如何动态传递一些东西。在这里你可以看到,密码是硬编码的,但我想传递一个变量,例如。
void generatePasswordConcurrency(SendPort sendPort) {
sendPort.send(_generateHashPassword('Passsowr1222!'));
}
如果方法_generateHashPassword会抛出错误,我该如何处理此错误?我尝试从ReceivePort
中捕获listen方法的错误 pwConPort.listen((data) {
print(data);
pwConPort.close();
}, onError: (err) {
print(err);
});
但仍然有未处理的异常消息。
Observatory listening on http://127.0.0.1:51433
in ShutdownIsolate: Unhandled exception:
Bad state: Errors
#0 _generateHashPassword (file:///D:/Dart/samples/bin/isolate_error.dart:26:9)
#1 generatePasswordConcurrency (file:///D:/Dart/samples/bin/isolate_error.dart:19:40)
#2 _startIsolate.isolateStartHandler (dart:isolate-patch/isolate_patch.dart:221)
#3 _RawReceivePortImpl._handleMessage (dart:isolate-patch/isolate_patch.dart:124)
结论我的问题:
如何将变量传递给隔离的被调用方法?
如何处理隔离错误?
答案 0 :(得分:7)
首先,
隔离不是线程,它们是独立的进程,更像是fork()
而非线程
使用隔离的并发编程: 独立工作者类似于线程但不共享内存,仅通过通信进行通信 消息。
因此,您无法访问与父进程相同的变量。这是飞镖队的一个选择,因为它是一种在js中编译你的飞镖代码时可用的机制。所以它需要在JS
中成为可能如何将变量传递给隔离的被调用方法?
要做到这一点,你需要看ReceivePort()
像单向通信方式,所以要以两种方式传递变量,你需要两个。
关于你的主要过程:
pwConPort.listen((data) {
if (isolateSendPort == null && data is SendPort) {
isolateSendPort = data; // Receive the communication object of the isolate
isolateSendPort.send("Passsowr1222!");
} else {
print("Generated password: ${data}");
pwConPort.close();
}
}, onError: (err) {
print("SendPortError: ${err}");
});
});
在你隔离入口点:
sendPort.send(isolateConPort.sendPort);
isolateConPort.listen((data) {
// code ....
});
注意:请注意您发送的邮件。一个进程和另一个进程之间发送的消息需要遵守一些规则
消息的内容可以是:原始值(null,num,bool, double,String),SendPort的实例,以及列表和地图 元素是这些中的任何一个。列表和地图也是允许的 环状的。
如何处理隔离错误?
隔离得到一个方法来监听孤立发送的侦听错误:addErrorListner 这是一个有用的功能。
但是!这种方法并没有在每个版块中实现,所以你需要在其他版本中执行此操作。
我选择的方式是在入口点函数中发送2个SendPort:
一个用于沟通
错误之一。
所以spawn函数看起来像:
Isolate.spawn(generatePasswordConcurrency, [pwConPort.sendPort, errorPort.sendPort])
和generatePasswordConcurrency
:
void generatePasswordConcurrency(List<SendPort> commList) {
var sendPort = commList[0];
var errorPort = commList[1];
var isolateConPort = new ReceivePort();
sendPort.send(isolateConPort.sendPort);
isolateConPort.listen((data) {
try {
sendPort.send(_generateHashPassword(data));
} catch (e) {
errorPort.send("error: ${e.toString()}");
}
});
}
这里是完整的代码:
import 'dart:isolate';
import 'package:dbcrypt/dbcrypt.dart';
main() {
var pwConPort = new ReceivePort();
var errorPort = new ReceivePort();
SendPort isolateSendPort = null;
Isolate.spawn(generatePasswordConcurrency, [pwConPort.sendPort, errorPort.sendPort])
.then((Isolate pcs) {
errorPort.listen((err) {
print("Error: ${err}");
pwConPort.close();
errorPort.close();
});
print(pcs);
pwConPort.listen((data) {
if (isolateSendPort == null && data is SendPort) {
isolateSendPort = data;
isolateSendPort.send("Passsowr1222!");
} else {
print("Generated password: ${data}");
pwConPort.close();
errorPort.close();
//pcs.kill();
}
}, onError: (err) {
print("SendPortError: ${err}");
});
});
}
void generatePasswordConcurrency(List<SendPort> commList) {
var sendPort = commList[0];
var errorPort = commList[1];
var isolateConPort = new ReceivePort();
sendPort.send(isolateConPort.sendPort);
isolateConPort.listen((data) {
try {
sendPort.send(_generateHashPassword(data));
} catch (e) {
errorPort.send("error: ${e.toString()}");
}
});
}
String _generateHashPassword(String password) {
var regex = new RegExp(r'^.*(?=.{7,})(?=.*\d)(?=.*[a-z])(?=.*[A-Z])(?=.*[^a-zA-Z0-9]).*$');
if (!regex.hasMatch(password)) {
throw new StateError('Errors');
}
return new DBCrypt().hashpw(password, new DBCrypt().gensalt());
}