Kapsam (bilgisayar bilimi) - Scope (computer science)

İçinde bilgisayar Programlama, dürbün bir ad bağlama - adın bir varlıkla ilişkilendirilmesi, örneğin değişken - bir parçası program ad bağlamanın geçerli olduğu yerde, adın varlığa atıfta bulunmak için kullanılabileceği yerdir. Programın diğer bölümlerinde ad, farklı bir varlığa (farklı bir bağa sahip olabilir) veya hiçbir şeye (bağlı olmayabilir) atıfta bulunabilir. Bir ad bağlamanın kapsamı aynı zamanda görünürlük bir varlığın, özellikle daha eski veya daha teknik literatürde - bu, referans adı değil, referans verilen varlığın perspektifindendir.

"Kapsam" terimi, aynı zamanda, herşey Bir programın bir bölümünde veya bir programın belirli bir noktasında geçerli olan ad bağları, daha doğru bir şekilde bağlam veya çevre.[a]

Açıkçası[b] ve pratikte çoğu programlama dili için "bir programın parçası", kaynak kodu (metin alanı) ve olarak bilinir sözcük kapsamı. Ancak bazı dillerde, "bir programın parçası", çalışma süresinin (yürütme sırasındaki süre) bir bölümünü ifade eder ve şu şekilde bilinir: dinamik kapsam. Bu terimlerin her ikisi de bir şekilde yanıltıcıdır - burada tartışıldığı gibi teknik terimleri yanlış kullanırlar tanım - ancak ayrımın kendisi doğru ve kesindir ve bunlar standart ilgili terimlerdir. Sözcüksel kapsam, bu makalenin ana odak noktasıdır ve dinamik kapsam, sözcük kapsamının aksine anlaşılır.

Çoğu durumda, Ad çözümlemesi Sözlük kapsamına dayalı olarak kullanımı ve uygulaması nispeten kolaydır, çünkü kullanımda bir adın hangi varlığa atıfta bulunduğunu belirlemek için kaynak kodunda geriye doğru okunabilir ve uygulamada bir kişi, bir programı. Zorluklar ortaya çıkıyor isim maskeleme, ileriye dönük beyanlar, ve kaldırma önemli ölçüde daha ince olanlar ortaya çıkarken yerel olmayan değişkenler, Özellikle de kapanışlar.

Tanım

Bir adın (sözcüksel) "kapsamının" kesin tanımı (tanımlayıcı ) belirsizdir - "bir varlıkla bir ismin bağlanmasının geçerli olduğu kaynak kod bölümüdür" ve spesifikasyonundaki 1960 tanımından neredeyse hiç değişmemiştir. ALGOL 60. Temsili dil özellikleri aşağıdadır.

ALGOL 60 (1960)[1]
Aşağıdaki miktar türleri ayırt edilir: basit değişkenler, diziler, etiketler, anahtarlar ve prosedürler. Bir miktarın kapsamı, o miktarla ilişkili tanımlayıcının beyanının geçerli olduğu ifadeler ve ifadeler kümesidir.
C (2007)[2]
Bir tanımlayıcı bir nesneyi belirtebilir; bir işlev; bir etiket veya bir yapının, birleşimin veya numaralandırmanın bir üyesi; a typedef isim; bir etiket adı; bir makro adı; veya bir makro parametresi. Aynı tanımlayıcı, programın farklı noktalarında farklı varlıkları belirtebilir. [...] Bir tanımlayıcının atadığı her farklı varlık için tanımlayıcı gözle görülür (yani kullanılabilir) yalnızca program metni olarak adlandırılan bir bölge içinde dürbün.
Git (2013)[3]
Bir bildirim, boş olmayan bir tanımlayıcıyı bir sabite, türe, değişkene, işleve, etikete veya pakete bağlar. [...] Bildirilen bir tanımlayıcının kapsamı, tanımlayıcının belirtilen sabiti, türü, değişkeni, işlevi, etiketi veya paketi ifade ettiği kaynak metnin kapsamıdır.

En yaygın olarak "kapsam", belirli bir adın belirli bir değişken -zaman beyan etkisi vardır - ancak işlevler, türler, sınıflar gibi diğer varlıklar için de geçerli olabilir. etiketler sabitler ve numaralandırmalar.

Sözcüksel kapsam ve dinamik kapsam

Kapsamdaki temel bir ayrım, "bir programın parçası" nın ne anlama geldiğidir. İle dillerde sözcük kapsamı (olarak da adlandırılır statik kapsam), ad çözümlemesi kaynak koddaki konuma bağlıdır ve sözcük bağlamı (olarak da adlandırılır statik bağlam), adlandırılmış değişken veya işlevin nerede tanımlandığı ile tanımlanır. Aksine, şu dillerde dinamik kapsam isim çözünürlüğü şunlara bağlıdır: program durumu tarafından belirlenen isim karşılaşıldığında yürütme bağlamı (olarak da adlandırılır çalışma zamanı bağlamı, arama bağlamı veya dinamik bağlam). Pratikte, sözcüksel kapsamla bir ad, yerel sözcük bağlamı aranarak çözülür, sonra bu, dış sözcük bağlamında arama yaparak başarısız olursa, vb., Dinamik kapsamda ise bir ad yerel yürütme bağlamı aranarak çözümlenir, o zaman eğer öyleyse dış yürütme bağlamında arama yaparak başarısız olur ve bu şekilde çağrı yığınında yukarı doğru ilerler.[4]

Çoğu modern dil değişkenler ve işlevler için sözcük kapsamını kullanır, ancak bazı dillerde dinamik kapsam kullanılır, özellikle Lisp'in bazı lehçeleri, bazı "komut dosyası" dilleri ve bazı şablon dilleri. [c] Perl 5 hem sözcüksel hem de dinamik kapsam sunar. Sözcük kapsamlı dillerde bile, kapsamı kapanışlar başlatılmamış olanlar için kafa karıştırıcı olabilir, çünkü bunlar kapanışın nerede çağrıldığına değil tanımlandığı sözcüksel bağlama bağlıdır.

Sözcüksel çözünürlük şurada belirlenebilir: Derleme zamanı ve olarak da bilinir erken bağlamadinamik çözünürlük genel olarak yalnızca şu şekilde belirlenebilir: Çalışma süresi ve bu nedenle olarak bilinir geç bağlama.

Ilgili kavramlar

İçinde nesne yönelimli programlama, dinamik gönderim bir nesne seçer yöntem çalışma zamanında, gerçek ad bağlamanın derleme zamanında mı yoksa çalışma zamanında mı yapılacağı dile bağlıdır. Fiili dinamik kapsam, makro dilleri, bunlar doğrudan ad çözümlemesi yapmaz, bunun yerine yerinde genişler.

Gibi bazı programlama çerçeveleri AngularJS "kapsam" terimini bu makalede kullanıldığından tamamen farklı bir şey ifade etmek için kullanın. Bu çerçevelerde kapsam, kullandıkları programlama dilinin yalnızca bir nesnesidir (JavaScript Değişkenleri için sözcüksel kapsam kullanan bir dildeki dinamik kapsamı taklit etmek için çerçeve tarafından belirli şekillerde kullanılan AngularJS durumunda). Şunlar AngularJS kapsamları programın herhangi bir bölümünde bağlam içinde olabilir veya olmayabilir (terimin genel anlamını kullanarak), diğer nesneler gibi dilin değişken kapsamının olağan kurallarını izleyerek ve kendi miras ve aşma kurallar. AngularJS bağlamında, bazen "$ kapsam" terimi (dolar işaretli) karışıklığı önlemek için kullanılır, ancak değişken adlarında dolar işaretinin kullanılması genellikle stil kılavuzları tarafından önerilmez.[5]

Kullanım

Kapsam, aşağıdakilerin önemli bir bileşenidir: Ad çözümlemesi,[d] bu sırayla temeldir dil semantiği. Ad çözümlemesi (kapsam dahil) programlama dilleri arasında farklılık gösterir ve bir programlama dili içinde varlık türüne göre değişir; kapsam kuralları denir kapsam kuralları (veya kapsam kuralları). Birlikte ad alanları kapsam kuralları çok önemlidir modüler programlama, bu nedenle programın bir bölümündeki değişiklik ilgisiz bir bölümü bozmaz.

Genel Bakış

