Java的一點內容(2)

来源:http://www.cnblogs.com/kobe-bryant/archive/2016/08/14/5771154.html
-Advertisement-
Play Games

1 面向對象的三個原則 封裝性 封裝的基本單元是類(class),類是一個抽象的邏輯結構,而類的對象是一個真實的物理實體;類的目的是封裝複雜性,在類內部存在隱藏實現複雜性機制; 封裝(encapsulation) 的兩個好處: 模塊化:可以獨立於其他對象的源代碼進行編寫和維護,可以很容易的將對象在系 ...


  • 1 面向對象的三個原則 封裝性 封裝的基本單元是類(class),類是一個抽象的邏輯結構,而類的對象是一個真實的物理實體;類的目的是封裝複雜性,在類內部存在隱藏實現複雜性機制;
    封裝(encapsulation) 的兩個好處:
    模塊化:可以獨立於其他對象的源代碼進行編寫和維護,可以很容易的將對象在系統中傳遞;
    隱藏信息:其他對象可以通過本對象的一個公共介面進行通信而不影響其他對象;
      繼承性 繼承是一個對象獲得另一個對象的屬性的過程,繼承機制是一個對象成為一個更具通用類的一個特定實例成為可能,避免了代碼的重覆編寫;
      多態性 (重載overload,方法名相同、參數的個數不同、參數的類型不同、返回的類型不同和覆蓋override) ;多態性就是“一種介面,多種方法”,可以為一組相關的動作設計一個通用的介面,其實類的函數的重載就是一種多態的體現;
    4 引入抽象編程的思想; 類的封裝就是一種抽象思想
  • Java中除了static方法和final方法(private方法本質上屬於final方法,因為不能被子類訪問)之外,其它所有的方法都是動態綁定,這意味著通常情況下,我們不必判定是否應該進行動態綁定—它會自動發生。
    • final方法會使編譯器生成更有效的代碼,這也是為什麼說聲明為final方法能在一定程度上提高性能(效果不明顯)。
    • 如果某個方法是靜態的,它的行為就不具有多態性:
      class StaticSuper {
          public static String staticGet() {
              return "Base staticGet()";
          }
      
          public String dynamicGet() {
              return "Base dynamicGet()";
          }
      }
      
      class StaticSub extends StaticSuper {
          public static String staticGet() {
              return "Derived staticGet()";
          }
      
          public String dynamicGet() {
              return "Derived dynamicGet()";
          }
      }
      
      public class StaticPolymorphism {
      
          public static void main(String[] args) {
              StaticSuper sup = new StaticSub();
              System.out.println(sup.staticGet());
              System.out.println(sup.dynamicGet());
          }
      
      }
      

      輸出:

      Base staticGet()
      Derived dynamicGet()

  • 構造函數並不具有多態性,它們實際上是static方法,只不過該static聲明是隱式的。因此,構造函數不能夠被override。

  • 在父類構造函數內部調用具有多態行為的函數將導致無法預測的結果,因為此時子類對象還沒初始化,此時調用子類方法不會得到我們想要的結果。

    class Glyph {
        void draw() {
            System.out.println("Glyph.draw()");
        }
        Glyph() {
            System.out.println("Glyph() before draw()");
            draw();
            System.out.println("Glyph() after draw()");
        }
    }
    
    class RoundGlyph extends Glyph {
        private int radius = 1;
    
        RoundGlyph(int r) {
            radius = r;
            System.out.println("RoundGlyph.RoundGlyph(). radius = " + radius);
        }
    
        void draw() {
            System.out.println("RoundGlyph.draw(). radius = " + radius);
        }
    }
    
    public class PolyConstructors {
    
        public static void main(String[] args) {
            new RoundGlyph(5);
    
        }
    
    }
    


    輸出:

    Glyph() before draw()
    RoundGlyph.draw(). radius = 0
    Glyph() after draw()
    RoundGlyph.RoundGlyph(). radius = 5

為什麼會這樣輸出?這就要明確掌握Java中構造函數的調用順序

