一般情况下,不应该在执行多线程认证的时候对其子线程进行身份认证,如:A线程的子线程B和子线程C。
当使用 Parallel.ForEach方法时,只有自身线程能够拥有相对应的权限,其子线程权限则为NULL,因此在某些特定的情况下将会抛出异常,如:
Parallel.ForEach(sork, new ParallelOptions() { MaxDegreeOfParallelism = 3 }, data => { if (!Thread.CurrentPrincipal.Identity.IsAuthenticated) { Console.WriteLine($"没有权限 {Thread.CurrentThread.ManagedThreadId}"); } });
这种情况下,第一次运行时可以通过的。但第二次就会引发 未将实例引用到对象 的异常。这主要是因为Parallel本身不阻塞线程,其本线程和子线程一起执行任务。但子线程CurrentPrincipal是为赋值的。
同样的,Task也是如此。
var task = new Task(() => { if (!Thread.CurrentPrincipal.Identity.IsAuthenticated) { Console.WriteLine($"没有权限 {Thread.CurrentThread.ManagedThreadId}"); } });
当然,不建议在复杂的方法里面对线程进行验证,对线程进行权限验证应该在使用多线程之前,当如果需要的话,还是很容易可以解决的。
IIdentity identity = new GenericIdentity("root"); Thread.CurrentPrincipal = new GenericPrincipal(identity, new[] { "root" }); var principal = Thread.CurrentPrincipal; Parallel.ForEach(sork, new ParallelOptions() { MaxDegreeOfParallelism = 3 }, data => { // 取父进程的授权给子线程即可 Thread.CurrentPrincipal = principal; if (!Thread.CurrentPrincipal.Identity.IsAuthenticated) { Console.WriteLine($"没有权限 {Thread.CurrentThread.ManagedThreadId}"); } }); var task = new Task(() => { // 取父进程的授权给子线程即可 Thread.CurrentPrincipal = principal; if (!Thread.CurrentPrincipal.Identity.IsAuthenticated) { Console.WriteLine($"没有权限 {Thread.CurrentThread.ManagedThreadId}"); } });