কাঁটাচামচ এবং এক্সেকের মধ্যে পার্থক্য


199

forkএবং মধ্যে পার্থক্য কি exec?


3
কাঁটাচামচ, এক্সিকিউটিভ
জোনাথন

9
@ জাস্টিন, কারণ আমরা চাই যে প্রোগ্রামিং প্রশ্নগুলির জন্য এসও জায়গা হয়ে উঠুক ।
প্যাক্সিডিয়াবল্লো

4
@ পোলারিস 878: ওহ, এটি এখন! : ডি
জানুস লেনার

তাই forkমূলত ক্লোনিং হয়: হে
সেবাস্টিয়ান Hojas

উত্তর:


364

ইউএনআইএক্সের চেতনা ব্যবহার forkএবং execউদাহরণ দিয়ে দেয় যে এটি নতুন প্রক্রিয়া শুরু করার খুব সহজ উপায় সরবরাহ করে।

forkকল মূলত বর্তমান প্রক্রিয়ার একটি ডুপ্লিকেট অভিন্ন তোলে প্রায় প্রতি উপায়। সবকিছু অনুলিপি করা হয় না (উদাহরণস্বরূপ, কিছু বাস্তবায়নে সংস্থান সীমা) তবে ধারণাটি যতটা সম্ভব একটি অনুলিপি তৈরি করা।

নতুন প্রক্রিয়া (শিশু) একটি পৃথক প্রক্রিয়া আইডি (পিআইডি) পায় এবং পুরানো প্রক্রিয়াটির পিআইডি (পিতামাতাকে) তার প্যারেন্ট পিআইডি (পিপিআইডি) হিসাবে রাখে। যেহেতু দুটি প্রক্রিয়া এখন একই কোড চলছে, তারা জানাতে পারে যে কোনটি তার রিটার্ন কোড দ্বারা fork- সন্তানের 0 হয়, পিতামাতারা সন্তানের পিআইডি পান। এটি অবশ্যই, forkকল ধরে কাজ করে ধরে নিচ্ছে - তা না থাকলে কোনও শিশু তৈরি হয় না এবং পিতামাতার একটি ত্রুটি কোড পায়।

execকল একটি উপায় মূলত একটি নতুন প্রোগ্রাম সঙ্গে সমগ্র বর্তমান প্রক্রিয়া প্রতিস্থাপন হয়। এটি প্রোগ্রামটিকে বর্তমান প্রক্রিয়া জায়গাতে লোড করে এবং এন্ট্রি পয়েন্ট থেকে চালায়।

সুতরাং, forkএবং execপ্রায়শই ব্যবহার করা হয় একটি বর্তমান প্রক্রিয়া শিশু হিসাবে একটি নতুন প্রোগ্রাম চলমান পেতে ক্রমানুসারে। শেলগুলি সাধারণত যখনই আপনি এই জাতীয় প্রোগ্রাম চালানোর চেষ্টা করেন find- শেল কাঁটাচামচ, তারপরে বাচ্চা findপ্রোগ্রামটি মেমোরিতে লোড করে , সমস্ত কমান্ড লাইন আর্গুমেন্ট, মান I / O ইত্যাদি সেটআপ করে।

তবে এগুলি একসাথে ব্যবহার করার দরকার নেই। আইং forkছাড়াই কোনও প্রোগ্রামের পক্ষে এটি পুরোপুরি গ্রহণযোগ্য হয় execযদি উদাহরণস্বরূপ, প্রোগ্রামটিতে পিতামাতার এবং সন্তানের কোড উভয়ই থাকে (আপনি যা করেন তার ক্ষেত্রে আপনার যত্নবান হওয়া দরকার, প্রতিটি প্রয়োগের ক্ষেত্রে বিধিনিষেধ থাকতে পারে)। এটি ডেমনগুলির জন্য যথেষ্ট পরিমাণে ব্যবহৃত হয়েছিল (এবং এখনও রয়েছে) যা কেবলমাত্র কোনও টিসিপি পোর্টে শোনেন এবং forkএকটি নির্দিষ্ট অনুরোধটি প্রক্রিয়া করার জন্য তাদের একটি অনুলিপি যখন পিতামাতা শোনায় ফিরে যায়।

