1. <ul id="0c1fb"></ul>

      <noscript id="0c1fb"><video id="0c1fb"></video></noscript>
      <noscript id="0c1fb"><listing id="0c1fb"><thead id="0c1fb"></thead></listing></noscript>

      99热在线精品一区二区三区_国产伦精品一区二区三区女破破_亚洲一区二区三区无码_精品国产欧美日韩另类一区

      RELATEED CONSULTING
      相關(guān)咨詢(xún)
      選擇下列產(chǎn)品馬上在線(xiàn)溝通
      服務(wù)時(shí)間:8:30-17:00
      你可能遇到了下面的問(wèn)題
      關(guān)閉右側(cè)工具欄

      新聞中心

      這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷(xiāo)解決方案
      Buffer源碼深入分析

      本機(jī)環(huán)境:
      Linux 4.4.0-21-generic #37-Ubuntu SMP Mon Apr 18 18:33:37 UTC 2016 x86_64 x86_64 x86_64 GNU/Linux

      成都創(chuàng)新互聯(lián)公司憑借專(zhuān)業(yè)的設(shè)計(jì)團(tuán)隊(duì)扎實(shí)的技術(shù)支持、優(yōu)質(zhì)高效的服務(wù)意識(shí)和豐厚的資源優(yōu)勢(shì),提供專(zhuān)業(yè)的網(wǎng)站策劃、成都網(wǎng)站制作、網(wǎng)站設(shè)計(jì)、外貿(mào)網(wǎng)站建設(shè)、網(wǎng)站優(yōu)化、軟件開(kāi)發(fā)、網(wǎng)站改版等服務(wù),在成都十載的網(wǎng)站建設(shè)設(shè)計(jì)經(jīng)驗(yàn),為成都數(shù)千家中小型企業(yè)策劃設(shè)計(jì)了網(wǎng)站。

      Buffer

      Buffer的類(lèi)圖如下:

      Buffer源碼深入分析

      除了Boolean,其他基本數(shù)據(jù)類(lèi)型都有對(duì)應(yīng)的Buffer,但是只有ByteBuffer才能和Channel交互。只有ByteBuffer才能產(chǎn)生Direct的buffer,其他數(shù)據(jù)類(lèi)型的Buffer只能產(chǎn)生Heap類(lèi)型的Buffer。ByteBuffer可以產(chǎn)生其他數(shù)據(jù)類(lèi)型的視圖Buffer,如果ByteBuffer本身是Direct的,則產(chǎn)生的各視圖Buffer也是Direct的。

      Direct和Heap類(lèi)型Buffer的本質(zhì)

      首選說(shuō)說(shuō)JVM是怎么進(jìn)行IO操作的。

      JVM在需要通過(guò)操作系統(tǒng)調(diào)用完成IO操作,比如可以通過(guò)read系統(tǒng)調(diào)用完成文件的讀取。read的原型是:ssize_t read(int fd,void *buf,size_t nbytes),和其他的IO系統(tǒng)調(diào)用類(lèi)似,一般需要緩沖區(qū)作為其中一個(gè)參數(shù),該緩沖區(qū)要求是連續(xù)的。

      Buffer分為Direct和Heap兩類(lèi),下面分別說(shuō)明這兩類(lèi)buffer。

      Heap

      Heap類(lèi)型的Buffer存在于JVM的堆上,這部分內(nèi)存的回收與整理和普通的對(duì)象一樣。Heap類(lèi)型的Buffer對(duì)象都包含一個(gè)對(duì)應(yīng)基本數(shù)據(jù)類(lèi)型的數(shù)組屬性(比如:final **[] hb),數(shù)組才是Heap類(lèi)型Buffer的底層緩沖區(qū)。
      但是Heap類(lèi)型的Buffer不能作為緩沖區(qū)參數(shù)直接進(jìn)行系統(tǒng)調(diào)用,主要因?yàn)橄旅鎯蓚€(gè)原因。

      • JVM在GC時(shí)可能會(huì)移動(dòng)緩沖區(qū)(復(fù)制-整理),緩沖區(qū)的地址不固定。

      • 系統(tǒng)調(diào)用時(shí),緩沖區(qū)需要是連續(xù)的,但是數(shù)組可能不是連續(xù)的(JVM的實(shí)現(xiàn)沒(méi)要求連續(xù))。

      所以使用Heap類(lèi)型的Buffer進(jìn)行IO時(shí),JVM需要產(chǎn)生一個(gè)臨時(shí)Direct類(lèi)型的Buffer,然后進(jìn)行數(shù)據(jù)復(fù)制,再使用臨時(shí)Direct的Buffer作為參數(shù)進(jìn)行操作系統(tǒng)調(diào)用。這造成很低的效率,主要是因?yàn)閮蓚€(gè)原因:

      • 需要把數(shù)據(jù)從Heap類(lèi)型的Buffer里面復(fù)制到臨時(shí)創(chuàng)建的Direct的Buffer里面。

      • 可能產(chǎn)生大量的Buffer對(duì)象,從而提高GC的頻率。所以在IO操作時(shí),可以通過(guò)重復(fù)利用Buffer進(jìn)行優(yōu)化。

      Direct

      Direct類(lèi)型的buffer,不存在于堆上,而是JVM通過(guò)malloc直接分配的一段連續(xù)的內(nèi)存,這部分內(nèi)存成為直接內(nèi)存,JVM進(jìn)行IO系統(tǒng)調(diào)用時(shí)使用的是直接內(nèi)存作為緩沖區(qū)。
      -XX:MaxDirectMemorySize,通過(guò)這個(gè)配置可以設(shè)置允許分配的最大直接內(nèi)存的大小(MappedByteBuffer分配的內(nèi)存不受此配置影響)。
      直接內(nèi)存的回收和堆內(nèi)存的回收不同,如果直接內(nèi)存使用不當(dāng),很容易造成OutOfMemoryError。JAVA沒(méi)有提供顯示的方法去主動(dòng)釋放直接內(nèi)存,sun.misc.Unsafe類(lèi)可以進(jìn)行直接的底層內(nèi)存操作,通過(guò)該類(lèi)可以主動(dòng)釋放和管理直接內(nèi)存。同理,也應(yīng)該重復(fù)利用直接內(nèi)存以提高效率。

      MappedByteBuffer和DirectByteBuffer之間的關(guān)系

      This is a little bit backwards: By rights MappedByteBuffer should be a subclass of DirectByteBuffer, but to keep the spec clear and simple, and for optimization purposes, it's easier to do it the other way around.This works because DirectByteBuffer is a package-private class.(本段話(huà)摘自MappedByteBuffer的源碼)

      實(shí)際上,MappedByteBuffer屬于映射buffer(自己看看虛擬內(nèi)存),但是DirectByteBuffer只是說(shuō)明該部分內(nèi)存是JVM在直接內(nèi)存區(qū)分配的連續(xù)緩沖區(qū),并不一是映射的。也就是說(shuō)MappedByteBuffer應(yīng)該是DirectByteBuffer的子類(lèi),但是為了方便和優(yōu)化,把MappedByteBuffer作為了DirectByteBuffer的父類(lèi)。另外,雖然MappedByteBuffer在邏輯上應(yīng)該是DirectByteBuffer的子類(lèi),而且MappedByteBuffer的內(nèi)存的GC和直接內(nèi)存的GC類(lèi)似(和堆GC不同),但是分配的MappedByteBuffer的大小不受-XX:MaxDirectMemorySize參數(shù)影響。
      MappedByteBuffer封裝的是內(nèi)存映射文件操作,也就是只能進(jìn)行文件IO操作。MappedByteBuffer是根據(jù)mmap產(chǎn)生的映射緩沖區(qū),這部分緩沖區(qū)被映射到對(duì)應(yīng)的文件頁(yè)上,屬于直接內(nèi)存在用戶(hù)態(tài),通過(guò)MappedByteBuffer可以直接操作映射緩沖區(qū),而這部分緩沖區(qū)又被映射到文件頁(yè)上,操作系統(tǒng)通過(guò)對(duì)應(yīng)內(nèi)存頁(yè)的調(diào)入和調(diào)出完成文件的寫(xiě)入和寫(xiě)出。

      MappedByteBuffer

      通過(guò)FileChannel.map(MapMode mode,long position, long size)得到MappedByteBuffer,下面結(jié)合源碼說(shuō)明MappedByteBuffer的產(chǎn)生過(guò)程。

      FileChannel.map的源碼:

      public MappedByteBuffer map(MapMode mode, long position, long size)
              throws IOException    {        ensureOpen();        if (position < 0L)            throw new IllegalArgumentException("Negative position");        if (size < 0L)            throw new IllegalArgumentException("Negative size");        if (position + size < 0)            throw new IllegalArgumentException("Position + size overflow");        //最大2G
              if (size > Integer.MAX_VALUE)            throw new IllegalArgumentException("Size exceeds Integer.MAX_VALUE");        int imode = -1;        if (mode == MapMode.READ_ONLY)
                  imode = MAP_RO;        else if (mode == MapMode.READ_WRITE)
                  imode = MAP_RW;        else if (mode == MapMode.PRIVATE)
                  imode = MAP_PV;        assert (imode >= 0);        if ((mode != MapMode.READ_ONLY) && !writable)            throw new NonWritableChannelException();        if (!readable)            throw new NonReadableChannelException();        long addr = -1;        int ti = -1;        try {            begin();
                  ti = threads.add();            if (!isOpen())                return null;            //size()返回實(shí)際的文件大小
                  //如果實(shí)際文件大小不符合,則增大文件的大小,文件的大小被改變,文件增大的部分默認(rèn)設(shè)置為0。
                  if (size() < position + size) { // Extend file size
                      if (!writable) {                    throw new IOException("Channel not open for writing " +                        "- cannot extend file to required size");
                      }                int rv;                do {                   //增大文件的大小
                          rv = nd.truncate(fd, position + size);
                      } while ((rv == IOStatus.INTERRUPTED) && isOpen());
                  }            //如果要求映射的文件大小為0,則不調(diào)用操作系統(tǒng)的mmap調(diào)用,只是生成一個(gè)空間容量為0的DirectByteBuffer
                  //并返回
                  if (size == 0) {
                      addr = 0;                // a valid file descriptor is not required
                      FileDescriptor dummy = new FileDescriptor();                if ((!writable) || (imode == MAP_RO))                    return Util.newMappedByteBufferR(0, 0, dummy, null);                else
                          return Util.newMappedByteBuffer(0, 0, dummy, null);
                  }            //allocationGranularity的大小在我的系統(tǒng)上是4K
                  //頁(yè)對(duì)齊,pagePosition為第多少頁(yè)
                  int pagePosition = (int)(position % allocationGranularity);            //從頁(yè)的最開(kāi)始映射
                  long mapPosition = position - pagePosition;            //因?yàn)閺捻?yè)的最開(kāi)始映射,增大映射空間
                  long mapSize = size + pagePosition;            try {                // If no exception was thrown from map0, the address is valid
                      //native方法,源代碼在openjdk/jdk/src/solaris/native/sun/nio/ch/FileChannelImpl.c,
                      //參見(jiàn)下面的說(shuō)明
                      addr = map0(imode, mapPosition, mapSize);
                  } catch (OutOfMemoryError x) {                // An OutOfMemoryError may indicate that we've exhausted memory
                      // so force gc and re-attempt map
                      System.gc();                try {
                          Thread.sleep(100);
                      } catch (InterruptedException y) {
                          Thread.currentThread().interrupt();
                      }                try {
                          addr = map0(imode, mapPosition, mapSize);
                      } catch (OutOfMemoryError y) {                    // After a second OOME, fail
                          throw new IOException("Map failed", y);
                      }
                  }            // On Windows, and potentially other platforms, we need an open
                  // file descriptor for some mapping operations.
                  FileDescriptor mfd;            try {
                      mfd = nd.duplicateForMapping(fd);
                  } catch (IOException ioe) {                unmap0(addr, mapSize);                throw ioe;
                  }            assert (IOStatus.checkAll(addr));            assert (addr % allocationGranularity == 0);            int isize = (int)size;
                  Unmapper um = new Unmapper(addr, mapSize, isize, mfd);            if ((!writable) || (imode == MAP_RO)) {                return Util.newMappedByteBufferR(isize,
                                                       addr + pagePosition,
                                                       mfd,
                                                       um);
                  } else {                return Util.newMappedByteBuffer(isize,
                                                      addr + pagePosition,
                                                      mfd,
                                                      um);
                  }
              } finally {
                  threads.remove(ti);            end(IOStatus.checkAll(addr));
              }
          }

      map0的源碼實(shí)現(xiàn):

      JNIEXPORT jlong JNICALLJava_sun_nio_ch_FileChannelImpl_map0(JNIEnv *env, jobject this,
                                           jint prot, jlong off, jlong len){    void *mapAddress = 0;
          jobject fdo = (*env)->GetObjectField(env, this, chan_fd);    //linux系統(tǒng)調(diào)用是通過(guò)整型的文件id引用文件的,這里得到文件id
          jint fd = fdval(env, fdo);    int protections = 0;    int flags = 0;    if (prot == sun_nio_ch_FileChannelImpl_MAP_RO) {
              protections = PROT_READ;
              flags = MAP_SHARED;
          } else if (prot == sun_nio_ch_FileChannelImpl_MAP_RW) {
              protections = PROT_WRITE | PROT_READ;
              flags = MAP_SHARED;
          } else if (prot == sun_nio_ch_FileChannelImpl_MAP_PV) {
              protections =  PROT_WRITE | PROT_READ;
              flags = MAP_PRIVATE;
          }    //這里就是操作系統(tǒng)調(diào)用了,mmap64是宏定義,實(shí)際最后調(diào)用的是mmap
          mapAddress = mmap64(        0,                    /* Let OS decide location */
              len,                  /* Number of bytes to map */
              protections,          /* File permissions */
              flags,                /* Changes are shared */
              fd,                   /* File descriptor of mapped file */
              off);                 /* Offset into file */
      
          if (mapAddress == MAP_FAILED) {        if (errno == ENOMEM) {            //如果沒(méi)有映射成功,直接拋出OutOfMemoryError
                  JNU_ThrowOutOfMemoryError(env, "Map failed");            return IOS_THROWN;
              }        return handle(env, -1, "Map failed");
          }    return ((jlong) (unsigned long) mapAddress);
      }

      雖然FileChannel.map()的zise參數(shù)是long,但是size的大小最大為Integer.MAX_VALUE,也就是最大只能映射最大2G大小的空間。實(shí)際上操作系統(tǒng)提供的MMAP可以分配更大的空間,但是JAVA限制在2G,ByteBuffer等Buffer也最大只能分配2G大小的緩沖區(qū)。
      MappedByteBuffer是通過(guò)mmap產(chǎn)生得到的緩沖區(qū),這部分緩沖區(qū)是由操作系統(tǒng)直接創(chuàng)建和管理的,最后JVM通過(guò)unmmap讓操作系統(tǒng)直接釋放這部分內(nèi)存。

      Haep****Buffer

      下面以ByteBuffer為例,說(shuō)明Heap類(lèi)型Buffer的細(xì)節(jié)。
      該類(lèi)型的Buffer可以通過(guò)下面方式產(chǎn)生:

      • ByteBuffer.allocate(int capacity)

      • ByteBuffer.wrap(byte[] array)
        使用傳入的數(shù)組作為底層緩沖區(qū),變更數(shù)組會(huì)影響緩沖區(qū),變更緩沖區(qū)也會(huì)影響數(shù)組。

      • ByteBuffer.wrap(byte[] array,int offset, int length)
        使用傳入的數(shù)組的一部分作為底層緩沖區(qū),變更數(shù)組的對(duì)應(yīng)部分會(huì)影響緩沖區(qū),變更緩沖區(qū)也會(huì)影響數(shù)組。

      DirectByteBuffer

      DirectByteBuffer只能通過(guò)ByteBuffer.allocateDirect(int capacity) 產(chǎn)生。
      ByteBuffer.allocateDirect()源碼如下:

            public static ByteBuffer allocateDirect(int capacity) {        return new DirectByteBuffer(capacity);
          }

      DirectByteBuffer()源碼如下:

          DirectByteBuffer(int cap) {                   // package-private
      
              super(-1, 0, cap, cap);        //直接內(nèi)存是否要頁(yè)對(duì)齊,我本機(jī)測(cè)試的不用
              boolean pa = VM.isDirectMemoryPageAligned();        //頁(yè)的大小,本機(jī)測(cè)試的是4K
              int ps = Bits.pageSize();        //如果頁(yè)對(duì)齊,則size的大小是ps+cap,ps是一頁(yè),cap也是從新的一頁(yè)開(kāi)始,也就是頁(yè)對(duì)齊了
              long size = Math.max(1L, (long)cap + (pa ? ps : 0));        //JVM維護(hù)所有直接內(nèi)存的大小,如果已分配的直接內(nèi)存加上本次要分配的大小超過(guò)允許分配的直接內(nèi)存的最大值會(huì)
              //引起GC,否則允許分配并把已分配的直接內(nèi)存總量加上本次分配的大小。如果GC之后,還是超過(guò)所允許的最大值,
              //則throw new OutOfMemoryError("Direct buffer memory");
              Bits.reserveMemory(size, cap);        long base = 0;        try {           //是吧,unsafe可以直接操作底層內(nèi)存
                  base = unsafe.allocateMemory(size);
              } catch (OutOfMemoryError x) {、            //沒(méi)有分配成功,把剛剛加上的已分配的直接內(nèi)存的大小減去。
                  Bits.unreserveMemory(size, cap);            throw x;
              }        unsafe.setMemory(base, size, (byte) 0);        if (pa && (base % ps != 0)) {            // Round up to page boundary
                  address = base + ps - (base & (ps - 1));
              } else {
                  address = base;
              }
              cleaner = Cleaner.create(this, new Deallocator(base, size, cap));
              att = null;
          }

      unsafe.allocateMemory()的源碼在openjdk/src/openjdk/hotspot/src/share/vm/prims/unsafe.cpp中。具體的源碼如下:

      UNSAFE_ENTRY(jlong, Unsafe_AllocateMemory(JNIEnv *env, jobject unsafe, jlong size))
        UnsafeWrapper("Unsafe_AllocateMemory");  size_t sz = (size_t)size;  if (sz != (julong)size || size < 0) {
          THROW_0(vmSymbols::java_lang_IllegalArgumentException());
        }  if (sz == 0) {    return 0;
        }
        sz = round_to(sz, HeapWordSize);  //最后調(diào)用的是 u_char* ptr = (u_char*)::malloc(size + space_before + space_after),也就是malloc。
        void* x = os::malloc(sz, mtInternal);  if (x == NULL) {
          THROW_0(vmSymbols::java_lang_OutOfMemoryError());
        }  //Copy::fill_to_words((HeapWord*)x, sz / HeapWordSize);
        return addr_to_java(x);
      UNSAFE_END

      JVM通過(guò)malloc分配得到連續(xù)的緩沖區(qū),這部分緩沖區(qū)可以直接作為緩沖區(qū)參數(shù)進(jìn)行操作系統(tǒng)調(diào)用。


      名稱(chēng)欄目:Buffer源碼深入分析
      文章網(wǎng)址:http://ef60e0e.cn/article/poggdh.html
      99热在线精品一区二区三区_国产伦精品一区二区三区女破破_亚洲一区二区三区无码_精品国产欧美日韩另类一区
      1. <ul id="0c1fb"></ul>

        <noscript id="0c1fb"><video id="0c1fb"></video></noscript>
        <noscript id="0c1fb"><listing id="0c1fb"><thead id="0c1fb"></thead></listing></noscript>

        五家渠市| 北碚区| 临江市| 义乌市| 洮南市| 朔州市| 苏尼特左旗| 民权县| 榆树市| 聂荣县| 韶关市| 环江| 新干县| 金川县| 壶关县| 铜陵市| 滨海县| 交口县| 且末县| 中江县| 华容县| 晋州市| 弋阳县| 剑阁县| 同心县| 日照市| 太仓市| 徐水县| 蓝山县| 确山县| 兴仁县| 闽侯县| 福安市| 巴中市| 安仁县| 西青区| 新蔡县| 兴文县| 巴彦淖尔市| 沛县| 渭源县|