青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

羅朝輝(飄飄白云)

關注嵌入式操作系統,移動平臺,圖形開發。-->加微博 ^_^

  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理 ::
  85 隨筆 :: 0 文章 :: 169 評論 :: 0 Trackbacks
深入淺出 Cocoa 之 Plugin
CC許可,轉載請注明出處

在前文 深入淺出 Cocoa 之 Framework 中講解了 Framework,接下來講解 plugin。如果你對 Framework 還不太熟悉的話,請閱讀那篇文中,在本例中使用到了 framework,并在本文中沒有詳細講述其創建和使用過程。

本文代碼下載:點擊這里

為什么要引入插件?
我們知道編譯程序時,會連接相關 framework,通常我們所連接的框架是 Foundation 和 Application 框架。當程序啟動運行時,每個被連接到的 framework 都會被加載到該程序的 objc 運行時環境中。如果我們想向正在運行的程序加載新的 framework,那該怎么辦呢?答案之一就是使用 plugin 機制。cocoa 的 plugin 機制通常由 NSBundle 類來實現,而實現動態加載的功能由函數 objc_addClass 來完成。一般我們無需與 objc_addClass 這個函數打交道,我們使用 NSBundle 來完成絕大部分與 plugin 相關的工作。

plugin 機制能夠讓我們開發出高度模塊化,可定制以及可擴展的應用程序,并能夠讓第三方為該應用程序添加新特性。想必很多人都熟悉 Eclipse,Eclipse 的 plugin 機制就非常方便與強大。

NSBundle 簡介
束(bundle)是文件系統中的一個目錄結構,它將程序會使用到的資源打包在一起。這些資源可包括編譯好的代碼,nib文件,配置文件,圖像,聲音,本地化資源等等。束是 Mac OS X 的一個核心特性,應用程序,Framework,插件都是一個束,只是擴展名各異,如應用程序的擴展名為 .app;Framework 的擴展名是 .framework;插件的擴展名默認為 .bundle。

一個 plugin 就是一個 bundle(束),xcode 默認以 .bundle 為擴展名。通常我們使用我們自己定義的擴展名,以便與系統或其他人編寫的 plugin 區分開來。我們通過 NSBundle 來載入 bundle,并把其中經過編譯的類注冊到 objc 運行時中,然后我們就能在程序中使用這些類了;我們也可以使用 bundle 中的所有資源。

plugin 構架
我們可以通過多種途徑來實現一個 plugin:
1,定義一個 objc protocol,讓 plugin 遵守該 protocol;
2,定義一個基類,讓 plugin 繼承該基類;
3,定義一個 C 回調函數接口,讓 plugin 實現改回調函數;
4,使用 CFPlugIn 來創建 plugin 接口;
在今天的例子中,使用的是第二種情況,這種情況稍稍復雜一些,我們需創建一個 framework 供宿主程序(使用插件的程序)和 plugin 使用,該 framework 的主要職責是提供基類接口。

plugin 的存放目錄
通常 plugin 總是存放在以下三個位置:
1,應用程序名.app/Contents/Plug-ins   這是程序的開發者存放隨產品發布的插件的地方。 
2,~/Library/Application Support/應用程序名/Plug-ins  用戶存放個人插件的地方。
3,/Library/Application Support/應用程序名/Plug-ins   系統中供全部用戶使用的插件。
 
在今天的例子中,使用的是第一種情況,即將插件存放在應用程序包中。

創建宿主程序
我們來創建一個名為 PluginDemo 的 cocoa application,該程序含有一個顯示已安裝 plugin 的 popup button 以及一個執行選中 plugin 的 button。


創建 framework
1,創建名為 PluginFramework 的 framework,向其中添加 plugin 基類:AbstractPlugin。如果你忘記怎樣創建和使用 framework,請參看前文:深入淺出 Cocoa 之 Framework


AbstractPlugin 類僅僅提供兩個接口:name 用來標識 plugin,run 用來供宿主程序運行插件。
- (NSString *)name;
- (IBAction)run:(id)sender;