Kapsamı tartışırken, üç temel kavram vardır: dürbün, kapsam, ve bağlam. Özellikle "Kapsam" ve "bağlam" sıklıkla karıştırılır: kapsam, bir ad bağlamanın bir özelliğidir, bağlam, bir programın bir kısmının bir özelliğidir, yani kaynak kodun bir bölümüdür (sözcük bağlamı veya statik bağlam) veya bir kısmı Çalışma süresi (yürütme bağlamı, çalışma zamanı bağlamı, arama bağlamı veya dinamik bağlam). Yürütme bağlamı, sözlü bağlamdan (geçerli yürütme noktasında) artı ek çalışma zamanı durumundan oluşur. çağrı yığını.[e] Açıkça söylemek gerekirse, bir program yürütme sırasında çeşitli ad bağlamalarının kapsamlarına girer ve çıkar ve yürütme noktasında ad bağlamaları "bağlam içinde" veya "bağlam içinde değildir", dolayısıyla ad bağlamaları "bağlama girer" veya "bağlam dışında kalır "program yürütme kapsama girerken veya çıkarken.[f] Bununla birlikte, pratikte kullanım çok daha gevşektir.

Kapsam, kaynak kodu düzeyinde bir kavramdır ve ad bağlamalarının bir özelliğidir, özellikle değişken veya işlev adı bağlamaları - kaynak kodundaki adlar Referanslar programdaki varlıklara - ve bir derleyicinin veya bir dilin yorumlayıcısının davranışının bir parçasıdır. Bu nedenle, kapsam sorunları benzerdir işaretçiler programlarda daha genel olarak kullanılan bir referans türü olan. Ad bağlam içindeyken, ancak değişken başlatılmamışken bir değişkenin değerini kullanmak, başvuruyu kaldırmaya (değerine erişme) benzerdir. vahşi işaretçi tanımsız olduğu gibi. Bununla birlikte, değişkenler bağlam dışına çıkıncaya kadar yok edilmediğinden, bir sarkan işaretçi bulunmuyor.

Değişkenler gibi varlıklar için kapsam, bir alt kümedir ömür (Ayrıca şöyle bilinir kapsam ) - bir isim yalnızca var olan (muhtemelen tanımlanmamış bir değere sahip) bir değişkene atıfta bulunabilir, ancak var olan değişkenler mutlaka görünür değildir: bir değişken olabilir ancak erişilemez olabilir (değer depolanır, ancak belirli bir bağlamda atıfta bulunulmaz), veya erişilebilir, ancak verilen ad üzerinden değil, bu durumda bağlam içinde değildir (program "adın kapsamı dışındadır"). Diğer durumlarda "ömür" önemsizdir — bir etiketin (kaynak kodda adlandırılmış konum) yaşam süresi programla aynıdır (statik olarak derlenen diller için), ancak bağlam içinde olabilir veya programın belirli bir noktasında olmayabilir ve aynı şekilde statik değişkenler —A statik genel değişken tüm program bağlamında, statik yerel değişken yalnızca bir işlev veya başka bir yerel bağlam içindeki bağlamdadır, ancak her ikisi de programın tüm çalışmasının ömrüne sahiptir.

Bir adın hangi varlığı ifade ettiğinin belirlenmesi Ad çözümlemesi veya ad bağlama (Özellikle de nesne yönelimli programlama ) ve diller arasında değişir. Bir ad verildiğinde, dil (uygun şekilde, derleyici veya yorumlayıcı) bağlamdaki tüm varlıkları eşleşmeler için kontrol eder; belirsizlik durumunda (aynı ada sahip bir global ve yerel değişken gibi aynı ada sahip iki varlık), bunları ayırt etmek için ad çözümleme kuralları kullanılır. Çoğu zaman, ad çözümleme Python LEGB (Yerel, Kapsayıcı, Global, Yerleşik) kuralı gibi "içten dışa bağlam" kuralına dayanır: adlar örtük olarak en dar ilgili bağlama çözümlenir. Bazı durumlarda ad çözümlemesi açıkça belirtilebilir, örneğin küresel ve yerel olmayan Python'da anahtar sözcükler; diğer durumlarda varsayılan kurallar geçersiz kılınamaz.

İki özdeş ad aynı anda bağlam içinde olduğunda, farklı varlıklara atıfta bulunulduğunda, biri isim maskeleme yüksek öncelikli adın (genellikle en içteki) düşük öncelikli adı "maskelediği" durumda meydana gelir. Değişkenler düzeyinde bu, değişken gölgeleme. Potansiyel nedeniyle mantık hataları bazı diller maskelemeden dolayı maskelemeye izin vermez veya caydırır, derleme zamanında veya çalışma zamanında bir hata veya uyarı verir.

Çeşitli Programlama dilleri farklı türde bildirimler ve adlar için çeşitli farklı kapsam kurallarına sahiptir. Bu tür kapsam kurallarının, dil semantiği ve sonuç olarak, programların davranışı ve doğruluğu hakkında. Gibi dillerde C ++, bağlanmamış bir değişkene erişim iyi tanımlanmış anlamlara sahip değildir ve sonuçta tanımlanmamış davranış, bir sarkan işaretçi; ve kapsamlarının dışında kullanılan bildirimler veya isimler, sözdizimi hataları.

Kapsamlar sıklıkla diğer dil yapılarına bağlanır ve dolaylı olarak belirlenir, ancak birçok dil de özellikle kapsamı kontrol etmek için yapılar sunar.

Kapsam seviyeleri

Kapsam, tek bir ifade kadar küçükten tüm programa kadar değişebilir ve aralarında birçok olası derecelendirme olabilir. En basit kapsam kuralı genel kapsamdır — tüm varlıklar tüm program boyunca görülebilir. En temel modüler kapsam kuralı, programın herhangi bir yerinde global bir kapsam ve bir işlev içinde yerel kapsam olan iki seviyeli kapsamdır. Daha sofistike modüler programlama, adların modül içinde göründüğü (modüle özel) ancak dışında görünmediği ayrı bir modül kapsamına izin verir. Bir işlev içinde, C gibi bazı diller, blok kapsamının kapsamı bir işlevin bir alt kümesiyle sınırlamasına izin verir; diğerleri, özellikle işlevsel diller, ifade kapsamına izin vererek kapsamı tek bir ifadeyle sınırlandırır. Diğer kapsamlar, modül kapsamına benzer şekilde davranan dosya kapsamını (özellikle C'de) ve işlevlerin dışında (özellikle Perl'de) kapsamı engellemeyi içerir.

İnce bir konu, tam olarak bir kapsamın başladığı ve bittiği zamandır. C gibi bazı dillerde, bir ismin kapsamı bildiriminde başlar ve bu nedenle belirli bir blok içinde bildirilen farklı isimler farklı kapsamlara sahip olabilir. Bu, işlevlerin kullanılmadan önce bildirilmesini gerektirir, ancak bunların tanımlanması gerekmez ve ileriye dönük beyan bazı durumlarda, özellikle karşılıklı özyineleme için. JavaScript veya Python gibi diğer dillerde, bir adın kapsamı, nerede tanımlandığına bakılmaksızın, ilgili bloğun başlangıcında (bir işlevin başlangıcı gibi) başlar ve belirli bir blok içindeki tüm adlar aynı kapsama sahiptir; JavaScript'te bu, değişken kaldırma. Bununla birlikte, ad bir değere bağlı olduğunda değişir ve tanımsız değere sahip bağlam içi adların davranışı farklılık gösterir: Python'da tanımsız adların kullanımı bir çalışma zamanı hatası verirken JavaScript'te tanımsız adlar ile bildirilir var (ancak ile beyan edilen isimler değil İzin Vermek ne de sabit), değere bağlı oldukları için işlev boyunca kullanılabilir Tanımsız.

İfade kapsamı

İsmin kapsamı bir ifade olarak bilinen ifade kapsamı. İfade kapsamı birçok dilde mevcuttur, özellikle işlevsel adlı bir özellik sunan diller let-ifadeler bir bildirimin kapsamının tek bir ifade olmasına izin verir. Bu, örneğin bir hesaplama için bir ara değere ihtiyaç duyulduğunda uygundur. Örneğin, Standart ML, Eğer f () İadeler 12, sonra hadi val x = f () içinde x * x son olarak değerlendirilen bir ifadedir 144, adlı geçici bir değişken kullanarak x aramaktan kaçınmak için f () iki defa. Blok kapsamına sahip bazı diller, bir ifadeye gömülecek bir blok için sözdizimi sunarak bu işlevselliğe yaklaşır; örneğin, yukarıda belirtilen Standart ML ifadesi şu şekilde yazılabilir: Perl gibi yapmak { benim $ x = f(); $ x * $ x }veya içinde GNU C gibi ({ int x = f(); x * x; }).

