一个例子理解AccessController.doPrivileged()

本文通过示例解析了Java安全模型中的AccessController.doPrivileged()方法如何工作,包括其在不同保护域间的权限检查流程及应用技巧。
部署运行你感兴趣的模型镜像

示例中需要在eclipse中创建2个project:ServiceCentre和TestService

java.policy

grant codeBase  " file:/D:/Workspaces/ExchangeConnect_V2_Trunk_Maven_workspace/ServiceCentre/bin/* "   {
    permission java.io.FilePermission 
"c:/TestService-1.0.jar""read";
    permission java.lang.RuntimePermission 
"createClassLoader";
}
;

grant codeBase 
" file:/c:/TestService-1.0.jar "   {
    permission java.io.FilePermission 
"C:/text.txt""read";
}
;




Project - ServiceCentre

package  test;

import  java.lang.reflect.Method;
import  java.net.URL;
import  java.net.URLClassLoader;

/**
 * 
@author Donf Yang
 
*/

public   class  ServiceCentreMain  {

    
public void loadService() {
        URL[] urls;
        
try {
            urls 
= new URL[] new URL("file:c:/TestService-1.0.jar") };
            URLClassLoader ll 
= new URLClassLoader(urls);
            
final Class a = ll.loadClass("test.TestService");
            Object o 
= a.newInstance();
            Method m 
= a.getMethod("doService"null);
            m.invoke(o, 
null);

        }
 catch (Exception e) {
            e.printStackTrace();
        }

    }


    
/**
     * 
@param args
     
*/

    
public static void main(String[] args) {
        ServiceCentreMain s 
= new ServiceCentreMain();
        s.loadService();
    }


}


 

Project - TestService
将TestService打包,放到C盘
package  test;

import  java.io.FilePermission;
import  java.security.AccessController;
import  java.security.Permission;

/**
 * 
@author Donf Yang
 * 
 
*/

public   class  TestService  {

    
public void doService() {
        
        doFileOperation();

    }


    
private void doFileOperation() {
        Permission perm 
= new FilePermission("C:/text.txt""read");
        AccessController.checkPermission(perm);
        System.out.println(
"TestService has permission");
    }


}

运行这个例子的时候,会出现权限错误,把doService()修改一下,就可以顺利通过
public   void  doService()  {

        
// doFileOperation();

        AccessController.doPrivileged(
new PrivilegedAction() {
            
public Object run() {
                doFileOperation();
                
return null;
            }

        }
);
    }

在这个例子中AccessControlContext的stack顺序为
2.  file:/D:/Workspaces/ExchangeConnect_V2_Trunk_Maven_workspace/ServiceCentre/bin/*
1 . file:/c:/TestService-1.0.jar

2没有权限,1有权限,使用doPrivileged后,不检查2



看一下java.security.AccessController的JavaDoc:

A caller can be marked as being  " privileged "  (see doPrivileged and below). When making access control decisions, the checkPermission method stops checking  if  it reaches a caller that was marked as  " privileged "  via a doPrivileged call without a context argument (see below  for  information about a context argument). If that caller ' s domain has the specified permission, no further checking is done and checkPermission returns quietly, indicating that the requested access is allowed. If that domain does not have the specified permission, an exception is thrown, as usual. 

其中提到的no further checking is done的意思是指stack中的checking

加入一个TestService2,文件操作在1,stack为(1,2,3为checking顺序)
3 . file:/D:/Workspaces/ExchangeConnect_V2_Trunk_Maven_workspace/ServiceCentre/bin/*
2 . file:/c:/TestService-1.0.jar
1.  file:/c:/TestService2-1.0.jar
 checking顺序为  1->2->3
如果doPrivileged是在2中调用,那么1,2需要具有权限,3不再进行检查
如果doPrivileged是在1中调用,那么1需要具有权限,2,3不再进行检查


总结:
1.  这里容易理解错误的地方是checking顺序,例如一个调用链 MethodA->MethodB->MethodC(这里的3个方法需要在3个不同的ProtectionDomain中),doPrivileged在MethodB中,很容易理解成检查A,B而不检查C,实际上stack中检查顺序为C->B->A,也就是检查C,B而不检查A

2. ServiceCentre不需要太多权限,而Service就需要使用doPrivileged来避免受到ServiceCentre的权限限制(如果service有足够的权限),Equinox中有很多这样的例子(Equinox扮演Service的角色)。

您可能感兴趣的与本文相关的镜像

ACE-Step

ACE-Step

音乐合成
ACE-Step

ACE-Step是由中国团队阶跃星辰(StepFun)与ACE Studio联手打造的开源音乐生成模型。 它拥有3.5B参数量,支持快速高质量生成、强可控性和易于拓展的特点。 最厉害的是,它可以生成多种语言的歌曲,包括但不限于中文、英文、日文等19种语言

评论 2
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值