একইভাবে, প্রোগ্রাম জানি যে, তারা সমাপ্ত করছি এবং মাত্র অন্য প্রোগ্রাম প্রয়োজন হবে না চালাতে চান fork, execএবং তারপর waitসন্তানের জন্য। তারা কেবল তাদের প্রক্রিয়া স্পেসে শিশুটিকে সরাসরি লোড করতে পারে।

কিছু ইউনিক্স বাস্তবায়নের একটি অপ্টিমাইজড রয়েছে forkযা তারা অনুলিপি-অন-লিখনকে বলে uses forkপ্রোগ্রামটি সেই জায়গাতে কিছু পরিবর্তন করার চেষ্টা না করা পর্যন্ত প্রক্রিয়া স্পেসের অনুলিপিটি বিলম্ব করার একটি কৌশল । এটি কেবলমাত্র সেইসব প্রোগ্রামগুলির জন্যই দরকারী এবং এটির জন্য forkনয় execযে তাদের একটি সম্পূর্ণ প্রক্রিয়া স্থান অনুলিপি করতে হবে না।

যদি exec হয় নিম্নলিখিত নামক fork(এবং এই কি বেশিরভাগই ঘটে থাকে), যে প্রক্রিয়া স্থান করার জন্য একটি লেখার কারণ এবং এটি তারপর সন্তান প্রক্রিয়ার জন্য অনুলিপি করা হয়েছে।

নোট একটি পুরো পরিবারের নেই execকল ( execl, execle, execveইত্যাদি) কিন্তু execপ্রেক্ষাপটে এখানে তাদের কোন মানে।

নিম্নলিখিত চিত্রটি সেই সাধারণ fork/execক্রিয়াকে চিত্রিত করে যেখানে bashশেলটি lsকমান্ডের সাহায্যে ডিরেক্টরিটি তালিকাবদ্ধ করতে ব্যবহৃত হয় :

+--------+
| pid=7  |
| ppid=4 |
| bash   |
+--------+
    |
    | calls fork
    V
+--------+             +--------+
| pid=7  |    forks    | pid=22 |
| ppid=4 | ----------> | ppid=7 |
| bash   |             | bash   |
+--------+             +--------+
    |                      |
    | waits for pid 22     | calls exec to run ls
    |                      V
    |                  +--------+
    |                  | pid=22 |
    |                  | ppid=7 |
    |                  | ls     |
    V                  +--------+
+--------+                 |
| pid=7  |                 | exits
| ppid=4 | <---------------+
| bash   |
+--------+
    |
    | continues
    V

52

fork()বর্তমান প্রক্রিয়াটিকে দুটি প্রক্রিয়াতে বিভক্ত করে। বা অন্য কথায়, আপনার দুর্দান্ত রৈখিক সহজেই প্রোগ্রামটি ভাবা সহজ হ'ল হঠাত দুটি পৃথক প্রোগ্রামে পরিণত হয় কোডের এক টুকরো:

 int pid = fork();

 if (pid == 0)
 {
     printf("I'm the child");
 }
 else
 {
     printf("I'm the parent, my child is %i", pid);
     // here we can kill the child, but that's not very parently of us
 }

এটি আপনার মনকে ধাক্কা দিতে পারে। এখন আপনার কাছে কোডের একটি টুকরা রয়েছে যা দুটি প্রসেস দ্বারা কার্যকর হ'ল বেশ অভিন্ন অবস্থা। শিশু প্রক্রিয়াটি কেবলমাত্র প্রক্রিয়াটির সমস্ত কোড এবং মেমরির উত্তরাধিকার সূত্রে প্রাপ্ত, যা fork()কলটি বন্ধ রেখেছিল সেখান থেকে শুরু করে । পার্থক্য শুধুমাত্রfork() আপনি পিতামাতা বা সন্তানের কিনা তা আপনাকে জানানোর জন্য রিটার্ন কোড। আপনি যদি পিতামাতা হন তবে ফেরতের মান হ'ল সন্তানের আইডি।