Python'da, üretici ifadelerindeki ve liste anlamalarındaki (Python 3'te) yardımcı değişkenler ifade kapsamına sahiptir.

C'de, değişken isimleri a işlev prototipi bu bağlamda olarak bilinen ifade kapsamı var işlev protokol kapsamı. Prototipteki değişken adlarına atıfta bulunulmadığından (gerçek tanımda farklı olabilirler) - bunlar sadece sahte isimlerdir - bunlar genellikle ihmal edilir, ancak örneğin dokümantasyon oluşturmak için kullanılabilirler.

Kapsamı engelle

Bir ismin kapsamı bir blok olarak bilinen blok kapsamı. Blok kapsamı, blok yapılı programlama dillerinin hepsinde olmasa da birçoğunda mevcuttur. Bu şununla başladı ALGOL 60, "[e] çok beyan ... sadece bu blok için geçerlidir.",[6] ve bugün özellikle Pascal ve C aileler ve gelenekler. Çoğu zaman bu blok bir fonksiyonun içinde yer alır, dolayısıyla kapsamı bir fonksiyonun bir kısmıyla sınırlar, ancak Perl gibi bazı durumlarda blok bir fonksiyonun içinde olmayabilir.

imzasız int sum_of_squares(sabit imzasız int N) {  imzasız int ret = 0;  için (imzasız int n = 1; n <= N; n++) {    sabit imzasız int n_squared = n * n;    ret += n_squared;  }  dönüş ret;}

Blok kapsamının kullanımının temsili bir örneği, burada gösterilen C kodudur, burada iki değişken döngü dahilindedir: döngü değişkeni n, bir kez başlatılan ve döngünün her yinelemesinde artırılan ve yardımcı değişken n_squared, her yinelemede başlatılır. Amaç, işlev kapsamına yalnızca belirli bir blokla ilgili değişkenler eklemekten kaçınmaktır - örneğin, bu, genel döngü değişkeninin ben yanlışlıkla zaten başka bir değere ayarlanmış. Bu örnekte ifade n * n genellikle bir yardımcı değişkene atanmaz ve döngünün gövdesi basitçe yazılır ret + = n * n ancak daha karmaşık örneklerde yardımcı değişkenler kullanışlıdır.

Bloklar öncelikle, if, while ve döngülerde olduğu gibi kontrol akışı için kullanılır ve bu durumlarda blok kapsamı, değişken kapsamının bir fonksiyonun yürütme akışının yapısına bağlı olduğu anlamına gelir. Bununla birlikte, blok kapsamı olan diller tipik olarak, tek amacı değişken kapsamın ince taneli kontrolüne izin vermek olan "çıplak" blokların kullanımına da izin verir. Örneğin, bir yardımcı değişken bir blokta tanımlanabilir, daha sonra kullanılabilir (örneğin, fonksiyon kapsamına sahip bir değişkene eklenebilir) ve blok sona erdiğinde atılabilir veya bir while döngüsü, döngü içinde kullanılan değişkenleri başlatan bir blok içinde yer alabilir. bu yalnızca bir kez başlatılmalıdır.

Aşağıdakiler gibi çeşitli programlama dillerinin incelikleri Algol 68 ve C (bu örnekte gösterilmiştir ve şu tarihten beri standartlaştırılmıştır) C99 ), blok kapsamı değişkenlerinin yalnızca bloğun gövdesi içinde değil, varsa kontrol ifadesi içinde de bildirilebilmesidir. Bu, işlev bildiriminde (işlev gövdesinin bloğu başlamadan önce) ve tüm işlev gövdesinin kapsamında bildirilen işlev parametrelerine benzer. Bu öncelikle döngüler için, while döngülerinden farklı olarak döngü koşulundan ayrı bir başlatma ifadesine sahip olan ve yaygın bir deyimdir.

Gölgeleme için blok kapsamı kullanılabilir. Bu örnekte, bloğun içinde yardımcı değişken de çağrılabilirdi n, parametre adını gölgelendiriyor, ancak bu, hata potansiyeli nedeniyle zayıf stil olarak kabul edilir. Ayrıca, Java ve C # gibi bazı C soyundan gelenler, blok kapsamı için desteğe sahip olmalarına rağmen (bir yerel değişkenin, bir fonksiyonun sonundan önce bağlam dışına çıkması sağlanabilir), bir yerel değişkenin diğerini gizlemesine izin vermez. . Bu tür dillerde, ikinci beyannamenin teşebbüsü n bir sözdizimi hatasına neden olur ve n değişkenlerin yeniden adlandırılması gerekir.

Bir değişkenin değerini ayarlamak için bir blok kullanılıyorsa, blok kapsamı, değişkenin bloğun dışında bildirilmesini gerektirir. Bu, koşullu ifadelerin kullanımını karmaşıklaştırır. tek görev. Örneğin, blok kapsamını kullanmayan Python'da, şu şekilde bir değişken başlatılabilir:

Eğer c:    a = "foo"Başka:    a = ""

nerede a sonra erişilebilir Eğer Beyan.

Blok kapsamı olan Perl'de bu, değişkeni bloktan önce bildirmeyi gerektirir:

benim $ a;Eğer (c) {    $ a = 'foo';} Başka {    $ a = '';}

Çoğu zaman, bunun yerine çoklu atama kullanılarak yeniden yazılır ve değişkeni varsayılan bir değerle başlatır. Python'da (gerekli olmadığı yerde) bu şöyle olacaktır:

a = ""Eğer c:    a = "foo"

Perl'de bu şöyle olur:

benim $ a = '';Eğer (c) {    $ a = 'foo';}

Tek bir değişken atama durumunda, bir alternatif, üçlü operatör bir bloktan kaçınmak için, ancak bu genel olarak çoklu değişken atamaları için mümkün değildir ve karmaşık mantık için okunması zordur.

Bu, C'de, özellikle dizi ataması için daha önemli bir sorundur, çünkü dizi başlatma otomatik olarak belleği tahsis edebilir, halihazırda başlatılmış bir değişkene dizgi ataması bellek ayırmayı, bir dizi kopyasını ve bunların başarılı olup olmadığını kontrol etmeyi gerektirir.

alt increment_counter {    benim $ counter = 0;    dönüş alt    {        dönüş ++$ counter;    }}

Bazı diller, blok kapsamı kavramının bir işlevin dışında değişen kapsamlarda uygulanmasına izin verir. Örneğin, sağdaki Perl kod parçacığında, $ counter blok kapsamına sahip bir değişken adıdır (kullanımından dolayı benim anahtar kelime), while increment_counter genel kapsamı olan bir işlev adıdır. Her çağrı increment_counter değerini artıracak $ counter birer birer ve yeni değeri döndür. Bu bloğun dışındaki kod arayabilir increment_counter, ancak başka türlü elde edemez veya değiştiremez $ counter. Bu deyim kişinin tanımlamasına izin verir kapanışlar Perl'de.

İşlev kapsamı

Bir ismin kapsamı bir fonksiyondur ve şu şekilde bilinir: işlev kapsamı. İşlev kapsamı, bir program oluşturmak için bir yol sunan çoğu programlama dilinde mevcuttur. yerel değişken bir işlevde veya altyordam: fonksiyon döndüğünde kapsamı biten (bağlamın dışına çıkan) bir değişken. Çoğu durumda değişkenin ömrü, işlev çağrısının süresidir; bu bir otomatik değişken, işlev başladığında (veya değişken bildirildiğinde) oluşturulur, işlev döndüğünde yok edilir - değişkenin kapsamı işlevin içindeyken, "içinde" nin anlamı kapsamın sözcüksel veya dinamik olmasına bağlıdır. Bununla birlikte, C gibi bazı diller de statik yerel değişkenler, değişkenin yaşam süresi programın tüm yaşam süresidir, ancak değişken yalnızca işlevin içindeyken bağlam içindedir. Statik yerel değişkenler söz konusu olduğunda, değişken program başlatıldığında oluşturulur ve yalnızca program sona erdiğinde yok edilir. statik genel değişken, ancak otomatik bir yerel değişken gibi, yalnızca bir işlev içindeki bağlam içindedir.

