在Java中的java 内存分配配问题

温馨提示!由于新浪微博认证机制调整,您的新浪微博帐号绑定已过期,请重新绑定!&&|&&
LOFTER精选
网易考拉推荐
用微信&&“扫一扫”
将文章分享到朋友圈。
用易信&&“扫一扫”
将文章分享到朋友圈。
阅读(1468)|
用微信&&“扫一扫”
将文章分享到朋友圈。
用易信&&“扫一扫”
将文章分享到朋友圈。
历史上的今天
在LOFTER的更多文章
loftPermalink:'',
id:'fks_',
blogTitle:'java中String内存分配问题',
blogAbstract:'
java中String有两种赋值方式:
1. String s = \"XXXX\";
2. String s = new String(\"XXXX\");
对于第一种,java虚拟机为它分配的内存在常量池,即:会先在常量池中找是否已经有一块内存存的是“XXXX”,如果有就让栈中的s指向它;如果没有,那就在常量池里为它分配一块存储“XXXX”。
对于第二中,是在堆内存中分配了一块内存,然后用栈中的s指向它。
下面是一个例子,能更好的理解:
public class TestString{&public static void main(String[] args){',
blogTag:'java,string,equals,javap',
blogUrl:'blog/static/',
isPublished:1,
istop:false,
modifyTime:8,
publishTime:1,
permalink:'blog/static/',
commentCount:0,
mainCommentCount:0,
recommendCount:1,
bsrk:-100,
publisherId:0,
recomBlogHome:false,
currentRecomBlog:false,
attachmentsFileIds:[],
groupInfo:{},
friendstatus:'none',
followstatus:'unFollow',
pubSucc:'',
visitorProvince:'',
visitorCity:'',
visitorNewUser:false,
postAddInfo:{},
mset:'000',
remindgoodnightblog:false,
isBlackVisitor:false,
isShowYodaoAd:false,
hostIntro:'',
hmcon:'1',
selfRecomBlogCount:'0',
lofter_single:''
{list a as x}
{if x.moveFrom=='wap'}
{elseif x.moveFrom=='iphone'}
{elseif x.moveFrom=='android'}
{elseif x.moveFrom=='mobile'}
${a.selfIntro|escape}{if great260}${suplement}{/if}
{list a as x}
推荐过这篇日志的人:
{list a as x}
{if !!b&&b.length>0}
他们还推荐了:
{list b as y}
转载记录:
{list d as x}
{list a as x}
{list a as x}
{list a as x}
{list a as x}
{if x_index>4}{break}{/if}
${fn2(x.publishTime,'yyyy-MM-dd HH:mm:ss')}
{list a as x}
{if !!(blogDetail.preBlogPermalink)}
{if !!(blogDetail.nextBlogPermalink)}
{list a as x}
{if defined('newslist')&&newslist.length>0}
{list newslist as x}
{if x_index>7}{break}{/if}
{list a as x}
{var first_option =}
{list x.voteDetailList as voteToOption}
{if voteToOption==1}
{if first_option==false},{/if}&&“${b[voteToOption_index]}”&&
{if (x.role!="-1") },“我是${c[x.role]}”&&{/if}
&&&&&&&&${fn1(x.voteTime)}
{if x.userName==''}{/if}
网易公司版权所有&&
{list x.l as y}
{if defined('wl')}
{list wl as x}{/list}&>&&>&&>&&>&Windows下Java进程内存分配问题汇总
Windows下Java进程内存分配问题汇总
上传大小:42KB
该资料很详细的讲述了在Windows环境下Java进程的内存分配问题,包括内核空间和用户空间内存分配,为什么Java堆的实际最大内存分配为1.5G,为什么需要同时配置xms和xmx以及如何扩大用户空间等。
1. How is the space in a basic Win32 process allocated ?
The 32-bit Windows secret !
Structure of a standard Win32 process
2. What are the Win32 java process memory requirements ?
Java Virtual Machine
JIT compiled code
Java threads
Hitting the buffers
3. Can we extend the Win-32 process space ?...展开收缩
嵌到我的页面
<input type="text" value="">
综合评分:3.8(5位用户评分)
所需积分:3
下载次数:14
审核通过送C币
创建者:qw
创建者:zhangguo5
课程推荐相关知识库
上传者其他资源上传者专辑
开发技术热门标签
VIP会员动态
您因违反CSDN下载频道规则而被锁定帐户,如有疑问,请联络:!
android服务器底层网络模块的设计方法
所需积分:0
剩余积分:720
您当前C币:0
可兑换下载积分:0
兑换下载分:
兑换失败,您当前C币不够,请先充值C币
消耗C币:0
你当前的下载分为234。
Windows下Java进程内存分配问题汇总
会员到期时间:
剩余下载次数:
你还不是VIP会员
开通VIP会员权限,免积分下载
你下载资源过于频繁,请输入验证码
您因违反CSDN下载频道规则而被锁定帐户,如有疑问,请联络:!
若举报审核通过,可奖励20下载分
被举报人:
freshman_2007
举报的资源分:
请选择类型
资源无法下载
资源无法使用
标题与实际内容不符
含有危害国家安全内容
含有反动色情等内容
含广告内容
版权问题,侵犯个人或公司的版权
*详细原因:javase(40)
本文可作为北京尚学堂java课程的学习笔记。
看下面这段代码。
class BirthDate {
public BirthDate(int d, int m, int y) {
//省略get set
public void display() {
System.out.println
(day + & - & + month + & - & + year);
public class Test{
public static void main(String args[]){
Test test = new Test();
int date = 9;
BirthDate d1= new BirthDate(7,7,1970);
BirthDate d2= new BirthDate(1,1,2000);
test.change1(date);
test.change2(d1);
test.change3(d2);
System.out.println(&date=& + date);
d1.display();
d2.display();
public void change1(int i){
public void change2(BirthDate b) {
b = new BirthDate(22,2,2004);
public void change3(BirthDate b) {
b.setDay(22);
7 - 7 - 1970
22 - 1 - 2000
我自己不明白的是change2这个方法,它竟然没有改变d1的&#20540;!
其实我们想想也就明白了,在change2运行的时候,栈内存中又多了一块区域,存放局部变量b。在change2运行时,b首先指向实参d1的位置。也就是7-7-1970 而后new 了一个新的birthday,假定它在堆内存的地址为5421 那么b的&#20540;改成了5421 等change2这个方法结束的时候,b这块内存也就消失了。d1自然什么都不变喽。
再看change3
运行这个方法的时候,b首先指向的也是实参d2的位置。我们直接通过b修改了那块内存的数据,那么d2这个变量的&#20540;自然也就改变了。
&&相关文章推荐
参考知识库
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
访问:215922次
积分:4630
积分:4630
排名:第5666名
原创:243篇
转载:16篇
评论:117条
阅读:7422
阅读:7178
文章:10篇
阅读:8952
文章:26篇
阅读:25936
(3)(4)(7)(2)(8)(14)(5)(6)(1)(15)(5)(5)(3)(9)(1)(6)(2)(13)(13)(2)(2)(10)(15)(22)(11)(14)(24)(4)(3)(4)(7)(7)(1)(1)(1)(2)(4)(6)(2)(1)(1)21715人阅读
技术文章(50)
本文将由浅入深详细介绍Java内存分配的原理,以帮助新手更轻松的学习Java。这类文章网上有很多,但大多比较零碎。本文从认知过程角度出发,将带给读者一个系统的介绍。
&&&&&&&& 进入正题前首先要知道的是Java程序运行在JVM(Java& Virtual&Machine,Java虚拟机)上,可以把JVM理解成Java程序和操作系统之间的桥梁,JVM实现了Java的平台无关性,由此可见JVM的重要性。所以在学习Java内存分配原理的时候一定要牢记这一切都是在JVM中进行的,JVM是内存分配原理的基础与前提。
简单通俗的讲,一个完整的Java程序运行过程会涉及以下内存区域:
l& 寄存器:JVM内部虚拟寄存器,存取速度非常快,程序不可控制。
l& 栈:保存局部变量的&#20540;,包括:1.用来保存基本数据类型的&#20540;;2.保存类的实例,即堆区对象的引用(指针)。也可以用来保存加载方法时的帧。
l& 堆:用来存放动态产生的数据,比如new出来的对象。注意创建出来的对象只包含属于各自的成员变量,并不包括成员方法。因为同一个类的对象拥有各自的成员变量,存储在各自的堆中,但是他们共享该类的方法,并不是每创建一个对象就把成员方法复制一次。
l& 常量池:JVM为每个已加载的类型维护一个常量池,常量池就是这个类型用到的常量的一个有序集合。包括直接常量(基本类型,String)和对其他类型、方法、字段的符号引用(1)。池中的数据和数组一样通过索引访问。由于常量池包含了一个类型所有的对其他类型、方法、字段的符号引用,所以常量池在Java的动态链接中起了核心作用。常量池存在于堆中。
l& 代码段:用来存放从硬盘上读取的源程序代码。
l& 数据段:用来存放static定义的静态成员。
下面是内存表示图:
&&&&&&&& 上图中大致描述了Java内存分配,接下来通过实例详细讲解Java程序是如何在内存中运行的(注:以下图片引用自尚学堂马士兵老师的J2SE课件,图右侧是程序代码,左侧是内存分配示意图,我会一一加上注释)。
预备知识:
1.一个Java文件,只要有main入口方法,我们就认为这是一个Java程序,可以单独编译运行。
2.无论是普通类型的变量还是引用类型的变量(俗称实例),都可以作为局部变量,他们都可以出现在栈中。只不过普通类型的变量在栈中直接保存它所对应的&#20540;,而引用类型的变量保存的是一个指向堆区的指针,通过这个指针,就可以找到这个实例在堆区对应的对象。因此,普通类型变量只在栈区占用一块内存,而引用类型变量要在栈区和堆区各占一块内存。
1.JVM自动寻找main方法,执行第一句代码,创建一个Test类的实例,在栈中分配一块内存,存放一个指向堆区对象的指针110925。
2.创建一个int型的变量date,由于是基本类型,直接在栈中存放date对应的&#20540;9。
3.创建两个BirthDate类的实例d1、d2,在栈中分别存放了对应的指针指向各自的对象。他们在实例化时调用了有参数的构造方法,因此对象中有自定义初始&#20540;。
调用test对象的change1方法,并且以date为参数。JVM读到这段代码时,检测到i是局部变量,因此会把i放在栈中,并且把date的&#20540;赋给i。
把1234赋给i。很简单的一步。
change1方法执行完毕,立即释放局部变量i所占用的栈空间。
调用test对象的change2方法,以实例d1为参数。JVM检测到change2方法中的b参数为局部变量,立即加入到栈中,由于是引用类型的变量,所以b中保存的是d1中的指针,此时b和d1指向同一个堆中的对象。在b和d1之间传递是指针。
change2方法中又实例化了一个BirthDate对象,并且赋给b。在内部执行过程是:在堆区new了一个对象,并且把该对象的指针保存在栈中的b对应空间,此时实例b不再指向实例d1所指向的对象,但是实例d1所指向的对象并无变化,这样无法对d1造成任何影响。
change2方法执行完毕,立即释放局部引用变量b所占的栈空间,注意只是释放了栈空间,堆空间要等待自动回收。
调用test实例的change3方法,以实例d2为参数。同理,JVM会在栈中为局部引用变量b分配空间,并且把d2中的指针存放在b中,此时d2和b指向同一个对象。再调用实例b的setDay方法,其实就是调用d2指向的对象的setDay方法。
调用实例b的setDay方法会影响d2,因为二者指向的是同一个对象。
&&&&&&&& change3方法执行完毕,立即释放局部引用变量b。
&&&&&&&& 以上就是Java程序运行时内存分配的大致情况。其实也没什么,掌握了思想就很简单了。无非就是两种类型的变量:基本类型和引用类型。二者作为局部变量,都放在栈中,基本类型直接在栈中保存&#20540;,引用类型只保存一个指向堆区的指针,真正的对象在堆里。作为参数时基本类型就直接传&#20540;,引用类型传指针。
&&&&&&&& 1.分清什么是实例什么是对象。Class a= new Class();此时a叫实例,而不能说a是对象。实例在栈中,对象在堆中,操作实例实际上是通过实例的指针间接操作对象。多个实例可以指向同一个对象。
&&&&&&&& 2.栈中的数据和堆中的数据销毁并不是同步的。方法一旦结束,栈中的局部变量立即销毁,但是堆中对象不一定销毁。因为可能有其他变量也指向了这个对象,直到栈中没有变量指向堆中的对象时,它才销毁,而且还不是马上销毁,要等垃圾回收扫描时才可以被销毁。
&&&&&&&& 3.以上的栈、堆、代码段、数据段等等都是相对于应用程序而言的。每一个应用程序都对应唯一的一个JVM实例,每一个JVM实例都有自己的内存区域,互不影响。并且这些内存区域是所有线程共享的。这里提到的栈和堆都是整体上的概念,这些堆栈还可以细分。
&&&&&&&& 4.类的成员变量在不同对象中各不相同,都有自己的存储空间(成员变量在堆中的对象中)。而类的方法却是该类的所有对象共享的,只有一套,对象使用方法的时候方法才被压入栈,方法不使用则不占用内存。
&&&&&&&& 以上分析只涉及了栈和堆,还有一个非常重要的内存区域:常量池,这个地方往往出现一些莫名其妙的问题。常量池是干嘛的上边已经说明了,也没必要理解多么深刻,只要记住它维护了一个已加载类的常量就可以了。接下来结合一些例子说明常量池的特性。
预备知识:
&&&&&&&& 基本类型和基本类型的包装类。基本类型有:byte、short、char、int、long、boolean。基本类型的包装类分别是:Byte、Short、Character、Integer、Long、Boolean。注意区分大小写。二者的区别是:基本类型体现在程序中是普通变量,基本类型的包装类是类,体现在程序中是引用变量。因此二者在内存中的存储位置不同:基本类型存储在栈中,而基本类型包装类存储在堆中。上边提到的这些包装类都实现了常量池技术,另外两种浮点数类型的包装类则没有实现。另外,String类型也实现了常量池技术。
public class test {
public static void main(String[] args) {
objPoolTest();
public static void objPoolTest() {
int i = 40;
int i0 = 40;
Integer i1 = 40;
Integer i2 = 40;
Integer i3 = 0;
Integer i4 = new Integer(40);
Integer i5 = new Integer(40);
Integer i6 = new Integer(0);
Double d1=1.0;
Double d2=1.0;
System.out.println(&i=i0\t& + (i == i0));
System.out.println(&i1=i2\t& + (i1 == i2));
System.out.println(&i1=i2+i3\t& + (i1 == i2 + i3));
System.out.println(&i4=i5\t& + (i4 == i5));
System.out.println(&i4=i5+i6\t& + (i4 == i5 + i6));
System.out.println(&d1=d2\t& + (d1==d2));
System.out.println();
结果分析:
1.i和i0均是普通类型(int)的变量,所以数据直接存储在栈中,而栈有一个很重要的特性:栈中的数据可以共享。当我们定义了int
i = 40;,再定义int i0 = 40;这时候会自动检查栈中是否有<span style="font-size:18 color:#这个数据,如果有,i0会直接指向i的<span style="font-size:18 color:#,不会再添加一个新的<span style="font-size:18 color:#。
2.i1和i2均是引用类型,在栈中存储指针,因为Integer是包装类。由于Integer
包装类实现了常量池技术,因此i1、i2的<span style="font-size:18 color:#均是从常量池中获取的,均指向同一个地址,因此i1=12。
3.很明显这是一个加法运算,Java的数学运算都是在栈中进行的,Java会自动对i1、i2进行拆箱操作转化成整型,因此i1在数&#20540;上等于i2&#43;i3。
4.i<span style="font-size:18 color:#和i5
均是引用类型,在栈中存储指针,因为Integer是包装类。但是由于他们各自都是new出来的,因此不再从常量池寻找数据,而是从堆中各自new一个对象,然后各自保存指向对象的指针,所以i4和i5不相等,因为他们所存指针不同,所指向对象不同。
5.这也是一个加法运算,和<span style="font-size:18 color:#同理。
6.d1和d2均是引用类型,在栈中存储指针,因为Double是包装类。但Double包装类没有实现常量池技术,因此Doubled1=1.0;相当于Double
d1=new Double(1.0);,是从堆new一个对象,d2同理。因此d1和d2存放的指针不同,指向的对象不同,所以不相等。
1.以上提到的几种基本类型包装类均实现了常量池技术,但他们维护的常量仅仅是【-128至<span style="font-size:18 color:#7】这个范围内的常量,如果常量&#20540;超过这个范围,就会从堆中创建对象,不再从常量池中取。比如,把上边例子改成Integer
i1 = 400; Integer i2 = 400;,很明显超过了<span style="font-size:18 color:#7,无法从常量池获取常量,就要从堆中new新的Integer对象,这时i1和i2就不相等了。
2.String类型也实现了常量池技术,但是稍微有点不同。String型是先检测常量池中有没有对应字符串,如果有,则取出来;如果没有,则把当前的添加进去。
凡是涉及内存原理,一般都是博大精深的领域,切勿听信一家之言,多读些文章。我在这只是浅析,里边还有很多猫腻,就留给读者探索思考了。希望本文能对大家有所帮助!
(1) 符号引用,顾名思义,就是一个符号,符号引用被使用的时候,才会解析这个符号。如果熟悉linux或unix系统的,可以把这个符号引用看作一个文件的软链接,当使用这个软连接的时候,才会真正解析它,展开它找到实际的文件
对于符号引用,在类加载层面上讨论比较多,源码级别只是一个形式上的讨论。
当一个类被加载时,该类所用到的别的类的符号引用都会保存在常量池,实际代码执行的时候,首次遇到某个别的类时,JVM会对常量池的该类的符号引用展开,转为直接引用,这样下次再遇到同样的类型时,JVM就不再解析,而直接使用这个已经被解析过的直接引用。
除了上述的类加载过程的符号引用说法,对于源码级别来说,就是依照引用的解析过程来区别代码中某些数据属于符号引用还是直接引用,如,System.out.println(&test& &#43;&abc&);//这里发生的效果相当于直接引用,而假设某个Strings
= &abc&; System.out.println(&test& &#43; s);//这里的发生的效果相当于符号引用,即把s展开解析,也就相当于s是&abc&的一个符号链接,也就是说在编译的时候,class文件并没有直接展看s,而把这个s看作一个符号,在实际的代码执行时,才会展开这个。
参考文章:
java内存分配研究:
Java常量池详解之一道比较蛋疼的面试题:
jvm常量池:
深入Java核心 Java内存分配原理精讲:
&&相关文章推荐
参考知识库
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
访问:419230次
积分:5100
积分:5100
排名:第4831名
原创:90篇
评论:370条
(1)(5)(6)(1)(8)(6)(6)(5)(5)(1)(1)(7)(4)(4)(4)(3)(25)}

我要回帖

更多关于 java内存分配机制 的文章

更多推荐

版权声明:文章内容来源于网络,版权归原作者所有,如有侵权请点击这里与我们联系,我们将及时删除。

点击添加站长微信