关于java和c++中的i++

  看到一个题目,大概是i=i++之类。

  乍一看很简单,都知道i++的先参与计算再自增,

  所以i不变。

  但是仔细一想,

  如果i不变,

  那么在赋值之后,

  自增的i,

  增加的是谁呢?

  逻辑顺序是怎样的呢?

//1.i=i++是这样的吗?;
int i=0,temp;//此处temp代表副本
i=0;
temp=i+1;//根据返回的值仍然为0判断,副本自增长,过了作用域,被GC回收了。

//2.还是这样的?
i=0;
i/*(此时i=1)*/=0/*(在执行i++过程中,改变了i的值,i=i+1,然而,i++返回运算前的副本,所以之前的自增被覆盖掉了)*/;

  查看了相关的文章关于i++的底层实现原理

  其中i=i++的描述是这样的

_temp = i;  

i = i + 1;  

i = _temp;

  

  符合第二种猜想。

  仔细想想,

  使用i++时,

  会直接影响i的值,

  所以不太可能对副本进行自增。

  i++的逻辑顺序应该是获取i,复制i,对i进行自增,返回i的副本。

public int add(int i){
        int temp=i;
        i=i+1;
        return temp;
    }

在查看这篇文章时,

  看到了另外一篇基于C++自增的文章,

++i和i++ 以及其底层实现

  其中,有这样一段代码,

#include <iostream>
using namespace std;  

int main()
{
    int i = 100;
    i += (i++);
    cout << i << endl;  

    i = 100;
    i += (++i);
    cout << i << endl;  

    return 0;
}

  

  输出结果为:201

        202

  原因大概是因为自增时,i的值被改变了。

  开始也没多想,

  只是试着去java中运行一下,

  心里理所当然的认为结果应该是一致的。

public class ForTest {
	public int add(int i){
		int temp=i;
		i=i+1;
		return temp;
	}
	public static void main(String[] args) {
		int i=100;
		/*
		 * i=i+(i++);
		 * i(100)=100+(i++(100));
		 *
		 */
		i+=(i++);
		System.out.println(i);

		i=100;
		i+=(++i);
		System.out.println(i);

		i=100;
		/* 前一个i++返回i自增前的值,但是i的值被改变,i=101.
		 * 后一个i++也返回自增前的值,但是i此时的值已经是101,
		 * 所以返回101,其后i自增为102.
		 * 最后运算并赋值
		 * i=100+101;
		 * tips:括号不影响i++的优先级,仍然先参与计算,再进行赋值;
		 * */
		i=(i++)+(i++);
		System.out.println(i);
	}
}

  结果是:200

      201

      201

  疑问:为什么会不同?

  思考:java和c++中,i的自增都是返回一个副本,

     jvm是用c和c++编写的,底层机制应该相同

     java的编译器是用java写的,编译的不同造成了不同结果。

c++中,i+=(++i)

i += 1; //100+1 = 101
i += i; //101+101 = 202

java中,i+=(++i)

i(1)=100;
i(2)=101;
i(101)=100+101;

    在c++中,运算时,直接从i相应的栈中取i,所以被改变的i马上被使用。

    在java中,运算时,虽然i被改变了,但是仍然作用不到之前的i,即在运算时,从上向下,从左向右,翻译成字节码。

    这又涉及到了所谓的编译型语言和解释型语言,

    参考了虚拟机随谈(一):解释器,树遍历解释器,基于栈与基于寄存器,大杂烩

       

时间: 2024-11-29 03:28:02

关于java和c++中的i++的相关文章

在java的GUI中暂停、继续其他线程的方法

在java GUI中,如果想通过某个按钮或其他用户可交互控控制件主线程之外的线程的暂停和继续,怎么办呢?这是我想到的一个办法. 这里提供一个java API文档链接:http://docs.oracle.com/javase/8/docs/api/index.html.参考里面的java.lang包中的Thread类,此类的suspend不推荐,用着方法是不安全的.其实每个对象都有一个wait()反法,我们就用这个方法. 自己建立一个类继承Thread类,里面定义两个变量: 1 public O

在java web项目中集成webservice