execউপলব্ধি করা কিছুটা সহজ, আপনি কেবল execএক্সিকিউটেবল টার্গেটটি ব্যবহার করে কোনও প্রক্রিয়া চালিত করতে বলুন এবং আপনার দুটি কোড একই কোড চালাচ্ছেন না বা একই রাজ্যের উত্তরাধিকার সূত্রে পাবেন না। @ স্টিভ হকিন্সের মত , বর্তমান প্রক্রিয়ায় লক্ষ্য নির্বাহের জন্য কার্যকর করার execপরে আপনার ব্যবহার করা যেতে পারে fork


6
শর্ত রয়েছে যখন pid < 0এবং fork()কল ব্যর্থ হয়েছিল
জোনাথন ফিংল্যান্ড

3
এটি আমার মনকে মোটেও উড়িয়ে দেয় না :-) দুটি প্রসেস দ্বারা সম্পাদিত কোডের এক টুকরোটি প্রতিবার একটি ভাগ করা লাইব্রেরি বা ডিএলএল ব্যবহার করা হয়।
paxdiablo

31

আমার কাছে মনে হয় মার্ক মার্কচাইন্ডের "অ্যাডভান্সড ইউনিক্স প্রোগ্রামিং" থেকে কিছু ধারণা fork()/ এর বিভিন্ন ভূমিকা বোঝার exec()জন্য বিশেষত উইন্ডোজ CreateProcess()মডেলটিতে ব্যবহৃত কারও জন্য সহায়ক ছিল :

একটি প্রোগ্রাম হ'ল নির্দেশাবলী এবং ডেটা সংগ্রহ যা ডিস্কে নিয়মিত ফাইলে রাখা হয়। (১.১.২ প্রোগ্রাম, প্রক্রিয়া এবং থ্রেড থেকে)

প্রোগ্রাম চালানোর জন্য, কার্নেলকে প্রথমে একটি নতুন প্রক্রিয়া তৈরি করতে বলা হয় , এটি একটি পরিবেশ যেখানে একটি প্রোগ্রাম কার্যকর করে। (এছাড়াও ১.১.২ প্রোগ্রাম, প্রক্রিয়া এবং থ্রেড থেকে)

কোনও প্রক্রিয়া এবং প্রোগ্রামের মধ্যে পার্থক্যটি পুরোপুরি না বুঝে এক্সেক বা কাঁটাচামচ কলগুলি বোঝা অসম্ভব। যদি এই শর্তাদি আপনার কাছে নতুন হয় তবে আপনি ফিরে যেতে পারেন এবং বিভাগ 1.1.2 পর্যালোচনা করতে পারেন। আপনি যদি এখনই এগিয়ে যেতে প্রস্তুত হন, আমরা এক বাক্যে পার্থক্যটি সংক্ষিপ্ত করব: একটি প্রক্রিয়া হল একটি মৃত্যুদন্ড কার্যকর করার পরিবেশ যা নির্দেশনা, ব্যবহারকারীর ডেটা এবং সিস্টেম-ডেটা বিভাগগুলি এবং সেই সাথে রানটাইমের সময়ে অর্জিত প্রচুর অন্যান্য সংস্থানগুলি নিয়ে গঠিত যদিও একটি প্রোগ্রাম হ'ল নির্দেশাবলী এবং ডেটা সম্বলিত একটি ফাইল যা কোনও প্রক্রিয়াটির নির্দেশনা এবং ব্যবহারকারী-ডেটা বিভাগগুলিকে আরম্ভ করতে ব্যবহৃত হয়। (5.3 execসিস্টেম কল থেকে)

একবার আপনি কোনও প্রোগ্রাম এবং একটি প্রক্রিয়ার মধ্যে পার্থক্য বুঝতে পারলে , আচরণ fork()এবং exec()ক্রিয়াকলাপটি এইভাবে সংক্ষেপিত করা যেতে পারে:

  • fork() বর্তমান প্রক্রিয়াটির একটি সদৃশ তৈরি করে
  • exec() বর্তমান প্রক্রিয়াতে প্রোগ্রামটিকে অন্য একটি প্রোগ্রামের সাথে প্রতিস্থাপন করে

(এটি মূলত প্যাক্সিডিয়াবলোর আরও বিস্তারিত উত্তরের ' ডামিদের জন্য' সংস্করণটি মূলত )


