چرا باید در مورد null در جاوا یاد بگیرید؟
چون اگر به null توجه نکنید مطمئن باشید که جاوا به طرز وحشتناکی شما را به دردسر می اندازدjava.lang.NullPointerException
و درس خود را یاد می گیرید اما راه سخت را خواهید رفت. نوشتن کد مقاوم در برابر خرابی یک هنر است و تیم شما، مشتریان و کاربران از آن قدردانی خواهند کرد. طبق تجربه من، یکی از دلایل اصلی NullPointerException
عدم اطلاع از null در جاوا است. بسیاری از شما در حال حاضر با null آشنا هستید، بقیه شما می توانید چیزهای قدیمی و جدیدی در مورد کلمه کلیدی null یاد بگیرید. بیایید نکات مهمی را در مورد null در جاوا مرور کنیم یا یاد بگیریم.
آنچه در جاوا null است
همانطور که گفتم، null یک مفهوم بسیار بسیار مهم در جاوا است. در ابتدا برای نشان دادن عدم وجود چیزی مانند عدم وجود کاربر، منبع یا هر چیز دیگری اختراع شد، اما برنامه نویسان جاوا را با تعداد زیادی گیج کرده استnull pointer exception
. در این آموزش، حقایق اساسی در مورد کلمه کلیدی null در جاوا را یاد می گیریم و چند ترفند برای جلوگیری از مشکل null pointer exceptions
و به حداقل رساندن بررسی های پوچ را یاد می گیریم.
-
اول از همه، null یک کلمه کلیدی در جاوا است، درست مانند
public
،static
یاfinal
. حروف کوچک و کوچک حساس است، شما نمی توانید null را به صورت Null یا NULL بنویسید، کامپایلر آن را تشخیص نمی دهد و یک خطا ایجاد می شود.Object obj = NULL; // Not Ok Object obj1 = null //Ok
اغلب برنامه نویسانی که از زبان های برنامه نویسی دیگر سوئیچ کرده اند، با این مشکل مواجه می شوند، اما هنگام استفاده از IDE های مدرن، مشکل ناچیز می شود. این روزها، IDE هایی مانند Eclipse یا NetBeans می توانند این خطا را در حین تایپ برطرف کنند، اما در دوران Notepad، Vim و Emacs، این یک مشکل رایج بود که می توانست زمان با ارزش زیادی را از بین ببرد.
-
Так же, How каждый примитив имеет meaning по умолчанию, например, у
int
это 0, у boolean это false, null это meaning по умолчанию любых ссылочных типов, проще говоря, для всех an objectов. Так же, How при создании логической переменной ее meaning по умолчанию равно false, так и любые ссылочные переменные в Java по умолчанию будут равны null. Это истинно для всех типов переменных: переменной-члена or локальной переменной, переменной экземпляра or статической переменной, кроме того, компилятор будет ругаться, если Вы используете локальную переменную не проинициализировав ее.private static Object myObj; public static void main(String args[]){ System.out.println("What is value of myObjc : " + myObj); } What is value of myObjc : null
Это истинно How для статических, так и для не статических an objectов, How Вы можете видеть здесь, я сделал
myObj
статической ссылкой, так что я могу использовать ее непосредственно в методеmain
, который является статическим методом и не позволяет обращаться к не статическим переменным изнутри. -
Несмотря на распространенное заблуждение, null это не an object (
Object
) и ни тип. Это просто специальное meaning, которое может быть назначено любому ссылочному типу, и Вы можете привести null к любому типу, How показано ниже:String str = null; // null can be assigned to String Integer itr = null; // you can assign null to Integer also Double dbl = null; // null can also be assigned to Double String myStr = (String) null; // null can be type cast to String Integer myItr = (Integer) null; // it can also be type casted to Integer Double myDbl = (Double) null; // yes it's possible, no error
Как Вы можете видеть, приведение null к любому ссылочному типу пройдет успешно How во время компиляции, так и во время выполнения программы. В отличии от того, что многие из Вас возможно подумали, это не приведет к выбрасыванию
NullPointerException
. -
null может быть назначен только ссылочному типу, Вы не можете назначить null примитивной переменной вроде
int
,double
,float
orboolean
. Компилятор выразит Вам свое недовольство если Вы сделаете How показано ниже:int i = null; // type mismatch : cannot convert from null to int short s = null; // type mismatch : cannot convert from null to short byte b = null: // type mismatch : cannot convert from null to byte double d = null; //type mismatch : cannot convert from null to double Integer itr = null; // this is ok int j = itr; // this is also ok, but NullPointerException at runtime
Как Вы можете видеть, когда мы непосредственно присваиваем null примитиву, то получаем ошибку процесса компиляции, но, если присвоить null an objectу класса-обертки, а затем присвоить этот an object соответствующему примитивному типу, компилятор не отреагирует, но мы будем вознаграждены
null pointer exception
во время выполнения. Это происходит из-за авто упаковки (autoboxing
) в Java, и мы еще встретимся с ним в следующем пункте. -
Любой класс-обертка со meaningм null будет выбрасывать
java.lang.NullPointerException
когда Java распакует(unbox
) его в примитивную переменную. Некоторые программисты делают ошибку допуская, что авто упаковка(autoboxing
) позаботится о конвертации null в meaning по умолчанию для соответствующего примитивного типа, например, 0 дляint
, false дляboolean
и т.д., но это не верно, в чем можно убедиться ниже:Integer iAmNull = null; int i = iAmNull; // Remember - No Compilation Error
Но, когда Вы запустите данный фрагмент codeа, в консоли Вы увидите
Exception in thread "main" java.lang.NullPointerException
Это часто происходит при работе с
HashMap
иInteger key
. Выполнение codeа, показанного ниже прервется, How только Вы его запустите.import java.util.HashMap; import java.util.Map; /** * An example of Autoboxing and NullPointerExcpetion * * @author WINDOWS 8 */ public class Test { public static void main(String args[]) throws InterruptedException { Map numberAndCount = new HashMap<>(); int[] numbers = {3, 5, 7,9, 11, 13, 17, 19, 2, 3, 5, 33, 12, 5}; for(int i : numbers){ int count = numberAndCount.get(i); numberAndCount.put(i, count++); // NullPointerException here } } }
Output: Exception in thread "main" java.lang.NullPointerException at Test.main(Test.java:25)
این کد بسیار ساده و بی ضرر به نظر می رسد. شما به سادگی شمارش می کنید که چند بار یک عدد در یک آرایه ظاهر می شود، یک تکنیک کلاسیک برای یافتن موارد تکراری. توسعهدهنده مقدار شمارش شده قبلی را میگیرد، آن را یک عدد افزایش میدهد و دوباره در آن قرار میدهد
Map
.Integer
او ممکن است فکر کند که بوکس خودکار تبدیل به را انجام می دهدint
، همانطور که در هنگام فراخوانی متد انجام می شودput()
، اما فراموش می کند که اگر عدد هنوز شمارش نشده باشد، متد صفرget()
خواهد شد ، زیرا مقدار پیش فرض است.HashMap
عدد صحیحnull pointer exception
استint
. -
instanceof
اگر هر متغیر مرجع با مقدار null یا خود null به عنوان پارامتر مشخص شود، عملگر false را برمی گرداند. مثال:Integer iAmNull = null; if(iAmNull instanceof Integer){ System.out.println("iAmNull is instance of Integer"); }else{ System.out.println("iAmNull is NOT an instance of Integer"); }
Output : iAmNull is NOT an instance of Integer
این یک ویژگی مهم اپراتور است
instanceof
که آن را برای آزمایش نوع ریخته گری مفید می کند. -
میدانید که نمیتوانید یک متد غیراستاتیک را روی یک متغیر مرجع تهی فراخوانی کنید، آن را فراخوانی میکند
NullPointerException
، اما ممکن است ندانید که میتوانید یک متد استاتیک را روی یک متغیر مرجع تهی فراخوانی کنید. زیرا روش های استاتیک از اتصال استاتیک استفاده می کنند، آنها دور نمی اندازندNullPointerException
. در اینجا یک مثال است:public class Testing { public static void main(String args[]){ Testing myObject = null; myObject.iAmStaticMethod(); myObject.iAmNonStaticMethod(); } private static void iAmStaticMethod(){ System.out.println("I am static method, can be called by null reference"); } private void iAmNonStaticMethod(){ System.out.println("I am NON static method, don't date to call me by null"); } }
Output: I am static method, can be called by null reference Exception in thread "main" java.lang.NullPointerException at Testing.main(Testing.java:11)
-
شما می توانید null را به عنوان پارامتر به روشی که هر نوع مرجعی را می پذیرد، به عنوان مثال:
public void print(Object obj)
را می توان مانند نامید
print(null)
این از دیدگاه کامپایلر طبیعی است، اما رفتار بعدی کاملاً به روش بستگی دارد. یک روش بی خطر پرتاب نمی شود
NullPointerException
، بلکه به سادگی از آن خارج می شود. اگر منطق کسب و کار اجازه می دهد، توصیه می شود روش های بی خطر بنویسید. -
شما می توانید با استفاده از عملگرهای
==
(برابر) و!=
(نه برابر) null را مقایسه کنید، اما نمی توانید آن را با سایر عملگرهای حسابی یا منطقی مانند<
(کمتر از) یا>
(بزرگتر از) استفاده کنید. برخلاف SQL، جاوا مطابقnull == null
شکل زیر مقدار true را برمی گرداند:public class Test { public static void main(String args[]) throws InterruptedException { String abc = null; String cde = null; if(abc == cde){ System.out.println("null == null is true in Java"); } if(null != null){ System.out.println("null != null is false in Java"); } // classical null check if(abc == null){ // do something } // not ok, compile time error if(abc > null){ } } }
Output: null == null is true in Java
NullPointerException
، می توانید کد خود را ایمن کنید. زیرا null را می توان به عنوان یک مقدار خالی یا بدون مقدار اولیه در نظر گرفت، این اغلب یک منبع سردرگمی است، به همین دلیل مهم است که رفتار یک متد را زمانی که یک مقدار null وارد می شود، مستند کنیم. همیشه به یاد داشته باشید، null مقدار پیشفرض متغیرهای مرجع است و شما نمیتوانید با استفاده از مرجع تهی در جاوا، متدهای نمونه را فراخوانی کنید یا به متغیرهای نمونه دسترسی پیدا کنید.
GO TO FULL VERSION