公司要求在项目中加入webservice服务,因为项目中使用了spring框架,所以在这里使用与spring兼容性较好的cxf来实现 cxf所需jar包 spring的jar包就不贴了 一:创建webservice服务器 1)创建一个服务接口 package com.service; import javax.jws.WebParam; import javax.jws.WebService; @WebService public interface IHelloWorld { public S

Java Web项目中缺少Java EE 6 Libraries怎么添加

具体步骤如下: 1.项目名称上点击鼠标右键,选择"Build Path-->Configure Build Path-->Java Build Path" 2.单击"Add Library..." 3.选择"User Library",单击"Next" 4.单击"User Libraries...",选择需要的Java EE 6 Libraries Java Web项目中缺少Java EE 6

c++  与  java  中的 继承

C++ 代码: #include <iostream> #include <string> using namespace std; class Parent { public: void fun() {cout<<"Parent fun"<<endl;} void fun(int a) {cout<<"Parent fun int a"<<endl;} void fun(int a, int

(第三章)Java内存模型(中)

一.volatile的内存语义 1.1 volatile的特性 理解volatile特性的一个好办法是把对volatile变量的单个读/写,看成是使用同一个锁对这些单个读/写操作做了同步.下面通过具体的示例来说明,示例代码如下: class VolatileFeaturesExample { volatile Long vl = 0L; //使用volatile声明64位的Long型变量 public void set(Long l) { vl = l; //单个volatile变量的写 } p

java实现.net中的枚举

Java 和 .net中的枚举不一样,在.net中,枚举是属于值类型的,而在java中确实引用类型的(其实就是一个特殊的类,enum默认集成java.lang.Enum类),所以在java中操作枚举类型就没有.net中那么方便了,比如:在.net中,可以很简单地给枚举赋值,也可以强制进行类型转换,非常方便,而在java中因为是引用类型,所以这样的操作是不行的,以下就是java中实现一些.net中枚举类型的功能,如下: enum Searson{ Spring("春", 2), Summ

用java网络编程中的TCP方式上传文本文件及出现的小问题

自己今天刚学java网络编程中的TCP传输,要用TCP传输文件时,自己也是遇到了一些问题,抽空把它整理了一下,供自己以后参考使用. 首先在这个程序中,我用一个客户端,一个服务端,从客户端上传一个文本文件给服务端,服务端接收数据并显示“上传成功”给客户端. 客户端: 1 import java.io.BufferedReader; 2 import java.io.FileReader; 3 import java.io.IOException; 4 import java.io.InputStr

Java Web开发中MVC设计模式简介

一.有关Java Web与MVC设计模式 学习过基本Java Web开发的人都已经了解了如何编写基本的Servlet,如何编写jsp及如何更新浏览器中显示的内容.但是我们之前自己编写的应用一般存在无条理性,对于一个小型的网站这样的编写没有任何问题,但是一但我们需要编写大型的web工程的话,我们现有的编写模式会造成web应用的可扩展性较差,而且一但出现问题不能准确的定位出问题出在哪里. Java是一门应用设计模式比较广泛的语言.目前主流提出的23种设计模式均可在Java语言编写的程序中所应用.目前

java配置ueditor中解决“未找到上传文件”错误提示

ueditor是一个功能十分强大的在线文本编辑器,但是在ssh框架中,确切的说实在struts2中由于其拦截器需要对request,session对象进行重新封装,这个过程中会把request对象中保存的一些内容清空,所以会导致ueditor的上传功能获取不到需要上传的内容导致“未找到上传文件”的错误! 参考网上资料和自己实验,最终的解决思路是,重写struts2中的一个转换的类,然后配置struts2使用我们重写的这个类.由于我们的工程中可能会有其他的上传等功能,为了不影响其他功能的时候,还需

1.java.io包中定义了多个流类型来实现输入和输出功能,

1.java.io包中定义了多个流类型来实现输入和输出功能,可以从不同的角度对其进行分 类,按功能分为:(C),如果为读取的内容进行处理后再输出,需要使用下列哪种流?(G)   A.输入流和输出流 B.字节流和字符流 C.节点流和处理流   D.File stream E.Pipe stream F.Random stream G.Filter stream