2012-01-17 9 views
6

मेरे पास एनएसटीस्क एक लंबी प्रीमेड शैल स्क्रिप्ट चला रहा है और मैं एनएसपी प्रोग्रेस इंडिकेटर को यह जांचने के लिए चाहता हूं कि कितना किया गया है। मैंने कई चीजों की कोशिश की है लेकिन इसे काम करने के लिए प्रतीत नहीं होता है। मुझे पता है कि अगर प्रगति पट्टी अनिश्चित है तो इसका उपयोग कैसे करें, लेकिन मैं इसे लोड करना चाहता हूं क्योंकि कार्य चल रहा है।एनएसटीस्क की प्रगति की जांच के लिए एक निर्धारित एनएसपी प्रोग्रेस इंडिकेटर का उपयोग कैसे करें? - कोको

यहाँ कैसे मैं स्क्रिप्ट चलाने रहा है:

मुझे लगता है कि उस कार्य की प्रगति की जाँच करता है में, जबकि ऐसा होता है और तदनुसार अद्यतन एक प्रगति बार डाल करने के लिए की जरूरत है।

उत्तर

7

यहां एक यूनिक्स स्क्रिप्ट चलाने वाले एसिंक एनएसटीस्क का एक उदाहरण दिया गया है।

echo "working" >&2

यह सूचना केंद्र द्वारा संसाधित और प्रदर्शित करने के लिए भेज दिया जाता है: यूनिक्स स्क्रिप्ट के भीतर echo आदेशों कि इस तरह मानक त्रुटि के लिए वर्तमान स्थिति वापस भेज रहे हैं।

एक निर्धारित प्रगति पट्टी को अद्यतन करने के लिए बस "25.0" "26.0" जैसे स्टेटस अपडेट भेजें और फ्लोट में कनवर्ट करें और प्रगति पट्टी पर भेजें।

नोट: मुझे यह प्रयोग करने के बाद और इस साइट और अन्य संदर्भों से कई युक्तियों का उपयोग करके यह काम मिल गया। तो मुझे आशा है कि यह आपके लिए सहायक होगा।

NSTask *unixTask; 
NSPipe *unixStandardOutputPipe; 
NSPipe *unixStandardErrorPipe; 
NSPipe *unixStandardInputPipe; 
NSFileHandle *fhOutput; 
NSFileHandle *fhError; 
NSData *standardOutputData; 
NSData *standardErrorData; 

यहां मुख्य कार्यक्रम मॉड्यूल हैं:

यहाँ घोषणाओं हैं