(1)在其他任何事物發生之前,將分配給對象的存儲空間初始化成二進位0;
(2)調用基類構造函數。從根開始遞歸下去,因為多態性此時調用子類覆蓋後的draw()方法(要在調用RoundGlyph構造函數之前調用),由於步驟1的緣故,我們此時會發現radius的值為0;
(3)按聲明順序調用成員的初始化方法;
(4)最後調用子類的構造函數。

    • 只有非private方法才可以被覆蓋,但是還需要密切註意覆蓋private方法的現象,這時雖然編譯器不會報錯,但是也不會按照我們所期望的來執行,即覆蓋private方法對子類來說是一個新的方法而非重載方法。因此,在子類中,新方法名最好不要與基類的private方法採取同一名字(雖然沒關係,但容易誤解,以為能夠覆蓋基類的private方法)

    • Java類中屬性域的訪問操作都由編譯器解析,因此不是多態的。父類和子類的同名屬性都會分配不同的存儲空間,如下:

      // Direct field access is determined at compile time.
      class Super {
          public int field = 0;
          public int getField() {
              return field;
          }
      }
      
      class Sub extends Super {
          public int field = 1;
          public int getField() {
              return field;
          }
          public int getSuperField() {
              return super.field;
          }
      }
      
      public class FieldAccess {
      
          public static void main(String[] args) {
              Super sup = new Sub();
              System.out.println("sup.filed = " + sup.field + 
                      ", sup.getField() = " + sup.getField());
              Sub sub = new Sub();
              System.out.println("sub.filed = " + sub.field + 
                      ", sub.getField() = " + sub.getField() + 
                      ", sub.getSuperField() = " + sub.getSuperField());
          }
      
      }
      


      輸出:

      sup.filed = 0, sup.getField() = 1
      sub.filed = 1, sub.getField() = 1, sub.getSuperField() = 0

      Sub子類實際上包含了兩個稱為field的域,然而在引用Sub中的field時所產生的預設域並非Super版本的field域,因此為了得到Super.field,必須顯式地指明super.field。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 關於測試的必要性什麼的已經在 重構與測試 里扯過了。倒也沒必要說,寫的代碼多了自然就明白這個東西重要性。 當時說了坐等被推動去學習單元測試來著,然而等著被人推動的結果就是根本就沒人來推你。o(∩_∩)o 所以還是自己主動來學,主動來總結了。 可測試性設計基礎理論知識 可測試性設計(Design fo ...
  • 網站配置一般用來保存網站的一些設置,寫在配置文件中比寫在資料庫中要合適一下,因為配置文件本身帶有緩存,隨網站啟動讀入緩存中,速度更快,而保存在資料庫中要單獨為一條記錄創建一個表,結構不夠清晰,而且讀寫也沒有配置文件容易實現。這次要做的是網站的基本信息,數據保存在SiteConfig.config。 ... ...
  • 先上效果圖 這個效果來自於三星S5的充電界面,當然有些細節差別,主要看思路.本文目的是技術交流,不要將效果直接運用於商業產品和項目. 電池背景 因為電池內部有好幾個部分,所以本例用了一個Grid來做背景,用Clip屬性剪切出一個電池的輪廓,這樣不僅顯示出一個電池的輪廓,還可以避免水波和氣泡跑顯示Gr ...
  • 一、介簡 easyliter框架的升級版本,並且正式命名為SqliteSugar框架,另外Sugar系列還有 MySql和MsSql版本,Oracle版本待開發中(因為客戶端太大一直在憂郁當中) 用Sqlite 也不址什麼高性能了,好用為主。 經過一天的努力,Sqlite 版本所有的例子都已經測試通 ...
  • 1.為什麼使用緩存 hibernate使用緩存減少對資料庫的訪問次數,從而提升hibernate的執行效率。hibernate中有兩種類型的緩存:一級緩存和二級緩存。 2.一級緩存 Hibenate中一級緩存,也叫做session的緩存,當調用session的save/saveOrUpdate/ge ...
  • Partial 部分方法顧明思議是方法的一部分,不完整的,在ide編譯時候,會將所有部分方法載入到一起統一編譯,如果分部方法沒有被實現,編譯器就不會、對他們進行編譯。 局部類型的限制 (1) 局部類型只適用於類、介面、結構,不支持委托和枚舉。(2) 同一個類型的各個部分必須都有修飾符 partial ...
  • Duktape 的 C 語言源文件和頭文件可用於任何構建系統,只需將它們加入到你的項目中參與構建即可。 ...
  • Duktape 是一個輕量級的嵌入式 JavaScript 引擎,專註於可移植性和低占用率。 Duktape 可以被輕鬆地集成進一個 C/C++ 項目中:只需要將 duktape.c、duktape.h 和 duk_config.h 三個文件加入你的構建項目中,並使用 Duktape API 來實現... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...