29

কাঁটাচামচ কল করার প্রক্রিয়াটির একটি অনুলিপি তৈরি করে। সাধারণত কাঠামো অনুসরণ করে এখানে চিত্র বর্ণনা লিখুন

int cpid = fork( );

if (cpid = = 0) 
{

  //child code

  exit(0);

}

//parent code

wait(cpid);

// end

(শিশু প্রক্রিয়া পাঠ্যের (কোড), ডেটা, স্ট্যাকের জন্য কলিং প্রক্রিয়া সমান) শিশু প্রক্রিয়া যদি ব্লক হয় তবে কোডটি কার্যকর করে।

এক্সইসি বর্তমান প্রক্রিয়াটিকে নতুন প্রক্রিয়ার কোড, ডেটা, স্ট্যাকের সাথে প্রতিস্থাপন করে। সাধারণত কাঠামো অনুসরণ করে এখানে চিত্র বর্ণনা লিখুন

int cpid = fork( );

if (cpid = = 0) 
{   
  //child code

  exec(foo);

  exit(0);    
}

//parent code

wait(cpid);

// end

(এক্সিকিউটিভ কল ইউনিক্স কার্নেল পরে শিশু প্রক্রিয়া পাঠ্য, ডেটা, স্ট্যাক এবং foo প্রক্রিয়া সম্পর্কিত পাঠ্য / ডেটা পূরণ করে) এভাবে শিশু প্রক্রিয়াটি বিভিন্ন কোডের সাথে থাকে (ফু এর কোড parent পিতামাতার মতো নয়)}


1
এটি প্রশ্নের সাথে কিছুটা সম্পর্কিত নয় তবে উপরের এই কোডটি যদি শিশু কোডটি প্রথমে কোডটি শেষ করে শেষ করে তবে রেস শর্ত তৈরি করে না? সেক্ষেত্রে পিতামাতার প্রক্রিয়াটি চিরকাল স্থায়ীভাবে অপেক্ষা করবে বাচ্চা নিজেকে শেষ করার অপেক্ষা করে, তাই না?
stdout

7

তারা নতুন শিশু প্রক্রিয়া তৈরি করতে একসাথে ব্যবহার করছে। প্রথমত, কলিং forkবর্তমান প্রক্রিয়াটির একটি অনুলিপি তৈরি করে (শিশু প্রক্রিয়া)। তারপরে, execনতুন প্রক্রিয়া সহ পিতামাতার প্রক্রিয়াটির অনুলিপি "প্রতিস্থাপন" করার জন্য শিশু প্রক্রিয়া থেকেই ডাকা হয়।

প্রক্রিয়াটি এরকম কিছু হয়:

child = fork();  //Fork returns a PID for the parent process, or 0 for the child, or -1 for Fail

if (child < 0) {
    std::cout << "Failed to fork GUI process...Exiting" << std::endl;
    exit (-1);
} else if (child == 0) {       // This is the Child Process
    // Call one of the "exec" functions to create the child process
    execvp (argv[0], const_cast<char**>(argv));
} else {                       // This is the Parent Process
    //Continue executing parent process
}

2
7th তম লাইনে উল্লেখ করা হয়েছে যে এক্সিকিউট () ফাংশন শিশু প্রক্রিয়া তৈরি করে .. এটি কি সত্যিই কারণ কারণ কাঁটাচামচ () ইতিমধ্যে শিশু প্রক্রিয়া তৈরি করেছে এবং এক্সিকিউটিভ () কলটি
সদ্য

4

কাঁটাচামচ () বর্তমান সন্তানের প্রক্রিয়াটির একটি অনুলিপি তৈরি করে, কাঁটাচামচ () কল করার ঠিক পরে থেকেই নতুন শিশুটিকে কার্যকর করা হয়। কাঁটাচামচ () এর পরে, তারা কাঁটা ফাংশনের () ফাংশনের রিটার্ন মান ব্যতীত অভিন্ন। (আরও তথ্যের জন্য আরটিএফএম।) দুটি ভাগ প্রক্রিয়াটি তখনই অন্যদিকে বিভক্ত হতে পারে, যার মধ্যে একটির সাথে অন্যের সাথে হস্তক্ষেপ করতে অক্ষম হয়, সম্ভবত কোনও ভাগ করা ফাইল হ্যান্ডলগুলি বাদ দিয়ে।