- (IBAction)buttonLaunchProgram:(id)sender { 

    [_unixTaskStdOutput setString:@"" ]; 
    [_unixProgressUpdate setStringValue:@""]; 
    [_unixProgressBar startAnimation:sender]; 

    [self runCommand]; 
} 
- (void)runCommand { 

    //setup system pipes and filehandles to process output data 
    unixStandardOutputPipe = [[NSPipe alloc] init]; 
    unixStandardErrorPipe = [[NSPipe alloc] init]; 

    fhOutput = [unixStandardOutputPipe fileHandleForReading]; 
    fhError = [unixStandardErrorPipe fileHandleForReading]; 

    //setup notification alerts 
    NSNotificationCenter *nc = [NSNotificationCenter defaultCenter]; 

    [nc addObserver:self selector:@selector(notifiedForStdOutput:) name:NSFileHandleReadCompletionNotification object:fhOutput]; 
    [nc addObserver:self selector:@selector(notifiedForStdError:) name:NSFileHandleReadCompletionNotification object:fhError]; 
    [nc addObserver:self selector:@selector(notifiedForComplete:) name:NSTaskDidTerminateNotification object:unixTask]; 

    NSMutableArray *commandLine = [NSMutableArray new]; 
    [commandLine addObject:@"-c"]; 
    [commandLine addObject:@"/usr/bin/kpu -ca"]; //put your script here 

    unixTask = [[NSTask alloc] init]; 
    [unixTask setLaunchPath:@"/bin/bash"]; 
    [unixTask setArguments:commandLine]; 
    [unixTask setStandardOutput:unixStandardOutputPipe]; 
    [unixTask setStandardError:unixStandardErrorPipe]; 
    [unixTask setStandardInput:[NSPipe pipe]]; 
    [unixTask launch]; 

    //note we are calling the file handle not the pipe 
    [fhOutput readInBackgroundAndNotify]; 
    [fhError readInBackgroundAndNotify]; 
} 
-(void) notifiedForStdOutput: (NSNotification *)notified 
{ 

    NSData * data = [[notified userInfo] valueForKey:NSFileHandleNotificationDataItem]; 
    NSLog(@"standard data ready %ld bytes",data.length); 

    if ([data length]){ 

     NSString * outputString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]; 
     NSTextStorage *ts = [_unixTaskStdOutput textStorage]; 
     [ts replaceCharactersInRange:NSMakeRange([ts length], 0) 
          withString:outputString]; 
    } 

    if (unixTask != nil) { 

     [fhOutput readInBackgroundAndNotify]; 
    } 

} 
-(void) notifiedForStdError: (NSNotification *)notified 
{ 

    NSData * data = [[notified userInfo] valueForKey:NSFileHandleNotificationDataItem]; 
    NSLog(@"standard error ready %ld bytes",data.length); 

    if ([data length]) { 

     NSString * outputString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]; 
     [_unixProgressUpdate setStringValue:outputString]; 
    } 

    if (unixTask != nil) { 

     [fhError readInBackgroundAndNotify]; 
    } 

} 
-(void) notifiedForComplete:(NSNotification *)anotification { 

    NSLog(@"task completed or was stopped with exit code %d",[unixTask terminationStatus]); 
    unixTask = nil; 

    [_unixProgressBar stopAnimation:self]; 
    [_unixProgressBar viewDidHide]; 

    if ([unixTask terminationStatus] == 0) { 
     [_unixProgressUpdate setStringValue:@"Success"]; 
    } 
    else { 
     [_unixProgressUpdate setStringValue:@"Terminated with non-zero exit code"]; 
    } 
} 
@end 
+0

धन्यवाद यह काम किया। मुझे यह पसंद है कि आप घोषणा कैसे करते हैं, कुछ लोग नहीं करते हैं और यह वास्तव में मुझे खराब करता है क्योंकि मुझे नहीं पता कि घोषणाओं के रूप में क्या रखा जाए। IBAction की शुरुआत में – drewsdunne

+0

फिर से धन्यवाद, उन चीजों के लिए घोषणाएं क्या हैं जिनके तार – drewsdunne

+0

सेट किए जा रहे हैं और एनएसएमयूटेबलएरे के साथ हिस्सा आवश्यक है? क्या मैं सिर्फ एक .sh फ़ाइल को इंगित नहीं कर सकता/उपयोग/usr/bin/sh – drewsdunne

1

आपको यह बताने के लिए कि आप कितनी प्रगति की है, आपको किसी कार्य की प्रगति को वापस या बाधित करने का कोई तरीका है। यदि आप एक शेल स्क्रिप्ट के बारे में बात कर रहे हैं तो आप एकाधिक स्क्रिप्ट में 1 स्क्रिप्ट को तोड़ सकते हैं और स्क्रिप्ट के किसी अनुभाग के पूरा होने पर प्रगति संकेतक अद्यतन कर सकते हैं। अन्य ऐप्स ने इस तरह की चीजें की हैं, आईआईआरसी स्पार्कल ने अपने डिकंप्रेशन कोड में कुछ कस्टम लॉजिक किया है ताकि भाग में असम्पीडित हो सके ताकि यह प्रगति संकेतक को अपडेट कर सके। यदि आप एक ही प्रभाव प्राप्त करना चाहते हैं तो आपको कुछ ऐसा करना होगा।

+0

ठीक है, मैं इस धन्यवाद – drewsdunne

+0

में यह does not काम पर ध्यान देंगे क्योंकि आप में एक से अधिक NSTasks नहीं चला सकते हैं एक क्रिया – drewsdunne

+0

आप ऐसा क्यों कहेंगे? आप एक एनएसटीस्क लॉन्च कर सकते हैं, इसके परिणाम प्राप्त कर सकते हैं, एक अलग कार्य लॉन्च कर सकते हैं और इसी तरह, मैंने ऐप लिखे हैं जो विभिन्न चीजों के लिए कई कार्यों को लॉन्च करते हैं। केवल अंतर यह है कि मैं कक्षा का उपयोग करता हूं जो एनएसटीस्क को समाहित करता है, लेकिन इसके अलावा आप एक ही https://github.com/Machx/Zangetsu/blob/master/Source/CWTask.m –