Önemli olarak, sözcüksel kapsamda işlev kapsamına sahip bir değişkenin kapsamı yalnızca sözcük bağlamı İşlevin içinde başka bir işlev çağrıldığında bağlamın dışına çıkar ve işlev geri döndüğünde bağlama geri döner - çağrılan işlevlerin işlevlerin yerel değişkenlerine erişimi yoktur ve yerel değişkenler yalnızca bildirildikleri işlevin gövdesi. Aksine, dinamik kapsamda kapsam, yürütme bağlamı fonksiyonun: yerel değişkenler bağlamda kal başka bir işlev çağrıldığında, yalnızca tanımlayıcı işlev sona erdiğinde bağlamın dışına çıkıyor ve bu nedenle yerel değişkenler, tanımlandıkları işlevin bağlamında yer alıyor ve tüm işlevler. Sözcük kapsamı olan dillerde ve yuvalanmış işlevler Yerel değişkenler, iç içe geçmiş işlevler bağlamındadır, çünkü bunlar aynı sözcük bağlamında bulunur, ancak sözcüksel olarak iç içe olmayan diğer işlevler için değildir. Çevreleyen bir işlevin yerel bir değişkeni, yerel olmayan değişken iç içe geçmiş işlev için. İşlev kapsamı ayrıca şunlar için de geçerlidir: anonim işlevler.

def Meydan(n):    dönüş n * ndef sum_of_squares(n):    Toplam = 0     ben = 0    süre ben <= n:        Toplam += Meydan(ben)        ben += 1    dönüş Toplam

Örneğin, sağdaki Python kod parçacığında iki işlev tanımlanmıştır: Meydan ve sum_of_squares. Meydan bir sayının karesini hesaplar; sum_of_squares bir sayıya kadar tüm karelerin toplamını hesaplar. (Örneğin, kare (4) 42 = 16, ve sum_of_squares (4) 02 + 12 + 22 + 32 + 42 = 30.)

Bu işlevlerin her birinin adında bir değişkeni vardır n işlevin argümanını temsil eder. Bu ikisi n değişkenler, aynı ada sahip olmalarına rağmen, tamamen ayrı ve ilgisizdirler, çünkü bunlar, işlev kapsamına sahip sözcüksel kapsamlara sahip yerel değişkenlerdir: her birinin kapsamı kendi, sözcüksel olarak ayrı bir işlevdir ve bu nedenle örtüşmezler. Bu nedenle, sum_of_squares arayabilir Meydan kendi başına n değiştiriliyor. Benzer şekilde, sum_of_squares isimli değişkenlere sahiptir Toplam ve ben; bu değişkenler, sınırlı kapsamları nedeniyle, adı verilen herhangi bir değişkene müdahale etmeyecektir. Toplam veya ben bu başka herhangi bir işleve ait olabilir. Başka bir deyişle, risk yok isim çarpışması bu isimler ile ilgisiz isimler arasında, aynı olsalar bile.

İsim maskeleme gerçekleşmiyor: sadece bir değişken isimli n kapsamlar çakışmadığından, herhangi bir zamanda bağlam içindedir. Buna karşılık, dinamik kapsamı olan bir dilde yazılmış benzer bir parça vardı, n çağıran işlevde, çağrılan işlevde bağlam içinde kalır — kapsamlar üst üste gelir - ve yeni tarafından maskelenir ("gölgelenir") n çağrılan işlevde.

İşlevler birinci sınıf nesneler ise ve bir işleve yerel olarak oluşturulup daha sonra döndürülebiliyorsa, işlev kapsamı önemli ölçüde daha karmaşıktır. Bu durumda, iç içe geçmiş işlevde yerel olmayan herhangi bir değişken (işlev tanımındaki, çevreleyen bağlamdaki değişkenlere çözümlenen bağlanmamış değişkenler) bir kapatma, çünkü yalnızca işlevin kendisi değil, aynı zamanda bağlamı da (değişkenler) döndürülmeli ve daha sonra potansiyel olarak farklı bir bağlamda çağrılmalıdır. Bu, derleyiciden önemli ölçüde daha fazla destek gerektirir ve program analizini karmaşıklaştırabilir.

Dosya kapsamı

İsmin kapsamı, olarak bilinen bir dosyadır. dosya kapsamı. Dosya kapsamı, büyük ölçüde C (ve C ++) için özeldir, burada bir dosyanın en üst seviyesinde (herhangi bir işlev içinde değil) bildirilen değişkenlerin ve işlevlerin kapsamı tüm dosya için - veya daha doğrusu C için, bildirimden sonuna kadar kaynak dosya veya daha doğrusu çeviri birimi (iç bağlantı). Bu, modüllerin dosyalarla tanımlandığı ve daha modern dillerde açık bir modül kapsamı ile değiştirildiği bir modül kapsamı biçimi olarak görülebilir. İç bağlama değişkenler ve işlevler ekleyen ve kendileri de daha fazla içerme ifadeleri olarak adlandırabilen include deyimlerinin varlığı nedeniyle, bir dosyanın gövdesinde neyin bağlam içinde olduğunu belirlemek zor olabilir.

Yukarıdaki C kodu pasajında, işlev adı sum_of_squares dosya kapsamına sahiptir.

Modül kapsamı