এক্সিকিউট () বর্তমান প্রক্রিয়াটিকে একটি নতুন দিয়ে প্রতিস্থাপন করে। কাঁটাচামচ () এর সাথে এর কোনও যোগসূত্র নেই, কেবলমাত্র একজন এক্সিকিউটিভ () প্রায়শই কাঁটাচামচ অনুসরণ করে () যখন যা চেয়েছিল তা বর্তমানের প্রতিস্থাপনের পরিবর্তে একটি ভিন্ন শিশু প্রক্রিয়া চালু করতে হবে।


3

মধ্যে মূল পার্থক্য fork()এবং exec()যে,

fork()সিস্টেম কল বর্তমানে চলমান প্রোগ্রাম এর ক্লোন তৈরি করে। মূল প্রোগ্রামটি কাঁটাচামচ () ফাংশন কলের পরে পরবর্তী লাইনের কোডের সাথে সম্পাদন অব্যাহত রাখে। ক্লোনটি কোডের পরবর্তী লাইনেও কার্যকর করা শুরু করে। নীচের কোডটি দেখুন যা আমি http://timmurphy.org/2014/04/26/ using-fork-in-cc-a-minimum-working- example/ থেকে পেয়েছি

#include <stdio.h>
#include <unistd.h>
int main(int argc, char **argv)
{
    printf("--beginning of program\n");
    int counter = 0;
    pid_t pid = fork();
    if (pid == 0)
    {
        // child process
        int i = 0;
        for (; i < 5; ++i)
        {
            printf("child process: counter=%d\n", ++counter);
        }
    }
    else if (pid > 0)
    {
        // parent process
        int j = 0;
        for (; j < 5; ++j)
        {
            printf("parent process: counter=%d\n", ++counter);
        }
    }
    else
    {
        // fork failed
        printf("fork() failed!\n");
        return 1;
    }
    printf("--end of program--\n");
    return 0;
}

এই প্রোগ্রামটি একটি পাল্টা ভেরিয়েবল ঘোষণা করে, শূন্যে সেট করে, fork()আইএনএন করার আগে । কাঁটাচামচ কল করার পরে, আমাদের দুটি সমান্তরালে চলমান দুটি প্রক্রিয়া রয়েছে, উভয়ই তাদের নিজস্ব সংস্করণের কাউন্টারকে বাড়িয়ে তুলছে। প্রতিটি প্রক্রিয়া সমাপ্তি এবং প্রস্থান করতে চালানো হবে। প্রক্রিয়াগুলি সমান্তরালভাবে চালিত হওয়ার কারণে, আমাদের জানার কোনও উপায় নেই যা প্রথমে শেষ হবে। এই প্রোগ্রামটি চালানো নীচের চিত্রের মতো একই জাতীয় কিছু মুদ্রণ করবে, যদিও ফলাফলগুলি এক রান থেকে পরের দিকে আলাদা হতে পারে।

--beginning of program
parent process: counter=1
parent process: counter=2
parent process: counter=3
child process: counter=1
parent process: counter=4
child process: counter=2
parent process: counter=5
child process: counter=3
--end of program--
child process: counter=4
child process: counter=5
--end of program--

exec()সিস্টেম কলের পরিবার কোডের অন্য টুকরা সঙ্গে একটি প্রক্রিয়া বর্তমানে নির্বাহ কোড প্রতিস্থাপন করে। প্রক্রিয়াটি তার পিআইডি বজায় রাখে তবে এটি একটি নতুন প্রোগ্রামে পরিণত হয়। উদাহরণস্বরূপ, নিম্নলিখিত কোডটি বিবেচনা করুন:

#include <stdio.h> 
#include <unistd.h> 
main() {
 char program[80],*args[3];
 int i; 
printf("Ready to exec()...\n"); 
strcpy(program,"date"); 
args[0]="date"; 
args[1]="-u"; 
args[2]=NULL; 
i=execvp(program,args); 
printf("i=%d ... did it work?\n",i); 
} 