2,在 PluginDemo 中連接和使用該 framework 來運行插件。如果你忘記怎樣連接和使用 framework,請參看前文:深入淺出 Cocoa 之 Framework。我們在按鈕響應函數中,運行選中的插件。
- (IBAction)runPlugin:(id)sender
{
    AbstractPlugin 
*plugin = [[pluginsController selectedObjects] lastObject];
    
if (!plugin)
        
return;

    [plugin run:sender];
}

創建 plugin
1,創建 plugin;


2,連接 PluginFramework;如果你忘記怎樣連接和使用 framework,請參看前文:深入淺出 Cocoa 之 Framework。

3,創建 UI 界面;


4,創建繼承自基類的 plugin 子類:PluginOne;

PluginOne 類繼承自 AbstractPlugin,它僅僅是顯示和隱藏一個 window,其實現如下:
#import "PluginOne.h"

@implementation PluginOne

@synthesize mainWindow;

- (id)init
{
    self 
= [super init];
    
if (self) {
        
// Initialization code here.
        
        [NSBundle loadNibNamed:
@"PluginOneMainWindow" owner:self];
    }
    
    
return self;
}

- (void)dealloc
{
    mainWindow 
= nil;

    [super dealloc];
}

- (NSString *)name;
{
    
return @"Plugin One";
}

- (IBAction)run:(id)sender;
{
    [mainWindow center];
    [mainWindow makeKeyAndOrderFront:sender];
}

- (IBAction)closeWindow:(id)sender;
{
    [mainWindow orderOut:sender];
}

@end

5,plugin 設置
下面我們來對 plugin 進行設置,我們可以設置其 Principal class,Wrapper Extension(擴展名)。


使用 plugin
1,宿主程序設置
前面說了,在這個例子中,我們打算將插件隨宿主程序一起發布,所以其存放位置就在宿主應用程序包中。因此我們需要在宿主程序種添加一個 Add Copy Files 的 build phase,如下所示:


2,載入plugin
在正式的應用中,我們應該在前面提到的三個目錄下去查找所有 plugin,因為這三個目錄都是 Cocoa 所推薦的 plugin 目錄。在這個例子中,演示的是隨宿主應用程序一起發布的程序,所以我只掃描了應用程序包中的目錄。
- (NSArray *)loadPlugins
{
    NSBundle 
*main = [NSBundle mainBundle];
    NSArray 
*allPlugins = [main pathsForResourcesOfType:@"bundle" inDirectory:@"../PlugIns"];
    
    NSMutableArray 
*availablePlugins = [[[NSMutableArray alloc] init] autorelease];
    
    id plugin 
= nil;
    NSBundle 
*pluginBundle = nil;
    
    
for (NSString *path in allPlugins) {
        pluginBundle 
= [NSBundle bundleWithPath:path];
        [pluginBundle load];
        
        Class principalClass 
= [pluginBundle principalClass];
        
if (![principalClass isSubclassOfClass:[AbstractPlugin class]]) {
            
continue;
        }
        
        plugin 
= [[principalClass alloc] init];
        
        
if ([plugin respondsToSelector:@selector(run:)])
        {
            [availablePlugins addObject:plugin];
            NSLog(
@" >> loading plugin %@ from %@", [plugin name], path);
        }
        
        [plugin release];
        plugin 
= nil;
        pluginBundle 
= nil;
    }

    
return availablePlugins;
}

該函數在 init 中被調用:
- (id)init
{
    self 
= [super init];
    
if (self) {
        plugins 
= [[self loadPlugins] retain];
        
//plugins = [[self loadAllPlugins] retain];
    }
    
    
return self;
}