Bir ismin kapsamı, olarak bilinen bir modüldür. modül kapsamı. Modül kapsamı şurada mevcuttur: modüler programlama dilleri modüller (çeşitli dosyaları kapsayabilir), bilgilerin gizlenmesine ve sınırlı bir arayüzün açığa çıkarılmasına izin verdikleri için karmaşık bir programın temel birimleridir. Modül kapsamına, Modula dil ailesi ve Python (Modula'dan etkilenmiştir) temsili çağdaş bir örnektir.

Bazılarında nesne yönelimli programlama C ++ gibi modüller için doğrudan desteğe sahip olmayan diller, bunun yerine sınıfların programın temel birimi olduğu ve bir sınıfın özel yöntemlere sahip olabileceği sınıf hiyerarşisi tarafından benzer bir yapı sağlanır. Bu, bağlamında doğru bir şekilde anlaşılmıştır dinamik gönderim isim çözümleme ve kapsamdan ziyade, genellikle benzer roller oynarlar. Bazı durumlarda, hem modüllere hem de sınıflara sahip olan Python'da olduğu gibi bu iki olanak da mevcuttur ve kod organizasyonu (modül düzeyinde bir işlev veya geleneksel olarak özel bir yöntem olarak) programcının seçimidir.

Global kapsam

Bir ismin kapsamı, bütün bir programdır ve küresel kapsam. Global kapsama sahip değişken isimleri - denir genel değişkenler - en azından bazı dillerde, kötü modülerlik ve işlev kapsamı veya blok kapsamı ile birlikte isim çakışmaları ve kasıtsız maskeleme olasılığı nedeniyle sık sık kötü uygulama olarak kabul edilir. Bununla birlikte, genel kapsam genellikle (dile bağlı olarak) işlevlerin adları, işlevlerin adları gibi çeşitli diğer tür adlar için kullanılır. sınıflar ve diğerlerinin isimleri veri tipleri. Bu durumlarda aşağıdaki gibi mekanizmalar ad alanları çarpışmaları önlemek için kullanılır.

Sözcüksel kapsam ve dinamik kapsam

Yerel değişkenlerin - yalnızca belirli bir işlevde bulunan sınırlı kapsama sahip değişken adlarının - kullanımı, aynı adlandırılmış iki değişken arasında bir ad çakışması riskinin önlenmesine yardımcı olur. Ancak, bu soruyu yanıtlamak için çok farklı iki yaklaşım vardır: Bir işlevin "içinde" olmak ne anlama gelir?

İçinde sözcük kapsamı (veya sözcük kapsamı; olarak da adlandırılır statik kapsam veya statik kapsam), bir değişken adının kapsamı belirli bir işlevse, kapsamı, işlev tanımının program metnidir: bu metin içinde değişken adı vardır ve değişkenin değerine bağlıdır, ancak bu metnin dışında değişken adı yok. Aksine, dinamik kapsam (veya dinamik kapsam), bir değişken adının kapsamı belirli bir işlevse, kapsamı, işlevin çalıştığı zaman dilimidir: işlev çalışırken, değişken adı vardır ve değerine bağlıdır, ancak işlev döndükten sonra değişken adı mevcut değil. Bu, eğer işlevi f ayrı olarak tanımlanmış bir işlevi çağırır gve sözlü kapsamda işlev g yapar değil izni var fyerel değişkenler (metnini varsayarak g metninin içinde değil f), dinamik kapsam altındayken, işlev g yapar izni var fyerel değişkenleri (çünkü g çağrı sırasında çağrılır f).

$ # bash dili$ x=1$ işlevi g() { Eko $ x ; x=2 ; }$ işlevi f() { yerel x=3 ; g ; }$ f # bu 1 mi yoksa 3 mü yazdırır?3$ Eko $ x # bu 1 mi yoksa 2 mi yazdırır?1

Örneğin sağdaki programı düşünün. İlk satır, x=1, global bir değişken oluşturur x ve onu başlatır 1. İkinci satır, işlevi g() { Eko $ x ; x=2 ; }, bir işlevi tanımlar g şu anki değeri yazdırır ("yankılar") xve sonra ayarlar x -e 2 (önceki değerin üzerine yazarak). Üçüncü satır, işlevi f() { yerel x=3 ; g ; } bir işlevi tanımlar f yerel bir değişken oluşturan x (aynı adlandırılmış global değişkeni gizleyerek) ve onu 3ve sonra arar g. Dördüncü satır, f, çağrılar f. Beşinci satır, Eko $ x, geçerli değerini yazdırır x.

Peki, bu program tam olarak ne yazdırıyor? Kapsam kurallarına bağlıdır. Bu programın dili sözcük kapsamını kullanan bir dil ise, o zaman g global değişkeni yazdırır ve değiştirir x (Çünkü g dışında tanımlanır f), böylece program yazdırır 1 ve daha sonra 2. Aksine, bu dil dinamik kapsam kullanıyorsa, o zaman g yazdırır ve değiştirir fyerel değişkeni x (Çünkü g içeriden çağrılır f), böylece program yazdırır 3 ve daha sonra 1. (Olduğu gibi, programın dili Bash dinamik kapsam kullanan; böylece program yazdırır 3 ve daha sonra 1. Aynı kod ile çalıştırıldıysa ksh93 Sözcük kapsamı kullanan, sonuçlar farklı olacaktır.)

Sözcük kapsamı

İle sözcük kapsamı, bir ad her zaman sözlü bağlamına atıfta bulunur. Bu, program metninin bir özelliğidir ve çalışma zamanından bağımsız yapılır çağrı yığını dil uygulaması ile. Bu eşleştirme yalnızca statik program metninin analizini gerektirdiğinden, bu tür kapsam aynı zamanda statik kapsam. Sözcük kapsamı standarttır Algol gibi tabanlı diller Pascal, Modula-2 ve Ada gibi modern işlevsel dillerde olduğu gibi ML ve Haskell. Ayrıca, C dili ve onun sözdizimsel ve anlamsal akrabaları, ancak farklı türden sınırlamalarla. Statik kapsam, programcının basit ad ikameleri olarak parametreler, değişkenler, sabitler, türler, işlevler vb. Gibi nesne referansları hakkında akıl yürütmesini sağlar. Bu, yerel adlandırma yapısı tek başına anlaşılabildiğinden, modüler kod ve bununla ilgili mantık yürütmeyi çok daha kolaylaştırır. Aksine, dinamik kapsam, programcıyı modül kodunun çağrılabileceği tüm olası yürütme bağlamlarını tahmin etmeye zorlar.

program Bir;var ben:tamsayı;    K:kömür;    prosedür B;    var K:gerçek;        L:tamsayı;        prosedür C;        var M:gerçek;        başla         (* kapsam A + B + C *)        son;     (* kapsam A + B *)    son; (* kapsam A *)son.

Örneğin, Pascal sözcüksel olarak kapsamlıdır. Sağdaki Pascal program parçasını düşünün. Değişken ben aynı isimli başka bir değişken tarafından asla gizlenmediği için her noktada görülebilir. kömür değişken K yalnızca ana programda görülebilir çünkü gerçek değişken K prosedürde görünür B ve C sadece. Değişken L sadece prosedürde de görülebilir B ve C ancak başka herhangi bir değişkeni gizlemez. Değişken M sadece prosedürde görülebilir C ve bu nedenle prosedürden erişilemez B veya ana program. Ayrıca prosedür C sadece prosedürde görülebilir B ve bu nedenle ana programdan çağrılamaz.

Başka bir prosedür olabilirdi C programda prosedür dışında beyan edildi B. Programdaki yer "C"bahsedildiğinde, iki prosedürden hangisinin adlandırıldığını belirler C temsil eder, dolayısıyla değişkenlerin kapsamıyla tam olarak benzerdir.

Sözcük kapsamının dillerde doğru uygulanması birinci sınıf yuvalanmış işlevler her bir işlev değerinin bağlı olduğu değişkenlerin değerlerinin bir kaydını taşımasını gerektirdiğinden önemsiz değildir (işlev ve bu bağlam çiftine bir kapatma ). Uygulamaya bağlı olarak ve bilgisayar Mimarisi, değişken bakmak Mayıs biraz verimsizleşmek[kaynak belirtilmeli ] sözcüksel olarak çok derinden yuvalanmış Bunu hafifletmek için iyi bilinen teknikler olmasına rağmen işlevler kullanılır.[7][8] Ayrıca, yalnızca kendi bağımsız değişkenlerine ve (hemen) yerel değişkenlere atıfta bulunan iç içe geçmiş işlevler için, tüm göreli konumlar şu adresten bilinebilir: Derleme zamanı. Bu nedenle, bu tür iç içe geçmiş işlevi kullanırken hiçbir ek yük oluşmaz. Aynısı, iç içe geçmiş işlevlerin kullanılmadığı bir programın belirli bölümleri için ve doğal olarak, yuvalanmış işlevlerin kullanılamadığı bir dilde yazılmış programlar için de geçerlidir (C dili gibi).

Tarih

Zorunlu dil için sözlüksel kapsam kullanıldı ALGOL 60 ve o zamandan beri diğer zorunlu dillerin çoğunda seçildi.[4]

Gibi diller Pascal ve C her ikisi de içine giren fikirlerden etkilendikleri için her zaman sözcük kapsamı olmuştur. ALGOL 60 ve ALGOL 68 (C sözcüksel olarak içermese de yuvalanmış işlevler ).

Perl Daha sonra statik kapsam ekleyen dinamik kapsama sahip bir dildir.

Orijinal Lisp tercüman (1960) dinamik kapsam kullandı. Derin ciltlemeStatik (sözcüksel) kapsama yaklaşan, LISP 1.5'te ( Funarg tarafından geliştirilen cihaz Steve Russell altında çalışmak John McCarthy ).

Hepsi erken Lisps en azından tercümanlara dayalı olduğunda dinamik kapsam kullandı. 1982'de Guy L. Steele Jr. ve Common LISP Group, Ortak LISP'ye genel bakış,[9] Lisp'in geçmişinin ve o ana kadarki farklı uygulamalarının kısa bir incelemesi ve Ortak Lisp uygulama olmalıdır. 102. sayfada şunları okuyoruz:

Çoğu LISP uygulaması dahili olarak tutarsızdır, çünkü yorumlayıcı ve derleyici varsayılan olarak doğru programlara farklı anlamlar atayabilir; bu öncelikle yorumlayıcının tüm değişkenlerin dinamik olarak kapsama alınacağını varsaymasından kaynaklanırken, derleyici aksini varsaymak zorunda kalmadıkça tüm değişkenlerin yerel olduğunu varsayar. Bu, kolaylık ve verimlilik uğruna yapılmıştır, ancak çok ince hatalara yol açabilir. Ortak LISP tanımı, yorumlayıcı ve derleyicinin doğru programlara aynı semantiği empoze etmesini açıkça gerektirerek bu tür anormallikleri önler.

Bu nedenle, Ortak LISP uygulamalarının, sözcük kapsamı. Yine Ortak LISP'ye genel bakış:

Ek olarak, Common LISP (çoğu MacLisp, InterLisp veya Lisp Machines Lisp'den ödünç alınmıştır) aşağıdaki olanakları sunar: (...) Tamamen sözcüksel kapsamlı değişkenler. Sözde "FUNARG sorunu"[10][11] hem aşağı hem de yukarı durumlarda tamamen çözüldü.

Aynı yıl içinde Ortak LISP'ye genel bakış (1982), derlenmiş, sözcük kapsamına sahip bir Lisp'in ilk tasarımları (yine Guy L. Steele Jr. tarafından) yayınlandı. Şema yayınlandı ve derleyici uygulamaları deneniyordu. At that time, lexical scope in Lisp was commonly feared to be inefficient to implement. İçinde A History of T,[12] Olin Shivers writes:

Herşey serious Lisps in production use at that time were dynamically scoped. No one who hadn't carefully read the Rabbit[13] thesis (written by Guy Lewis Steele Jr. in 1978) believed lexical scope would fly; even the few people who vardı read it were taking a bit of a leap of faith that this was going to work in serious production use.

The term "lexical scope" dates at least to 1967,[14] while the term "lexical scoping" dates at least to 1970, where it was used in Proje MAC to describe the scope rules of the Lisp dialect MDL (then known as "Muddle").[15]

Dynamic scope

İle dinamik kapsam, a name refers to execution context. It is uncommon in modern languages.[4] In technical terms, this means that each name has a global yığın of bindings. Introducing a local variable with name x pushes a binding onto the global x stack (which may have been empty), which is popped off when the kontrol akışı leaves the scope. Değerlendirme x in any context always yields the top binding. Note that this cannot be done at compile-time because the binding stack only exists at Çalışma süresi, which is why this type of scope is called dinamik dürbün.

Generally, certain bloklar are defined to create bindings whose lifetime is the execution time of the block; this adds some features of static scope to the dynamic scope process. However, since a section of code can be called from many different locations and situations, it can be difficult to determine at the outset what bindings will apply when a variable is used (or if one exists at all). This can be beneficial; uygulaması principle of least knowledge suggests that code avoid depending on the nedenleri for (or circumstances of) a variable's value, but simply use the value according to the variable's definition. This narrow interpretation of shared data can provide a very flexible system for adapting the behavior of a function to the current state (or policy) of the system. However, this benefit relies on careful documentation of all variables used this way as well as on careful avoidance of assumptions about a variable's behavior, and does not provide any mechanism to detect interference between different parts of a program. Some languages, like Perl ve Ortak Lisp, allow the programmer to choose static or dynamic scope when defining or redefining a variable. Examples of languages that use dynamic scope include Logo, Emacs Lisp, Lateks and the shell languages bash, kısa çizgi, ve Güç kalkanı.

Dynamic scope is fairly easy to implement. To find an name's value, the program could traverse the runtime stack, checking each activation record (each function's stack frame) for a value for the name. In practice, this is made more efficient via the use of an association list, which is a stack of name/value pairs. Pairs are pushed onto this stack whenever declarations are made, and popped whenever variables go out of context.[16] Shallow binding is an alternative strategy that is considerably faster, making use of a central reference table, which associates each name with its own stack of meanings. This avoids a linear search during run-time to find a particular name, but care should be taken to properly maintain this table.[16] Note that both of these strategies assume a last-in-first-out (LIFO ) ordering to bindings for any one variable; in practice all bindings are so ordered.

An even simpler implementation is the representation of dynamic variables with simple global variables. The local binding is performed by saving the original value in an anonymous location on the stack that is invisible to the program. When that binding scope terminates, the original value is restored from this location. In fact, dynamic scope originated in this manner. Early implementations of Lisp used this obvious strategy for implementing local variables, and the practice survives in some dialects which are still in use, such as GNU Emacs Lisp. Lexical scope was introduced into Lisp later. This is equivalent to the above shallow binding scheme, except that the central reference table is simply the global variable binding context, in which the current meaning of the variable is its global value. Maintaining global variables isn't complex. For instance, a symbol object can have a dedicated slot for its global value.

Dynamic scope provides an excellent abstraction for thread local storage, but if it is used that way it cannot be based on saving and restoring a global variable. A possible implementation strategy is for each variable to have a thread-local key. When the variable is accessed, the thread-local key is used to access the thread-local memory location (by code generated by the compiler, which knows which variables are dynamic and which are lexical). If the thread-local key does not exist for the calling thread, then the global location is used. When a variable is locally bound, the prior value is stored in a hidden location on the stack. The thread-local storage is created under the variable's key, and the new value is stored there. Further nested overrides of the variable within that thread simply save and restore this thread-local location. When the initial, outermost override's context terminates, the thread-local key is deleted, exposing the global version of the variable once again to that thread.

İle referential transparency the dynamic scope is restricted to the argument stack of the current function only, and coincides with the lexical scope.

Macro expansion

In modern languages, makro genişletme içinde önişlemci is a key example of de facto dynamic scope. The macro language itself only transforms the source code, without resolving names, but since the expansion is done in place, when the names in the expanded text are then resolved (notably free variables), they are resolved based on where they are expanded (loosely "called"), as if dynamic scope were occurring.

C ön işlemcisi, için kullanılır makro genişletme, has de facto dynamic scope, as it does not do name resolution by itself. For example, the macro:

#define ADD_A(x) x + a

will expand to add a to the passed variable, with this name only later resolved by the compiler based on where the macro ADD_A is "called" (properly, expanded), is in dynamic scope, and is independent of where the macro is defined. Properly, the C preprocessor only does sözcük analizi, expanding the macro during the tokenization stage, but not parsing into a syntax tree or doing name resolution.

For example, in the following code, the a in the macro is resolved (after expansion) to the local variable at the expansion site:

#define ADD_A(x) x + ageçersiz add_one(int *x) {  sabit int a = 1;  *x = ADD_A(*x);}geçersiz add_two(int *x) {  sabit int a = 2;  *x = ADD_A(*x);}

Qualified names

As we have seen, one of the key reasons for scope is that it helps prevent name collisions, by allowing identical names to refer to distinct things, with the restriction that the names must have separate scopes. Sometimes this restriction is inconvenient; when many different things need to be accessible throughout a program, they generally all need names with global scope, so different techniques are required to avoid name collisions.

To address this, many languages offer mechanisms for organizing global names. The details of these mechanisms, and the terms used, depend on the language; but the general idea is that a group of names can itself be given a name — a prefix — and, when necessary, an entity can be referred to by a qualified name consisting of the name plus the prefix. Normally such names will have, in a sense, two sets of scopes: a scope (usually the global scope) in which the qualified name is visible, and one or more narrower scopes in which the unqualified name (without the prefix) is visible as well. And normally these groups can themselves be organized into groups; that is, they can be yuvalanmış.

Although many languages support this concept, the details vary greatly. Some languages have mechanisms, such as ad alanları içinde C ++ ve C #, that serve almost exclusively to enable global names to be organized into groups. Other languages have mechanisms, such as paketleri içinde Ada ve yapılar içinde Standart ML, that combine this with the additional purpose of allowing some names to be visible only to other members of their group. And object-oriented languages often allow classes or singleton objects to fulfill this purpose (whether or not they Ayrıca have a mechanism for which this is the primary purpose). Furthermore, languages often meld these approaches; Örneğin, Perl 's packages are largely similar to C++'s namespaces, but optionally double as classes for object-oriented programming; ve Java organizes its variables and functions into classes, but then organizes those classes into Ada-like packages.

Dile göre

Scope rules for representative languages follow.

C

In C, scope is traditionally known as bağlantı veya görünürlük, particularly for variables. C is a lexically scoped language with global scope (known as external linkage), a form of module scope or file scope (known as internal linkage), and local scope (within a function); within a function scopes can further be nested via block scope. However, standard C does not support nested functions.

The lifetime and visibility of a variable are determined by its storage class. There are three types of lifetimes in C: static (program execution), automatic (block execution, allocated on the stack), and manual (allocated on the heap). Only static and automatic are supported for variables and handled by the compiler, while manually allocated memory must be tracked manually across different variables. There are three levels of visibility in C: external linkage (global), internal linkage (roughly file), and block scope (which includes functions); block scopes can be nested, and different levels of internal linkage is possible by use of includes. Internal linkage in C is visibility at the translation unit level, namely a source file after being processed by the C ön işlemcisi, notably including all relevant includes.

C programs are compiled as separate nesne dosyaları, which are then linked into an executable or library via a bağlayıcı. Thus name resolution is split across the compiler, which resolves names within a translation unit (more loosely, "compilation unit", but this is properly a different concept), and the linker, which resolves names across translation units; görmek bağlantı for further discussion.

In C, variables with block scope enter context when they are declared (not at the top of the block), go out of context if any (non-nested) function is called within the block, come back into context when the function returns, and go out of context at the end of the block. In the case of automatic local variables, they are also allocated on declaration and deallocated at the end of the block, while for static local variables, they are allocated at program initialization and deallocated at program termination.

The following program demonstrates a variable with block scope coming into context partway through the block, then exiting context (and in fact being deallocated) when the block ends:

#Dahil etmek <stdio.h>int ana(geçersiz) {  kömür x = 'm';  printf("%c n", x);  {    printf("%c n", x);    kömür x = 'b';    printf("%c n", x);  }  printf("%c n", x);}

The program outputs:

mmbm

There are other levels of scope in C.[17] Variable names used in a function prototype have function prototype visibility, and exit context at the end of the function prototype. Since the name is not used, this is not useful for compilation, but may be useful for documentation. Label names for GOTO statement have function scope, while case label names for deyimleri değiştir have block scope (the block of the switch).

C ++

All the variables that we intend to use in a program must have been declared with its type specifier in an earlierpoint in the code, like we did in the previous code at the beginning of the body of the function main when wedeclared that a, b, and result were of type int.A variable can be either of global or local scope. A global variable is a variable declared in the main body of thesource code, outside all functions, while a local variable is one declared within the body of a function or a block.

Modern versiyonlar izin vermek nested lexical scope.

Swift

Swift has a similar rule for scopes with C++, but contains different access modifiers.

DeğiştiriciImmediate scopeDosyaContaining module/packageDünyanın geri kalanı
açıkEvetEvetEvetYes, allows subclass
halka açıkEvetEvetEvetYes, disallows subclass
EvetEvetEvetHayır
fileprivateEvetEvetHayırHayır
özelEvetHayırHayırHayır

Git

Git is lexically scoped using blocks.[3]

Java

Java is lexically scoped.

A Java class can contain three types of variables:[18]

Local variables
are defined inside a method, or a particular block. These variables are local to where they were defined and lower levels. For example, a loop inside a method can use that method's local variables, but not the other way around. The loop's variables (local to that loop) are destroyed as soon as the loop ends.
Member variables
olarak da adlandırılır alanlar are variables declared within the class, outside of any method. By default, these variables are available for all methods within that class and also for all classes in the package.
Parametreler
are variables in method declarations.

In general, a set of brackets defines a particular scope, but variables at top level within a class can differ in their behavior depending on the modifier keywords used in their definition.The following table shows the access to members permitted by each modifier.[19]

DeğiştiriciSınıfPaket içeriğiAlt sınıfDünya
halka açıkEvetEvetEvetEvet
korumalıEvetEvetEvetHayır
(no modifier)EvetEvetHayırHayır
özelEvetHayırHayırHayır

JavaScript

JavaScript has simple scope rules,[20] but variable initialization and name resolution rules can cause problems, and the widespread use of closures for callbacks means the lexical context of a function when defined (which is used for name resolution) can be very different from the lexical context when it is called (which is irrelevant for name resolution). JavaScript objects have name resolution for properties, but this is a separate topic.

JavaScript has lexical scope [21] nested at the function level, with the global context being the outermost context. This scope is used for both variables and for functions (meaning function declarations, as opposed to variables of function type).[22] Block scope with the İzin Vermek ve sabit keywords is standard since ECMAScript 6. Block scope can be produced by wrapping the entire block in a function and then executing it; bu olarak bilinir immediately-invoked function expression (IIFE) pattern.

While JavaScript scope is simple—lexical, function-level—the associated initialization and name resolution rules are a cause of confusion. Firstly, assignment to a name not in scope defaults to creating a new global variable, not a local one. Secondly, to create a new local variable one must use the var anahtar kelime; the variable is then created at the top of the function, with value Tanımsız and the variable is assigned its value when the assignment expression is reached:

A variable with an Initialiser is assigned the value of its AssignmentExpression ne zaman VariableStatement is executed, not when the variable is created.[23]

Bu olarak bilinir variable hoisting[24]—the declaration, but not the initialization, is hoisted to the top of the function. Thirdly, accessing variables before initialization yields Tanımsız, rather than a syntax error. Fourthly, for function declarations, the declaration and the initialization are both hoisted to the top of the function, unlike for variable initialization. For example, the following code produces a dialog with output Tanımsız, as the local variable declaration is hoisted, shadowing the global variable, but the initialization is not, so the variable is undefined when used:

a = 1;işlevi f() {  uyarmak(a);  var a = 2;}f();

Further, as functions are first-class objects in JavaScript and are frequently assigned as callbacks or returned from functions, when a function is executed, the name resolution depends on where it was originally defined (the lexical context of the definition), not the lexical context or execution context where it is called. The nested scopes of a particular function (from most global to most local) in JavaScript, particularly of a closure, used as a callback, are sometimes referred to as the scope chain, by analogy with the prototype chain of an object.

Kapanışlar can be produced in JavaScript by using nested functions, as functions are first-class objects.[25] Returning a nested function from an enclosing function includes the local variables of the enclosing function as the (non-local) lexical context of the returned function, yielding a closure. Örneğin:

işlevi newCounter() {  // return a counter that is incremented on call (starting at 0)  // and which returns its new value  var a = 0;  var b = işlevi() { a++; dönüş a; };  dönüş b;}c = newCounter();uyarmak(c() + ' ' + c());  // outputs "1 2"

Closures are frequently used in JavaScript, due to being used for callbacks. Indeed, any hooking of a function in the local context as a callback or returning it from a function creates a closure if there are any unbound variables in the function body (with the context of the closure based on the nested scopes of the current lexical context, or "scope chain"); this may be accidental. When creating a callback based on parameters, the parameters must be stored in a closure, otherwise it will accidentally create a closure that refers to the variables in the enclosing context, which may change.[26]

Name resolution of properties of JavaScript objects is based on inheritance in the prototype tree—a path to the root in the tree is called a prototype chain—and is separate from name resolution of variables and functions.

Lisp

Lisp dialects have various rules for scope.

The original Lisp used dynamic scope; öyleydi Şema esinlenerek Algol, that introduced static (lexical) scope to the Lisp family.

Maclisp used dynamic scope by default in the interpreter and lexical scope by default in compiled code, though compiled code could access dynamic bindings by use of ÖZEL declarations for particular variables.[27] Ancak, Maclisp treated lexical binding more as an optimization than one would expect in modern languages, and it did not come with the kapatma feature one might expect of lexical scope in modern Lisps. A separate operation, *FUNCTION, was available to somewhat clumsily work around some of that issue.[28]

Ortak Lisp adopted lexical scope from Şema,[29] olduğu gibi Clojure.

ISLISP has lexical scope for ordinary variables. It also has dynamic variables, but they are in all cases explicitly marked; they must be defined by a defdynamic special form, bound by a dynamic-let special form, and accessed by an explicit dinamik special form.[30]

Some other dialects of Lisp, like Emacs Lisp, still use dynamic scope by default. Emacs Lisp now has lexical scope available on a per-buffer basis.[31]

Python

For variables, Python has function scope, module scope, and global scope. Names enter context at the start of a scope (function, module, or global scope), and exit context when a non-nested function is called or the scope ends. If a name is used prior to variable initialization, this raises a runtime exception. If a variable is simply accessed (not assigned to), name resolution follows the LEGB (Local, Enclosing, Global, Built-in) rule which resolves names to the narrowest relevant context. However, if a variable is assigned to, it defaults to declaring a variable whose scope starts at the start of the level (function, module, or global), not at the assignment. Both these rules can be overridden with a küresel veya yerel olmayan (in Python 3) declaration prior to use, which allows accessing global variables even if there is a masking nonlocal variable, and assigning to global or nonlocal variables.

As a simple example, a function resolves a variable to the global scope:

>>> def f():...     Yazdır(x)...>>> x = "global">>> f()küresel

Bunu not et x is defined before f is called, so no error is raised, even though it is defined after its reference in the definition of f. Lexically this is a forward reference, which is allowed in Python.

Here assignment creates a new local variable, which does not change the value of the global variable:

>>> def f():...     x = "f"...     Yazdır(x)...>>> x = "global">>> Yazdır(x)küresel>>> f()f>>> Yazdır(x)küresel

Assignment to a variable within a function causes it to be declared local to the function, hence its scope is the entire function, and thus using it prior to this assignment raises an error. This differs from C, where the scope of the local variable start at its declaration. This code raises an error:

>>> def f():...     Yazdır(x)...     x = "f"...>>> x = "global">>> f()Geri izleme (en son çağrı son):  Dosya "", line 1, içinde <module>  Dosya "", line 2, içinde fUnboundLocalError: local variable 'x' referenced before assignment

The default name resolution rules can be overridden with the küresel veya yerel olmayan (in Python 3) keywords. In the below code, the global x declaration in g anlamına gelir x resolves to the global variable. It thus can be accessed (as it has already been defined), and assignment assigns to the global variable, rather than declaring a new local variable. Note that no küresel declaration is needed in f—since it does not assign to the variable, it defaults to resolving to the global variable.

>>> def f():...     Yazdır(x)...>>> def g():...     küresel x...     Yazdır(x)...     x = "g"...>>> x = "global">>> f()küresel>>> g()küresel>>> f()g

küresel can also be used for nested functions. In addition to allowing assignment to a global variable, as in an unnested function, this can also be used to access the global variable in the presence of a nonlocal variable:

>>> def f():...     def g():...         küresel x...         Yazdır(x)...     x = "f"...     g()...>>> x = "global">>> f()küresel

For nested functions, there is also the yerel olmayan declaration, for assigning to a nonlocal variable, similar to using küresel in an unnested function:

>>> def f():...     def g():...         yerel olmayan x  # Python 3 only...         x = "g"...     x = "f"...     g()...     Yazdır(x)...>>> x = "global">>> f()g>>> Yazdır(x)küresel

R

R is a lexically scoped language, unlike other implementations of S where the values of free variables are determined by a set of global variables, while in R they are determined by the context in which the function was created.[32] The scope contexts may be accessed using a variety of features (such as parent.frame()) which can simulate the experience of dynamic scope should the programmer desire.

There is no block scope:

a <- 1{  a <- 2}İleti(a)## 2

Functions have access to scope they were created in:

a <- 1f <- işlevi() {  İleti(a)}f()## 1

Variables created or modified within a function stay there:

a <- 1f <- işlevi() {  İleti(a)  a <- 2  İleti(a)}f()## 1## 2İleti(a)## 1

Variables created or modified within a function stay there unless assignment to enclosing scope is explicitly requested:

a <- 1f <- işlevi() {  İleti(a)  a <<- 2  İleti(a)}f()## 1## 2İleti(a)## 2

Although R has lexical scope by default, function scopes can be changed:

a <- 1f <- işlevi() {  İleti(a)}my_env <- new.env()my_env$a <- 2f()## 1çevre(f) <- my_envf()## 2

Ayrıca bakınız

Notlar

  1. ^ Görmek tanım for meaning of "scope" versus "context".
  2. ^ "Dynamic scope" bases name resolution on kapsam (lifetime), not dürbün, and thus is formally inaccurate.
  3. ^ Örneğin, Jinja template engine for Python by default uses both lexical scope (for imports) and dynamic scope (for includes), and allows behavior to be specified with keywords; görmek Import Context Behavior.
  4. ^ "Name resolution" and "name binding" are largely synonymous; narrowly speaking "resolution" determines to which name a particular use of a name refers, without associating it with any meaning, as in higher-order abstract syntax, while "binding" associates the name with an actual meaning. In practice the terms are used interchangeably.
  5. ^ İçin kendi kendini değiştiren kod the lexical context itself can change during run time.
  6. ^ By contrast, *"a name binding's context", *"a name binding coming into scope" or *"a name binding going out of scope" are all incorrect—a name binding has scope, while a part of a program has context.

Referanslar

  1. ^ "Report on the Algorithmic Language Algol 60", 2.7. Quantities, kinds and scopes
  2. ^ WG14 N1256 (2007 updated version of the C99 standard), 6.2.1 Scopes of identifiers, 2007-09-07
  3. ^ a b Go Programlama Dili Spesifikasyonu: Declarations and scope, Version of Nov 13, 2013
  4. ^ a b c Borning A. CSE 341 -- Lexical and Dynamic Scoping. Washington Üniversitesi.
  5. ^ Crockford, Douglas. "Code Conventions for the JavaScript Programming Language". Alındı 2015-01-04.
  6. ^ Backus, J. W.; Wegstein, J. H .; Van Wijngaarden, A.; Woodger, M.; Bauer, F. L.; Green, J .; Katz, C .; McCarthy, J .; Perlis, A. J.; Rutishauser, H .; Samelson, K .; Vauquois, B. (1960). "Algoritmik dil ALGOL 60 hakkında rapor verin". ACM'nin iletişimi. 3 (5): 299. doi:10.1145/367236.367262. S2CID  278290.
  7. ^ "Programming Language Pragmatics ", LeBlank-Cook symbol table
  8. ^ "A Symbol Table Abstraction to Implement Languages with Explicit Scope Control ", LeBlank-Cook, 1983
  9. ^ Louis Steele, Guy (August 1982). "An overview of Common LISP". LFP '82: Proceedings of the 1982 ACM Symposium on LISP and Functional Programming: 98–107. doi:10.1145/800068.802140. ISBN  0897910826. S2CID  14517358.
  10. ^ Joel, Moses (June 1970). "The Function of FUNCTION in LISP". MIT AI Memo 199. MIT Artificial Intelligence Lab.
  11. ^ Steele, Guy Lewis Jr.; Sussman, Gerald Jay (May 1978). "The Art of the Interpreter; or, The Modularity Complex (Parts Zero, One and Two)". MIT AI Memo 453. MIT Artificial Intelligence Lab.
  12. ^ Shivers, Olin. "History of T". Paul Graham. Alındı 5 Şubat 2020.
  13. ^ Steele, Guy Lewis Jr. (May 1978). "RABBIT: A Compiler for SCHEME". MIT. hdl:1721.1/6913. Alıntı dergisi gerektirir | günlük = (Yardım)
  14. ^ "sözcük kapsamı ", Computer and Program Organization, Part 3, s. 18, içinde Google Kitapları, University of Michigan. Engineering Summer Conferences, 1967
  15. ^ "sözcük kapsamı ", Project MAC Progress Report, Volume 8, s. 80, at Google Kitapları, 1970.
  16. ^ a b Scott 2009, 3.4 Implementing Scope, p. 143.
  17. ^ "Dürbün ", XL C/C++ V8.0 for Linux, IBM
  18. ^ "Declaring Member Variables (The Java™ Tutorials > Learning the Java Language > Classes and Objects)". docs.oracle.com. Alındı 19 Mart 2018.
  19. ^ "Controlling Access to Members of a Class (The Java™ Tutorials > Learning the Java Language > Classes and Objects)". docs.oracle.com. Alındı 19 Mart 2018.
  20. ^ "Everything you need to know about Javascript variable scope ", Saurab Parakh, Coding is Cool, 2010-02-08
  21. ^ "Annotated ES5". es5.github.io. Alındı 19 Mart 2018.
  22. ^ "Fonksiyonlar". MDN Web Belgeleri. Alındı 19 Mart 2018.
  23. ^ "12.2 Variable Statement ", Annotated ECMAScript 5.1, Last updated: 2012-05-28
  24. ^ "JavaScript Scoping and Hoisting ", Ben Cherry, Adequately Good, 2010-02-08
  25. ^ Javascript Closures, Richard Cornford. Mart 2004
  26. ^ "Explaining JavaScript Scope And Closures ", Robert Nyman, October 9, 2008
  27. ^ Pitman, Kent (16 Aralık 2007). "Gözden Geçirilmiş Maclisp Kılavuzu (Pitmanual), Sunday Morning Edition". MACLISP.info. HyperMeta Inc. Declarations and the Compiler, Concept "Variables". Alındı 20 Ekim 2018. If the variable to be bound has been declared to be special, the binding is compiled as code to imitate the way the interpreter binds variables
  28. ^ Pitman, Kent (16 Aralık 2007). "Gözden Geçirilmiş Maclisp Kılavuzu (Pitmanual), Sunday Morning Edition". MACLISP.info. HyperMeta Inc. The Evaluator, Special Form *FUNCTION. Alındı 20 Ekim 2018. *FUNCTION is intended to help solve the “funarg problem,” however it only works in some easy cases.
  29. ^ Pitman, Kent; et al. (webbed version of ANSI standard X3.226-1994) (1996). "Common Lisp HyperSpec". Lispworks.com. LispWorks Ltd. 1.1.2 History. Alındı 20 Ekim 2018. MacLisp improved on the Lisp 1.5 notion of special variables ... The primary influences on Common Lisp were Lisp Machine Lisp, MacLisp, NIL, S-1 Lisp, Spice Lisp, and Scheme.
  30. ^ "Programming Language ISLISP, ISLISP Working Draft 23.0" (PDF). ISLISP.info. 11.1 The lexical principle. Alındı 20 Ekim 2018. Dynamic bindings are established and accessed by a separate mechanism (i.e., defdynamic, dynamic-let, ve dinamik).
  31. ^ "Lexical Binding". EmacsWiki. Alındı 20 Ekim 2018. Emacs 24 has optional lexical binding, which can be enabled on a per-buffer basis.
  32. ^ "R FAQ". cran.r-project.org. Alındı 19 Mart 2018.