এই প্রোগ্রামটি execvp()তারিখ প্রোগ্রামের সাথে তার কোডটি প্রতিস্থাপন করতে ফাংশনটি কল করে । কোডটি যদি exec1.c নামক কোনও ফাইলে সংরক্ষণ করা হয়, তবে এটি সম্পাদন করে নিম্নলিখিত আউটপুট উত্পাদন করে:

Ready to exec()... 
Tue Jul 15 20:17:53 UTC 2008 

প্রোগ্রামটি লাইনটি আউটপুট করে exec রেডি টু এক্সিকিউট ()। । । The এবং এক্সিকিউপি () ফাংশন কল করার পরে, এর কোডটি তারিখ প্রোগ্রামের সাথে প্রতিস্থাপন করে। লক্ষ করুন যে লাইন -। । । এটি কাজ করে - প্রদর্শিত হয় না, কারণ সেই সময়ে কোডটি প্রতিস্থাপন করা হয়েছে। পরিবর্তে, আমরা ―date -u.‖ নির্বাহের ফলাফল দেখি


1

এখানে চিত্র বর্ণনা লিখুনfork():

এটি চলমান প্রক্রিয়াটির একটি অনুলিপি তৈরি করে। চলমান প্রক্রিয়াটিকে অভিভাবক প্রক্রিয়া এবং নতুন তৈরি প্রক্রিয়াটিকে শিশু প্রক্রিয়া বলা হয় । দুটির পার্থক্যের উপায় হ'ল প্রত্যাবর্তিত মানটি দেখে:

  1. fork() পিতামাতার মধ্যে শিশু প্রক্রিয়াটির প্রক্রিয়া শনাক্তকারী (পিড) প্রদান করে

  2. fork() সন্তানের 0 প্রদান করে

exec():

এটি একটি প্রক্রিয়ার মধ্যে একটি নতুন প্রক্রিয়া সূচনা করে। এটি বিদ্যমান প্রোগ্রামটি প্রতিস্থাপন করে বর্তমান প্রক্রিয়াতে একটি নতুন প্রোগ্রাম লোড করে।

fork()+ exec():

নতুন প্রোগ্রাম চালু করার সময় প্রথমে fork()একটি নতুন প্রক্রিয়া তৈরি করা এবং তারপরে exec()(যেমন মেমরির মধ্যে লোড করা এবং চালানো) প্রোগ্রামটি বাইনারি চালিত হওয়ার কথা।

int main( void ) 
{
    int pid = fork();
    if ( pid == 0 ) 
    {
        execvp( "find", argv );
    }

    //Put the parent to sleep for 2 sec,let the child finished executing 
    wait( 2 );

    return 0;
}

0

ধারণা fork()এবং exec()ধারণাটি বোঝার প্রধান উদাহরণটি হ'ল শেল , কমান্ড ইন্টারপ্রেটার প্রোগ্রাম যা ব্যবহারকারীরা সাধারণত সিস্টেমে লগইন করার পরে চালায় The শেলটি কমান্ডের নাম হিসাবে কমান্ড লাইনের প্রথম শব্দটিকে ব্যাখ্যা করে ts

অনেক কমান্ড জন্য, শেল কাটাচামচ ও শিশু প্রক্রিয়া execs নাম কমান্ড পরামিতি হিসেবে কমান্ড লাইন অবশিষ্ট শব্দ চিকিত্সা সঙ্গে যুক্ত কমান্ড।

শেল কমান্ড তিন ধরনের পারেন। প্রথমত, একটি আদেশ একটি এক্সিকিউটেবল ফাইল হতে পারে যা উত্স কোড সংকলন দ্বারা উত্পাদিত অবজেক্ট কোড ধারণ করে (উদাহরণস্বরূপ একটি সি প্রোগ্রাম)। দ্বিতীয়ত, একটি কমান্ড একটি এক্সিকিউটেবল ফাইল হতে পারে যাতে শেল কমান্ড লাইনের ক্রম থাকে। অবশেষে, একটি কমান্ড একটি অভ্যন্তরীণ শেল কমান্ড হতে পারে ((এক্সিকিউটেবল ফাইলের পরিবর্তে-> সিডি , এলএস ইত্যাদি)

আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.