Tasks遇到的一些坑,关于在子线程中对线程权限认证。

  一般情况下,不应该在执行多线程认证的时候对其子线程进行身份认证,如: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}");
                }
            });

 

posted @ 2019-04-22 22:57  指左转右  阅读(514)  评论(0编辑  收藏  举报