0

पीएस (प्रोसेस आईडी) कमांड के साथ पीएसआई (प्रोसेस आईडी) प्राप्त करें, जिसका उपयोग आप पीएस (प्रोसेस स्टेट) कमांड के साथ करते हुए करते हैं, आप अपनी प्रक्रिया का राज्य प्राप्त कर सकते हैं जो आपके कोड में मूल्य प्रगति पट्टी में दिखाने के लिए उपयोग करता है

+0

मैंने पीएस और पीआईडी ​​के बारे में कभी नहीं सुना है क्या आप मुझे एक उदाहरण दिखा सकते हैं? – drewsdunne

+0

ps - किसी भी प्रक्रिया की स्थिति प्राप्त करने के लिए आदेश। यदि आप टर्मिनल में इस आदेश को पिड (प्रक्रिया आईडी, प्रक्रिया के लिए अद्वितीय) के साथ चलाते हैं तो यह प्रक्रिया की स्थिति प्रदर्शित करता है जैसे pid.say उदाहरण के लिए: ps -l 185 –

+0

लेकिन अगर मैं टर्मिनल – drewsdunne

0

आपकी स्क्रिप्ट (nstask) आपके obj c नियंत्रक के साथ संवाद करने का एक आसान तरीका संचार त्रुटि के रूप में मानक त्रुटि का उपयोग कर है। यह नहीं कह रहा कि यह सही है लेकिन काफी अच्छा काम करता है। आपको अपना कार्य एक एसिंक्रोनस नास्टस्क के रूप में सेट करना होगा और मानक इनपुट और मानक त्रुटि से अलग पढ़ने के लिए अधिसूचनाओं का उपयोग करना होगा। अगर आपको इसकी ज़रूरत है तो मैं इसे बाद में पोस्ट कर सकता हूं। शाब्दिक स्थिति अपडेट के लिए एक आउटलेट के लिए एक पाइप और filehandle और यह तार के माध्यम से आपका मानक त्रुटि

echo "now starting" >&2 
for i in $(ls /tmp) 
do 
echo $i 2>&1 
done 
echo "now ending" >&2 

प्रक्रिया या यह प्रगति को प्रदर्शित करता है के लिए एक नाव में बदलने का:

अपनी स्क्रिप्ट इस तरह लपेटें। यानी

echo "25.0" >&2. Can be captured and converted. 

आप वास्तविक मानक त्रुटि कब्जा करने के लिए तो यह जाल और यह विलय मेरी उदाहरण की तरह बाहर std करने की जरूरत है। यह एक आदर्श दृष्टिकोण नहीं है लेकिन मैं इसे अक्सर उपयोग करता हूं और यह अच्छी तरह से काम करता है।

मेरे आईपैड पर और कोड नहीं है। अगर आपको एक बेहतर उदाहरण की आवश्यकता है तो मुझे बताएं।

+0

में चलाता हूं तो यह मुझे अपने आवेदन में वापस कैसे कॉल करेगा, ऐसा लगता है कि यह काम कर सकता है लेकिन मुझे नहीं लगता कि मुझे यह कैसे करना है, इसलिए बेहतर उदाहरण की सराहना की जाएगी – drewsdunne

+0

ठीक है, यहां है संदर्भ। Async मोड में NSTask चलाएं। मानक आउटपुट और मानक त्रुटि विभाजित करें। इन चैनलों के डेटा का जवाब देने के लिए अधिसूचना केंद्र का प्रयोग करें। एक चल रहे पाठ स्थिति बनाने के लिए मानक त्रुटि का उपयोग करें। असल में, आपके प्रोग्राम में चलते समय, इस तरह की इको कमांड के साथ मानक त्रुटि के लिए स्टेटस अपडेट भेजें: गूंजें "स्थिति रिपोर्ट"> और 2। यह मानक त्रुटि पाइप पर दिखाई देगा और आपका प्रोग्राम स्क्रीन पर इसे प्रदर्शित कर सकता है। या स्थिति टेक्स्ट की बजाय, 25.0 जैसी संख्याएं भेजें और उन्हें फ़्लोट में परिवर्तित करें और फिर अपनी निर्धारित प्रगति पट्टी को अपडेट करें। – CocoaEv

संबंधित मुद्दे