下面提供一個函數掃描前面提到的三個目錄,你可以用這個函數提到上面代碼中對 loadPlugins 的調用:
- (NSArray *)loadAllPlugins
{
    NSString 
*appName       = @"PluginOne/Plugins";
    NSString 
*appSupport    = @"Library/Application Support";
    appSupport              
= [appSupport stringByAppendingPathComponent:appName];

    NSString 
*appPath       = [[NSBundle mainBundle] builtInPlugInsPath];
    NSString 
*userPath      = [NSHomeDirectory() stringByAppendingPathComponent:appSupport];
    NSString 
*sysPath       = [@"/" stringByAppendingPathComponent:appSupport];
    
    NSArray
* paths = [NSArray arrayWithObjects:appPath, userPath, sysPath, nil];
    
    NSMutableArray 
* availablePlugins = [[[NSMutableArray alloc] init] autorelease];
    
for (NSString * path in paths)
    {
        NSLog(
@" >> Search in directory: %@", path);

        NSArray 
*contents = [[NSFileManager defaultManager] contentsOfDirectoryAtPath:path error:NULL];
        
for (NSString *fileName in contents)
        {
            
if ( [[fileName pathExtension] isEqualToString:@"plugin"|| [[fileName pathExtension] isEqualToString:@"bundle"])
            {
                NSString 
*fullPath = [path stringByAppendingPathComponent:fileName];
                NSBundle 
*pluginBundle = [NSBundle bundleWithPath:fullPath];
                
if (pluginBundle && [pluginBundle load])
                {
                    Class principalClass 
= [pluginBundle principalClass];
                    
if (![principalClass isSubclassOfClass:[AbstractPlugin class]]) {
                        
continue;
                    }
                    
                    id plugin 
= [[principalClass alloc] init];
                    
                    
if ([plugin respondsToSelector:@selector(run:)])
                    {
                        [availablePlugins addObject:plugin];
                        NSLog(
@" >> loading plugin %@ from %@", [plugin name], path);
                    }
                    
                    [plugin release];
                    plugin 
= nil;
                }
            }
        }
    }
    
    
return availablePlugins;
}

運行結果
顯示 plugin 列表的 popupbutton 的內容被綁定到該 plugins 數組,所以程序啟動之后,就能顯示 plugin 的列表。運行結果如下:


點擊運行之后,就能顯示出插件主界面:


Reference
Code Loading Programming Topics provides information about writing plug-ins using the Objective-C language.
Bundle Programming Guide provides an overview to bundles, including their purpose, types, structure, and the API for accessing bundle resources.

posted on 2011-10-28 14:44 羅朝輝 閱讀(2567) 評論(0)  編輯 收藏 引用 所屬分類: Cocoa 開發
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            一本色道久久88综合亚洲精品ⅰ| 亚洲美女91| 久久国产欧美日韩精品| 欧美在线观看视频一区二区三区 | 久久综合久久美利坚合众国| 伊人狠狠色丁香综合尤物| 另类综合日韩欧美亚洲| 欧美gay视频| 亚洲一区国产视频| 欧美亚洲在线| 亚洲国产婷婷香蕉久久久久久99| 亚洲欧洲一区二区在线播放| 欧美视频一区二区| 久久人91精品久久久久久不卡| 久久婷婷激情| 亚洲一区影院| 久久一区二区三区av| 在线视频亚洲欧美| 久久精品视频在线看| 夜夜精品视频一区二区| 午夜日韩av| 亚洲美女精品成人在线视频| 亚洲欧美自拍偷拍| 亚洲精品乱码| 欧美亚洲免费电影| 夜夜爽www精品| 久久精品国产一区二区三区| 亚洲天堂av在线免费观看| 久久久精彩视频| 亚洲欧美一区在线| 麻豆乱码国产一区二区三区| 性欧美激情精品| 欧美日韩不卡| 亚洲丶国产丶欧美一区二区三区 | 久久综合久久综合九色| 亚洲欧美日韩在线一区| 欧美va天堂va视频va在线| 小辣椒精品导航| 欧美日韩免费精品| 欧美激情aaaa| 国内精品久久久久伊人av| 一本一道久久综合狠狠老精东影业 | 日韩午夜免费| 亚洲精品中文字| 久久综合久久综合九色| 久久久水蜜桃| 国产伦精品一区二区三区免费迷 | av成人毛片| 在线看国产日韩| 欧美中文字幕不卡| 欧美一区三区三区高中清蜜桃| 欧美日本亚洲| 亚洲黄色一区| 一区二区冒白浆视频| 欧美激情一区三区| 亚洲国产精品视频| 亚洲精品欧美日韩专区| 男女精品网站| 亚洲国产日韩在线一区模特| 亚洲激情视频网| 免费观看在线综合| 亚洲黄色在线看| 日韩视频第一页| 欧美片第1页综合| 一区二区欧美在线| 午夜精品区一区二区三| 国产精品久久久久久久久久免费看 | 亚洲欧美日韩另类| 欧美在线视频一区| 国模大胆一区二区三区| 久久国产高清| 欧美~级网站不卡| 亚洲人体1000| 欧美欧美天天天天操| 亚洲免费电影在线观看| 亚洲综合成人婷婷小说| 国产欧美一区二区在线观看| 羞羞答答国产精品www一本| 久久精品国产综合| 亚洲激情一区| 国产精品免费网站在线观看| 篠田优中文在线播放第一区| 欧美va天堂va视频va在线| 亚洲免费成人| 国产伦精品一区二区三区免费| 久久精品国产亚洲aⅴ| 欧美国产第二页| 99国产精品久久久久老师| 欧美极品影院| 亚洲自拍偷拍色片视频| 欧美99久久| 亚洲一区二区三区国产| 国产性天天综合网| 欧美1级日本1级| 亚洲日本一区二区三区| 午夜精品婷婷| 亚洲精品久久| 国产亚洲成av人片在线观看桃 | 亚洲制服欧美中文字幕中文字幕| 久久久久天天天天| 一区二区三区精品| 极品少妇一区二区| 国产精品h在线观看| 久久综合99re88久久爱| 9i看片成人免费高清| 欧美成人精品不卡视频在线观看 | 国产一区在线播放| 欧美国产日本高清在线| 欧美亚洲网站| 最新成人av网站| 久热国产精品视频| 久久精品欧洲| 狠狠久久综合婷婷不卡| 欧美日韩免费网站| 久久精品国产亚洲高清剧情介绍| 99视频有精品| 亚洲人成网站影音先锋播放| 久久国产精品毛片| 午夜在线视频一区二区区别| 日韩午夜中文字幕| 亚洲韩国精品一区| 在线观看欧美| 在线色欧美三级视频| 国产精品视频yy9299一区| 欧美极品在线视频| 欧美国产欧美亚洲国产日韩mv天天看完整 | 亚洲一品av免费观看| 亚洲福利一区| 亚洲第一区色| 136国产福利精品导航网址应用| 国产欧美日韩在线视频| 国产精品另类一区| 国产精品夫妻自拍| 国产精品v欧美精品v日本精品动漫| 欧美精品观看| 欧美日韩精品在线| 欧美日韩一区不卡| 欧美三区不卡| 国产精品免费在线| 国产欧美日韩精品在线| 国产精品视频观看| 国产伦精品一区二区三区免费| 国产精品美腿一区在线看| 国产精品入口尤物| 国产亚洲激情| 亚洲成人影音| 99爱精品视频| 亚洲一区免费观看| 欧美在线免费视屏| 久久婷婷久久| 亚洲第一精品久久忘忧草社区| 亚洲电影免费观看高清完整版在线 | 亚洲四色影视在线观看| 亚洲一区二区三区色| 性做久久久久久久免费看| 久久精品国产免费看久久精品| 久久网站热最新地址| 欧美激情久久久久| 欧美婷婷六月丁香综合色| 国产精品大全| 国产午夜久久久久| 亚洲国产合集| 正在播放欧美视频| 亚洲欧洲av一区二区| 久久视频在线视频| 亚洲国产精品一区二区www| 亚洲美女毛片| 久久精品成人欧美大片古装| 欧美福利精品| 国产欧美精品在线播放| 亚洲国产成人久久综合一区| 亚洲网站在线| 免费一级欧美片在线观看| 欧美日韩亚洲一区三区| 国产一区日韩一区| 黄色欧美成人| 99国产精品99久久久久久粉嫩| 亚洲欧美日韩一区| 免费成年人欧美视频| 日韩视频免费大全中文字幕| 香蕉久久夜色精品国产使用方法| 欧美69wwwcom| 国产小视频国产精品| 99re6这里只有精品视频在线观看| 欧美一区二区三区免费看| 亚洲激情综合| 久久久久久久国产| 国产精品久久久久久超碰| 亚洲激情不卡| 久久综合给合| 亚洲午夜未删减在线观看| 男人插女人欧美| 狠狠久久亚洲欧美专区| 亚洲摸下面视频| 亚洲精品一区中文| 欧美69视频| 亚洲国产毛片完整版| 久久免费视频在线| 午夜久久资源| 国产欧美一区二区三区在线老狼|